mirror of
https://github.com/HeidiSQL/HeidiSQL.git
synced 2025-08-06 18:24:26 +08:00
2013 lines
79 KiB
ObjectPascal
2013 lines
79 KiB
ObjectPascal
unit VirtualTrees;
|
|
|
|
// The contents of this file are subject to the Mozilla Public License
|
|
// Version 1.1 (the "License"); you may not use this file except in compliance
|
|
// with the License. You may obtain a copy of the License at http://www.mozilla.org/MPL/
|
|
//
|
|
// Alternatively, you may redistribute this library, use and/or modify it under the terms of the
|
|
// GNU Lesser General Public License as published by the Free Software Foundation;
|
|
// either version 2.1 of the License, or (at your option) any later version.
|
|
// You may obtain a copy of the LGPL at http://www.gnu.org/copyleft/.
|
|
//
|
|
// Software distributed under the License is distributed on an "AS IS" basis,
|
|
// WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License for the
|
|
// specific language governing rights and limitations under the License.
|
|
//
|
|
// The original code is VirtualTrees.pas, released September 30, 2000.
|
|
//
|
|
// The initial developer of the original code is digital publishing AG (Munich, Germany, www.digitalpublishing.de),
|
|
// most code was written by Mike Lischke 2000-2009 (public@soft-gems.net, www.soft-gems.net)
|
|
//
|
|
// Portions created by digital publishing AG are Copyright
|
|
// (C) 1999-2001 digital publishing AG. All Rights Reserved.
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
//
|
|
// For a list of recent changes please see file CHANGES.TXT
|
|
//
|
|
// Credits for their valuable assistance and code donations go to:
|
|
// Freddy Ertl, Marian Aldenhoevel, Thomas Bogenrieder, Jim Kuenemann, Werner Lehmann, Jens Treichler,
|
|
// Paul Gallagher (IBO tree), Ondrej Kelle, Ronaldo Melo Ferraz, Heri Bender, Roland Beduerftig (BCB)
|
|
// Anthony Mills, Alexander Egorushkin (BCB), Mathias Torell (BCB), Frank van den Bergh, Vadim Sedulin, Peter Evans,
|
|
// Milan Vandrovec (BCB), Steve Moss, Joe White, David Clark, Anders Thomsen, Igor Afanasyev, Eugene Programmer,
|
|
// Corbin Dunn, Richard Pringle, Uli Gerhardt, Azza, Igor Savkic, Daniel Bauten, Timo Tegtmeier, Dmitry Zegebart,
|
|
// Andreas Hausladen, Joachim Marder, Roman Kassebaum, Vincent Parrett, Dietmar Roesler, Sanjay Kanade,
|
|
// and everyone that sent pull requests: https://github.com/Virtual-TreeView/Virtual-TreeView/pulls?q=
|
|
// Beta testers:
|
|
// Freddy Ertl, Hans-Juergen Schnorrenberg, Werner Lehmann, Jim Kueneman, Vadim Sedulin, Moritz Franckenstein,
|
|
// Wim van der Vegt, Franc v/d Westelaken
|
|
// Indirect contribution (via publicly accessible work of those persons):
|
|
// Alex Denissov, Hiroyuki Hori (MMXAsm expert)
|
|
// Documentation:
|
|
// Markus Spoettl and toolsfactory GbR (http://www.doc-o-matic.com/, sponsoring Virtual TreeView development
|
|
// with a free copy of the Doc-O-Matic help authoring system), Sven H. (Step by step tutorial)
|
|
// Source repository:
|
|
// https://github.com/Virtual-TreeView/Virtual-TreeView
|
|
// Accessability implementation:
|
|
// Marco Zehe (with help from Sebastian Modersohn)
|
|
// Port to Firemonkey:
|
|
// Karol Bieniaszewski (github user livius2)
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
interface
|
|
|
|
{$if CompilerVersion < 24}{$MESSAGE FATAL 'This version supports only RAD Studio XE3 and higher. Please use V5 from http://www.jam-software.com/virtual-treeview/VirtualTreeViewV5.5.3.zip or https://github.com/Virtual-TreeView/Virtual-TreeView/archive/V5_stable.zip'}{$ifend}
|
|
|
|
{$booleval off} // Use fastest possible boolean evaluation
|
|
|
|
// For some things to work we need code, which is classified as being unsafe for .NET.
|
|
{$WARN UNSAFE_TYPE OFF}
|
|
{$WARN UNSAFE_CAST OFF}
|
|
{$WARN UNSAFE_CODE OFF}
|
|
|
|
{$LEGACYIFEND ON}
|
|
{$WARN UNSUPPORTED_CONSTRUCT OFF}
|
|
|
|
{$HPPEMIT '#include <objidl.h>'}
|
|
{$HPPEMIT '#include <oleidl.h>'}
|
|
{$HPPEMIT '#include <oleacc.h>'}
|
|
{$ifdef BCB}
|
|
{$HPPEMIT '#pragma comment(lib, "VirtualTreesCR")'}
|
|
{$else}
|
|
{$HPPEMIT '#pragma comment(lib, "VirtualTreesR")'}
|
|
{$endif}
|
|
{$HPPEMIT '#pragma comment(lib, "Shell32")'}
|
|
{$HPPEMIT '#pragma comment(lib, "uxtheme")'}
|
|
{$HPPEMIT '#pragma link "VirtualTrees.Accessibility"'}
|
|
|
|
uses
|
|
Winapi.Windows, Winapi.Messages, Winapi.ActiveX,
|
|
System.Classes, System.SysUtils,
|
|
Vcl.Graphics, Vcl.Controls, Vcl.ImgList, Vcl.Menus, Vcl.Themes,
|
|
VirtualTrees.Types,
|
|
VirtualTrees.Header,
|
|
VirtualTrees.BaseTree,
|
|
{$IFDEF VT_FMX}
|
|
VirtualTrees.AncestorFMX,
|
|
{$ELSE}
|
|
VirtualTrees.AncestorVCL
|
|
{$ENDIF}
|
|
;
|
|
|
|
|
|
{$MinEnumSize 1, make enumerations as small as possible}
|
|
|
|
type
|
|
// Some aliases for backward compatiblity
|
|
PVirtualNode = VirtualTrees.Types.PVirtualNode;
|
|
TVirtualNode = VirtualTrees.Types.TVirtualNode;
|
|
TVTHeaderColumnLayout = VirtualTrees.Types.TVTHeaderColumnLayout;
|
|
TSmartAutoFitType = VirtualTrees.Types.TSmartAutoFitType;
|
|
TVirtualTreeStates = VirtualTrees.Types.TVirtualTreeStates;
|
|
TCheckState = VirtualTrees.Types.TCheckState;
|
|
TCheckType = VirtualTrees.Types.TCheckType;
|
|
TSortDirection = VirtualTrees.Types.TSortDirection;
|
|
TColumnIndex = VirtualTrees.Types.TColumnIndex;
|
|
TVTColumnOption = VirtualTrees.Types.TVTColumnOption;
|
|
TVTHeaderHitInfo = VirtualTrees.Types.TVTHeaderHitInfo;
|
|
TVTHeaderHitPosition = VirtualTrees.Types.TVTHeaderHitPosition;
|
|
TVTHeaderHitPositions = VirtualTrees.Types.TVTHeaderHitPositions;
|
|
THeaderState = VirtualTrees.Types.THeaderState;
|
|
THeaderStates = VirtualTrees.Types.THeaderStates;
|
|
TDropMode = VirtualTrees.Types.TDropMode;
|
|
TFormatArray = VirtualTrees.Types.TFormatArray;
|
|
TVTHeaderOption = VirtualTrees.Types.TVTHeaderOption;
|
|
TVTHeaderOptions = VirtualTrees.Types.TVTHeaderOptions;
|
|
TVTHeaderStyle = VirtualTrees.Types.TVTHeaderStyle;
|
|
TVTExportType = VirtualTrees.Types.TVTExportType;
|
|
TVTImageKind = VirtualTrees.Types.TVTImageKind;
|
|
TVTExportMode = VirtualTrees.Types.TVTExportMode;
|
|
TVTOperationKind = VirtualTrees.Types.TVTOperationKind;
|
|
TVTUpdateState = VirtualTrees.Types.TVTUpdateState;
|
|
TVTCellPaintMode = VirtualTrees.Types.TVTCellPaintMode;
|
|
TVirtualNodeState = VirtualTrees.Types.TVirtualNodeState;
|
|
TVirtualNodeInitState = VirtualTrees.Types.TVirtualNodeInitState;
|
|
TVirtualNodeInitStates = VirtualTrees.Types.TVirtualNodeInitStates;
|
|
TVTTooltipLineBreakStyle = VirtualTrees.Types.TVTTooltipLineBreakStyle;
|
|
TVTNodeAttachMode = VirtualTrees.Types.TVTNodeAttachMode;
|
|
TNodeArray = VirtualTrees.Types.TNodeArray;
|
|
THitInfo = VirtualTrees.Types.THitInfo;
|
|
THitPosition = VirtualTrees.Types.THitPosition;
|
|
TVTPaintOption = VirtualTrees.Types.TVTPaintOption;
|
|
TVTAutoOption = VirtualTrees.Types.TVTAutoOption;
|
|
TVTAutoOptions = VirtualTrees.Types.TVTAutoOptions;
|
|
TVTSelectionOption = VirtualTrees.Types.TVTSelectionOption;
|
|
TVstTextType = VirtualTrees.Types.TVstTextType;
|
|
TVTHintMode = VirtualTrees.Types.TVTHintMode;
|
|
TBaseVirtualTree = VirtualTrees.BaseTree.TBaseVirtualTree;
|
|
IVTEditLink = VirtualTrees.BaseTree.IVTEditLink;
|
|
TVTHeaderNotifyEvent = VirtualTrees.BaseTree.TVTHeaderNotifyEvent;
|
|
TVTCompareEvent = VirtualTrees.BaseTree.TVTCompareEvent;
|
|
TVirtualTreeColumn = VirtualTrees.Header.TVirtualTreeColumn;
|
|
TVirtualTreeColumns = VirtualTrees.Header.TVirtualTreeColumns;
|
|
TVTHeader = VirtualTrees.Header.TVTHeader;
|
|
TVTHeaderClass = VirtualTrees.Header.TVTHeaderClass;
|
|
THeaderPaintInfo = VirtualTrees.Header.THeaderPaintInfo;
|
|
TVTConstraintPercent = VirtualTrees.Header.TVTConstraintPercent;
|
|
TVTFixedAreaConstraints = VirtualTrees.Header.TVTFixedAreaConstraints;
|
|
TColumnsArray = VirtualTrees.Header.TColumnsArray;
|
|
TCanvas = Vcl.Graphics.TCanvas;
|
|
|
|
const
|
|
// Aliases for increased compatibility with V7, feel free to extend by pull requests
|
|
NoColumn = VirtualTrees.Types.NoColumn;
|
|
InvalidColumn = VirtualTrees.Types.InvalidColumn;
|
|
sdAscending = VirtualTrees.Types.TSortDirection.sdAscending;
|
|
sdDescending = VirtualTrees.Types.TSortDirection.sdDescending;
|
|
toAutoSort = VirtualTrees.Types.TVTAutoOption.toAutoSort;
|
|
toCheckSupport = VirtualTrees.Types.TVTMiscOption.toCheckSupport;
|
|
toEditable = VirtualTrees.Types.TVTMiscOption.toEditable;
|
|
toShowRoot = VirtualTrees.Types.TVTPaintOption.toShowRoot;
|
|
ctNone = VirtualTrees.Types.TCheckType.ctNone;
|
|
ctTriStateCheckBox = VirtualTrees.Types.TCheckType.ctTriStateCheckBox;
|
|
ctCheckBox = VirtualTrees.Types.TCheckType.ctCheckBox;
|
|
ctRadioButton = VirtualTrees.Types.TCheckType.ctRadioButton;
|
|
ctButton = VirtualTrees.Types.TCheckType.ctButton;
|
|
|
|
csUncheckedNormal = VirtualTrees.Types.TCheckState.csUncheckedNormal;
|
|
csUncheckedPressed = VirtualTrees.Types.TCheckState.csUncheckedPressed;
|
|
csCheckedNormal = VirtualTrees.Types.TCheckState.csCheckedNormal;
|
|
csCheckedPressed = VirtualTrees.Types.TCheckState.csCheckedPressed;
|
|
csMixedNormal = VirtualTrees.Types.TCheckState.csMixedNormal;
|
|
csMixedPressed = VirtualTrees.Types.TCheckState.csMixedPressed;
|
|
csUncheckedDisabled = VirtualTrees.Types.TCheckState.csUncheckedDisabled;
|
|
csCheckedDisabled = VirtualTrees.Types.TCheckState.csCheckedDisabled;
|
|
csMixedDisable = VirtualTrees.Types.TCheckState.csMixedDisabled;
|
|
|
|
coVisible = VirtualTrees.Types.TVTColumnOption.coVisible;
|
|
vsDisabled = VirtualTrees.Types.TVirtualNodeState.vsDisabled;
|
|
etHTML = VirtualTrees.Types.TVTExportType.etHTML;
|
|
hiOnItemButton = VirtualTrees.Types.THitPosition.hiOnItemButton;
|
|
dmOnNode = VirtualTrees.Types.TDropMode.dmOnNode;
|
|
hlbForceMultiLine = VirtualTrees.Types.TVTTooltipLineBreakStyle.hlbForceMultiLine;
|
|
hmHintAndDefault = VirtualTrees.Types.TVTHintMode.hmHintAndDefault;
|
|
hmTooltip = VirtualTrees.Types.TVTHintMode.hmTooltip;
|
|
|
|
type
|
|
TCustomVirtualStringTree = class;
|
|
|
|
{$IFDEF VT_FMX}
|
|
TVTAncestor = TVTAncestorFMX;
|
|
{$ELSE}
|
|
TVTAncestor = TVTAncestorVcl;
|
|
{$ENDIF}
|
|
|
|
// Describes the source to use when converting a string tree into a string for clipboard etc.
|
|
TVSTTextSourceType = (
|
|
tstAll, // All nodes are rendered. Initialization is done on the fly.
|
|
tstInitialized, // Only initialized nodes are rendered.
|
|
tstSelected, // Only selected nodes are rendered.
|
|
tstCutCopySet, // Only nodes currently marked as being in the cut/copy clipboard set are rendered.
|
|
tstVisible, // Only visible nodes are rendered.
|
|
tstChecked // Only checked nodes are rendered
|
|
);
|
|
|
|
TVSTGetTextEvent = procedure(Sender: TBaseVirtualTree; Node: PVirtualNode; Column: TColumnIndex;
|
|
TextType: TVSTTextType; var CellText: string) of object;
|
|
TVSTGetHintEvent = procedure(Sender: TBaseVirtualTree; Node: PVirtualNode; Column: TColumnIndex;
|
|
var LineBreakStyle: TVTTooltipLineBreakStyle; var HintText: string) of object;
|
|
// New text can only be set for variable caption.
|
|
TVSTNewTextEvent = procedure(Sender: TBaseVirtualTree; Node: PVirtualNode; Column: TColumnIndex;
|
|
NewText: string) of object;
|
|
/// <summary>String tree event for custom handling of string abbreviations.</summary>
|
|
/// <param name="Sender">The instance that fired the event.</param>
|
|
/// <param name="TargetCanvas">Teh canvas on that the sending control will paint.</param>
|
|
/// <param name="Node">The Node that is going to be painted.</param>
|
|
/// <param name="Column">The column index that is going to be painted.</param>
|
|
/// <param name="Result">Var parameter that contains the caption or string that should be used.</param>
|
|
/// <param name="Done">Boolean var paramter: Assign True if a string is passed in the Result parameter. Leave the default value False if no shorting is need or the control shuld do it. </param>
|
|
/// <remarks>
|
|
/// If the text of a node does not fit into its cell (in grid mode) or is too wide for the width of the tree view it is being abbreviated with an ellipsis (...). By default the ellipsis is added to the end of the node text.
|
|
/// Occasionally you may want to shorten the node text at a different position, for example if the node text is a path string and not the last folder or filename should be cut off but rather some mid level folders if possible.
|
|
/// </remarks>
|
|
TVSTShortenStringEvent = procedure(Sender: TBaseVirtualTree; TargetCanvas: TCanvas; Node: PVirtualNode;
|
|
Column: TColumnIndex; const S: string; TextSpace: TDimension; var Result: string;
|
|
var Done: Boolean) of object;
|
|
TVTMeasureTextEvent = procedure(Sender: TBaseVirtualTree; TargetCanvas: TCanvas; Node: PVirtualNode;
|
|
Column: TColumnIndex; const Text: string; var Extent: TDimension) of object;
|
|
TVTDrawTextEvent = procedure(Sender: TBaseVirtualTree; TargetCanvas: TCanvas; Node: PVirtualNode;
|
|
Column: TColumnIndex; const Text: string; const CellRect: TRect; var DefaultDraw: Boolean) of object;
|
|
|
|
/// Event arguments of the OnGetCellText event
|
|
TVSTGetCellTextEventArgs = record
|
|
Node: PVirtualNode;
|
|
Column: TColumnIndex;
|
|
CellText: string;
|
|
StaticText: string;
|
|
StaticTextAlignment: TAlignment;
|
|
ExportType: TVTExportType;
|
|
constructor Create(pNode: PVirtualNode; pColumn: TColumnIndex; pExportType: TVTExportType = TVTExportType.etNone);
|
|
end;
|
|
|
|
/// Event signature which is called when text is painted on the canvas or needed for the export.
|
|
TVSTGetCellTextEvent = procedure(Sender: TCustomVirtualStringTree; var E: TVSTGetCellTextEventArgs) of object;
|
|
|
|
TCustomVirtualStringTree = class(TVTAncestor)
|
|
private
|
|
FInternalDataOffset: Cardinal; // offset to the internal data of the string tree
|
|
FDefaultText: string; // text to show if there's no OnGetText event handler (e.g. at design time)
|
|
FTextHeight: Integer; // true size of the font
|
|
FEllipsisWidth: Integer; // width of '...' for the current font
|
|
|
|
FOnGetText: TVSTGetTextEvent; // used to retrieve the string to be displayed for a specific node
|
|
fOnGetCellText: TVSTGetCellTextEvent; // used to retrieve the normal and static text of a tree node
|
|
FOnGetHint: TVSTGetHintEvent; // used to retrieve the hint to be displayed for a specific node
|
|
FOnNewText: TVSTNewTextEvent; // used to notify the application about an edited node caption
|
|
FOnShortenString: TVSTShortenStringEvent; // used to allow the application a customized string shortage
|
|
FOnMeasureTextWidth: TVTMeasureTextEvent; // used to adjust the width of the cells
|
|
FOnMeasureTextHeight: TVTMeasureTextEvent;
|
|
FOnDrawText: TVTDrawTextEvent; // used to custom draw the node text
|
|
/// Returns True if the property DefaultText has a value that differs from the default value, False otherwise.
|
|
function IsDefaultTextStored(): Boolean;
|
|
function GetImageText(Node: PVirtualNode; Kind: TVTImageKind;
|
|
Column: TColumnIndex): string;
|
|
function GetOptions: TCustomStringTreeOptions;
|
|
function GetStaticText(Node: PVirtualNode; Column: TColumnIndex): string;
|
|
function GetText(Node: PVirtualNode; Column: TColumnIndex): string;
|
|
procedure ReadText(Reader: TReader);
|
|
procedure WriteText(Writer: TWriter);
|
|
procedure ResetInternalData(Node: PVirtualNode; Recursive: Boolean);
|
|
procedure SetDefaultText(const Value: string);
|
|
procedure SetOptions(const Value: TCustomStringTreeOptions);
|
|
procedure SetText(Node: PVirtualNode; Column: TColumnIndex; const Value: string);
|
|
procedure WMSetFont(var Msg: TWMSetFont); message WM_SETFONT;
|
|
procedure GetDataFromGrid(const AStrings : TStringList; const IncludeHeading : Boolean = True);
|
|
protected
|
|
/// <summary>Contains the name of the string that should be restored as selection</summary>
|
|
/// <seealso cref="TVTSelectionOption.toRestoreSelection">
|
|
FPreviouslySelected: TStringList;
|
|
procedure InitializeTextProperties(var PaintInfo: TVTPaintInfo);
|
|
procedure PaintNormalText(var PaintInfo: TVTPaintInfo; TextOutFlags: Integer; Text: string); virtual;
|
|
procedure PaintStaticText(const PaintInfo: TVTPaintInfo; pStaticTextAlignment: TAlignment; const Text: string); virtual; // [IPK] - private to protected
|
|
procedure AdjustPaintCellRect(var PaintInfo: TVTPaintInfo; var NextNonEmpty: TColumnIndex); override;
|
|
function CanExportNode(Node: PVirtualNode): Boolean;
|
|
function CalculateStaticTextWidth(Canvas: TCanvas; Node: PVirtualNode; Column: TColumnIndex; const Text: string): TDimension; virtual;
|
|
function CalculateTextWidth(Canvas: TCanvas; Node: PVirtualNode; Column: TColumnIndex; const Text: string): TDimension; virtual;
|
|
function ColumnIsEmpty(Node: PVirtualNode; Column: TColumnIndex): Boolean; override;
|
|
procedure DefineProperties(Filer: TFiler); override;
|
|
function DoCreateEditor(Node: PVirtualNode; Column: TColumnIndex): IVTEditLink; override;
|
|
procedure DoAddToSelection(Node: PVirtualNode); override;
|
|
function DoGetNodeHint(Node: PVirtualNode; Column: TColumnIndex; var LineBreakStyle: TVTTooltipLineBreakStyle): string; override;
|
|
function DoGetNodeTooltip(Node: PVirtualNode; Column: TColumnIndex; var LineBreakStyle: TVTTooltipLineBreakStyle): string; override;
|
|
function DoGetNodeExtraWidth(Node: PVirtualNode; Column: TColumnIndex; Canvas: TCanvas = nil): TDimension; override;
|
|
function DoGetNodeWidth(Node: PVirtualNode; Column: TColumnIndex; Canvas: TCanvas = nil): TDimension; override;
|
|
procedure DoGetText(var pEventArgs: TVSTGetCellTextEventArgs); virtual;
|
|
function DoIncrementalSearch(Node: PVirtualNode; const Text: string): Integer; override;
|
|
procedure DoNewText(Node: PVirtualNode; Column: TColumnIndex; const Text: string); virtual;
|
|
procedure DoPaintNode(var PaintInfo: TVTPaintInfo); override;
|
|
function DoShortenString(Canvas: TCanvas; Node: PVirtualNode; Column: TColumnIndex; const S: string; Width: TDimension;
|
|
EllipsisWidth: TDimension = 0): string; virtual;
|
|
procedure DoTextDrawing(var PaintInfo: TVTPaintInfo; const Text: string; CellRect: TRect; DrawFormat: Cardinal); virtual;
|
|
function DoTextMeasuring(Canvas: TCanvas; Node: PVirtualNode; Column: TColumnIndex; const Text: string): TSize; virtual;
|
|
function GetOptionsClass: TTreeOptionsClass; override;
|
|
procedure GetRenderStartValues(Source: TVSTTextSourceType; var Node: PVirtualNode;
|
|
var NextNodeProc: TGetNextNodeProc);
|
|
function InternalData(Node: PVirtualNode): Pointer;
|
|
procedure MainColumnChanged; override;
|
|
function ReadChunk(Stream: TStream; Version: Integer; Node: PVirtualNode; ChunkType,
|
|
ChunkSize: Integer): Boolean; override;
|
|
procedure ReadOldStringOptions(Reader: TReader);
|
|
function RenderOLEData(const FormatEtcIn: TFormatEtc; out Medium: TStgMedium; ForClipboard: Boolean): HResult; override;
|
|
procedure SetChildCount(Node: PVirtualNode; NewChildCount: Cardinal); override;
|
|
procedure WriteChunks(Stream: TStream; Node: PVirtualNode); override;
|
|
|
|
property DefaultText: string read FDefaultText write SetDefaultText stored False;// Stored via own writer
|
|
property EllipsisWidth: Integer read FEllipsisWidth;
|
|
property TreeOptions: TCustomStringTreeOptions read GetOptions write SetOptions;
|
|
|
|
property OnGetHint: TVSTGetHintEvent read FOnGetHint write FOnGetHint;
|
|
property OnGetText: TVSTGetTextEvent read FOnGetText write FOnGetText;
|
|
property OnGetCellText: TVSTGetCellTextEvent read fOnGetCellText write fOnGetCellText;
|
|
property OnNewText: TVSTNewTextEvent read FOnNewText write FOnNewText;
|
|
property OnShortenString: TVSTShortenStringEvent read FOnShortenString write FOnShortenString;
|
|
property OnMeasureTextWidth: TVTMeasureTextEvent read FOnMeasureTextWidth write FOnMeasureTextWidth;
|
|
property OnMeasureTextHeight: TVTMeasureTextEvent read FOnMeasureTextHeight write FOnMeasureTextHeight;
|
|
property OnDrawText: TVTDrawTextEvent read FOnDrawText write FOnDrawText;
|
|
public
|
|
constructor Create(AOwner: TComponent); override;
|
|
destructor Destroy(); override;
|
|
function AddChild(Parent: PVirtualNode; UserData: Pointer = nil): PVirtualNode; override;
|
|
function ComputeNodeHeight(Canvas: TCanvas; Node: PVirtualNode; Column: TColumnIndex; S: string = ''): TDimension; virtual;
|
|
function ContentToClipboard(Format: Word; Source: TVSTTextSourceType): HGLOBAL;
|
|
procedure ContentToCustom(Source: TVSTTextSourceType);
|
|
function ContentToHTML(Source: TVSTTextSourceType; const Caption: string = ''): String;
|
|
function ContentToRTF(Source: TVSTTextSourceType): RawByteString;
|
|
function ContentToText(Source: TVSTTextSourceType; Separator: Char): String; overload;
|
|
function ContentToUnicode(Source: TVSTTextSourceType; Separator: WideChar): string; overload; deprecated 'Use ContentToText instead';
|
|
function ContentToText(Source: TVSTTextSourceType; const Separator: string): string; overload;
|
|
procedure GetTextInfo(Node: PVirtualNode; Column: TColumnIndex; const AFont: TFont; var R: TRect;
|
|
var Text: string); override;
|
|
function InvalidateNode(Node: PVirtualNode): TRect; override;
|
|
function Path(Node: PVirtualNode; Column: TColumnIndex; Delimiter: Char): string;
|
|
procedure ReinitNode(Node: PVirtualNode; Recursive: Boolean; ForceReinit:
|
|
Boolean = False); override;
|
|
procedure RemoveFromSelection(Node: PVirtualNode); override;
|
|
function SaveToCSVFile(const FileNameWithPath : TFileName; const IncludeHeading : Boolean) : Boolean;
|
|
/// Alternate text for images used in Accessibility.
|
|
property ImageText[Node: PVirtualNode; Kind: TVTImageKind; Column: TColumnIndex]: string read GetImageText;
|
|
property StaticText[Node: PVirtualNode; Column: TColumnIndex]: string read GetStaticText;
|
|
property Text[Node: PVirtualNode; Column: TColumnIndex]: string read GetText write SetText;
|
|
end;
|
|
|
|
[ComponentPlatformsAttribute(pidWin32 or pidWin64)]
|
|
TVirtualStringTree = class(TCustomVirtualStringTree)
|
|
private
|
|
function GetOptions: TStringTreeOptions;
|
|
procedure SetOptions(const Value: TStringTreeOptions);
|
|
protected
|
|
function GetOptionsClass: TTreeOptionsClass; override;
|
|
public
|
|
property Canvas;
|
|
property RangeX;
|
|
property LastDragEffect;
|
|
property CheckImageKind; // should no more be published to make #622 fix working
|
|
published
|
|
property AccessibleName;
|
|
property Action;
|
|
property Align;
|
|
property Alignment;
|
|
property Anchors;
|
|
property AnimationDuration;
|
|
property AutoExpandDelay;
|
|
property AutoScrollDelay;
|
|
property AutoScrollInterval;
|
|
property Background;
|
|
property BackGroundImageTransparent;
|
|
property BackgroundOffsetX;
|
|
property BackgroundOffsetY;
|
|
property BiDiMode;
|
|
property BevelEdges;
|
|
property BevelInner;
|
|
property BevelOuter;
|
|
property BevelKind;
|
|
property BevelWidth;
|
|
property BorderStyle;
|
|
property BottomSpace;
|
|
property ButtonFillMode;
|
|
property ButtonStyle;
|
|
property BorderWidth;
|
|
property ChangeDelay;
|
|
property ClipboardFormats;
|
|
property Color;
|
|
property Colors;
|
|
property Constraints;
|
|
property Ctl3D;
|
|
property CustomCheckImages;
|
|
property DefaultNodeHeight;
|
|
property DefaultPasteMode;
|
|
property DefaultText;
|
|
property DragCursor;
|
|
property DragHeight;
|
|
property DragKind;
|
|
property DragImageKind;
|
|
property DragMode;
|
|
property DragOperations;
|
|
property DragType;
|
|
property DragWidth;
|
|
property DrawSelectionMode;
|
|
property EditDelay;
|
|
property EmptyListMessage;
|
|
property Enabled;
|
|
property Font;
|
|
property Header;
|
|
property HintMode;
|
|
property HotCursor;
|
|
property Images;
|
|
property IncrementalSearch;
|
|
property IncrementalSearchDirection;
|
|
property IncrementalSearchStart;
|
|
property IncrementalSearchTimeout;
|
|
property Indent;
|
|
property LineMode;
|
|
property LineStyle;
|
|
property Margin;
|
|
property NodeAlignment;
|
|
property NodeDataSize;
|
|
property OperationCanceled;
|
|
property ParentBiDiMode;
|
|
property ParentColor default False;
|
|
property ParentCtl3D;
|
|
property ParentFont;
|
|
property ParentShowHint;
|
|
property PopupMenu;
|
|
property RootNodeCount;
|
|
property ScrollBarOptions;
|
|
property SelectionBlendFactor;
|
|
property SelectionCurveRadius;
|
|
property ShowHint;
|
|
property StateImages;
|
|
property StyleElements;
|
|
{$if CompilerVersion >= 34}property StyleName;{$ifend}
|
|
property TabOrder;
|
|
property TabStop default True;
|
|
property TextMargin;
|
|
property TreeOptions: TStringTreeOptions read GetOptions write SetOptions;
|
|
property Visible;
|
|
property WantTabs;
|
|
|
|
property OnAddToSelection;
|
|
property OnAdvancedHeaderDraw;
|
|
property OnAfterAutoFitColumn;
|
|
property OnAfterAutoFitColumns;
|
|
property OnAfterCellPaint;
|
|
property OnAfterColumnExport;
|
|
property OnAfterColumnWidthTracking;
|
|
property OnAfterGetMaxColumnWidth;
|
|
property OnAfterHeaderExport;
|
|
property OnAfterHeaderHeightTracking;
|
|
property OnAfterItemErase;
|
|
property OnAfterItemPaint;
|
|
property OnAfterNodeExport;
|
|
property OnAfterPaint;
|
|
property OnAfterTreeExport;
|
|
property OnBeforeAutoFitColumn;
|
|
property OnBeforeAutoFitColumns;
|
|
property OnBeforeCellPaint;
|
|
property OnBeforeColumnExport;
|
|
property OnBeforeColumnWidthTracking;
|
|
property OnBeforeDrawTreeLine;
|
|
property OnBeforeGetMaxColumnWidth;
|
|
property OnBeforeHeaderExport;
|
|
property OnBeforeHeaderHeightTracking;
|
|
property OnBeforeItemErase;
|
|
property OnBeforeItemPaint;
|
|
property OnBeforeNodeExport;
|
|
property OnBeforePaint;
|
|
property OnBeforeTreeExport;
|
|
property OnCanSplitterResizeColumn;
|
|
property OnCanSplitterResizeHeader;
|
|
property OnCanSplitterResizeNode;
|
|
property OnChange;
|
|
property OnChecked;
|
|
property OnChecking;
|
|
property OnClick;
|
|
property OnCollapsed;
|
|
property OnCollapsing;
|
|
property OnColumnChecked;
|
|
property OnColumnChecking;
|
|
property OnColumnClick;
|
|
property OnColumnDblClick;
|
|
property OnColumnExport;
|
|
property OnColumnResize;
|
|
property OnColumnVisibilityChanged;
|
|
property OnColumnWidthDblClickResize;
|
|
property OnColumnWidthTracking;
|
|
property OnCompareNodes;
|
|
property OnContextPopup;
|
|
property OnCreateDataObject;
|
|
property OnCreateDragManager;
|
|
property OnCreateEditor;
|
|
property OnDblClick;
|
|
property OnDragAllowed;
|
|
property OnDragOver;
|
|
property OnDragDrop;
|
|
property OnDrawHint;
|
|
property OnDrawText;
|
|
property OnEditCancelled;
|
|
property OnEdited;
|
|
property OnEditing;
|
|
property OnEndDock;
|
|
property OnEndDrag;
|
|
property OnEndOperation;
|
|
property OnEnter;
|
|
property OnExit;
|
|
property OnExpanded;
|
|
property OnExpanding;
|
|
property OnFocusChanged;
|
|
property OnFocusChanging;
|
|
property OnFreeNode;
|
|
property OnGetCellText;
|
|
property OnGetCellIsEmpty;
|
|
property OnGetCursor;
|
|
property OnGetHeaderCursor;
|
|
property OnGetText;
|
|
property OnPaintText;
|
|
property OnGetHelpContext;
|
|
property OnGetHintKind;
|
|
property OnGetHintSize;
|
|
property OnGetImageIndex;
|
|
property OnGetImageIndexEx;
|
|
property OnGetImageText;
|
|
property OnGetHint;
|
|
property OnGetLineStyle;
|
|
property OnGetNodeDataSize;
|
|
property OnGetPopupMenu;
|
|
property OnGetUserClipboardFormats;
|
|
property OnHeaderAddPopupItem;
|
|
property OnHeaderClick;
|
|
property OnHeaderDblClick;
|
|
property OnHeaderDragged;
|
|
property OnHeaderDraggedOut;
|
|
property OnHeaderDragging;
|
|
property OnHeaderDraw;
|
|
property OnHeaderDrawQueryElements;
|
|
property OnHeaderHeightDblClickResize;
|
|
property OnHeaderHeightTracking;
|
|
property OnHeaderMouseDown;
|
|
property OnHeaderMouseMove;
|
|
property OnHeaderMouseUp;
|
|
property OnHotChange;
|
|
property OnIncrementalSearch;
|
|
property OnInitChildren;
|
|
property OnInitNode;
|
|
property OnKeyAction;
|
|
property OnKeyDown;
|
|
property OnKeyPress;
|
|
property OnKeyUp;
|
|
property OnLoadNode;
|
|
property OnLoadTree;
|
|
property OnMeasureItem;
|
|
property OnMeasureTextWidth;
|
|
property OnMeasureTextHeight;
|
|
property OnMouseDown;
|
|
property OnMouseMove;
|
|
property OnMouseUp;
|
|
property OnMouseWheel;
|
|
property OnMouseEnter;
|
|
property OnMouseLeave;
|
|
property OnNewText;
|
|
property OnNodeClick;
|
|
property OnNodeCopied;
|
|
property OnNodeCopying;
|
|
property OnNodeDblClick;
|
|
property OnNodeExport;
|
|
property OnNodeHeightDblClickResize;
|
|
property OnNodeHeightTracking;
|
|
property OnNodeMoved;
|
|
property OnNodeMoving;
|
|
property OnPaintBackground;
|
|
property OnPrepareButtonBitmaps;
|
|
property OnRemoveFromSelection;
|
|
property OnRenderOLEData;
|
|
property OnResetNode;
|
|
property OnResize;
|
|
property OnSaveNode;
|
|
property OnSaveTree;
|
|
property OnScroll;
|
|
property OnShortenString;
|
|
property OnShowScrollBar;
|
|
property OnBeforeGetCheckState;
|
|
property OnStartDock;
|
|
property OnStartDrag;
|
|
property OnStartOperation;
|
|
property OnStateChange;
|
|
property OnStructureChange;
|
|
property OnUpdating;
|
|
property OnCanResize;
|
|
property OnGesture;
|
|
property Touch;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
implementation
|
|
uses
|
|
System.TypInfo, // for migration stuff
|
|
System.StrUtils,
|
|
System.Types, // prevent inline compiler warning
|
|
System.UITypes, // prevent inline compiler warning
|
|
VirtualTrees.StyleHooks,
|
|
VirtualTrees.ClipBoard,
|
|
VirtualTrees.Utils,
|
|
VirtualTrees.Export,
|
|
VirtualTrees.EditLink,
|
|
VirtualTrees.BaseAncestorVcl{to eliminate H2443 about inline expanding}
|
|
;
|
|
|
|
const
|
|
cDefaultText = 'Node';
|
|
RTLFlag: array[Boolean] of Integer = (0, ETO_RTLREADING);
|
|
AlignmentToDrawFlag: array[TAlignment] of Cardinal = (DT_LEFT, DT_RIGHT, DT_CENTER);
|
|
gInitialized: Integer = 0; // >0 if global structures have been initialized; otherwise 0
|
|
|
|
//// initialization of stuff global to the unit
|
|
procedure InitializeGlobalStructures();
|
|
begin
|
|
if (gInitialized > 0) or (AtomicIncrement(gInitialized) <> 1) then // Ensure threadsafe that this code is executed only once
|
|
exit;
|
|
|
|
// Clipboard format registration.
|
|
// Specialized string tree formats.
|
|
CF_HTML := RegisterVTClipboardFormat(CFSTR_HTML, TCustomVirtualStringTree, 80);
|
|
CF_VRTFNOOBJS := RegisterVTClipboardFormat(CFSTR_RTFNOOBJS, TCustomVirtualStringTree, 84);
|
|
CF_VRTF := RegisterVTClipboardFormat(CFSTR_RTF, TCustomVirtualStringTree, 85);
|
|
CF_CSV := RegisterVTClipboardFormat(CFSTR_CSV, TCustomVirtualStringTree, 90);
|
|
// Predefined clipboard formats. Just add them to the internal list.
|
|
RegisterVTClipboardFormat(CF_TEXT, TCustomVirtualStringTree, 100);
|
|
RegisterVTClipboardFormat(CF_UNICODETEXT, TCustomVirtualStringTree, 95);
|
|
end;
|
|
|
|
|
|
//----------------- TCustomVirtualString -------------------------------------------------------------------------------
|
|
|
|
constructor TCustomVirtualStringTree.Create(AOwner: TComponent);
|
|
|
|
begin
|
|
InitializeGlobalStructures();
|
|
inherited;
|
|
FPreviouslySelected := nil;
|
|
FDefaultText := cDefaultText;
|
|
FInternalDataOffset := AllocateInternalDataArea(SizeOf(Cardinal));
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
procedure TCustomVirtualStringTree.GetRenderStartValues(Source: TVSTTextSourceType; var Node: PVirtualNode;
|
|
var NextNodeProc: TGetNextNodeProc);
|
|
|
|
begin
|
|
case Source of
|
|
tstInitialized:
|
|
begin
|
|
Node := GetFirstInitialized;
|
|
NextNodeProc := GetNextInitialized;
|
|
end;
|
|
tstSelected:
|
|
begin
|
|
Node := GetFirstSelected;
|
|
NextNodeProc := GetNextSelected;
|
|
end;
|
|
tstCutCopySet:
|
|
begin
|
|
Node := GetFirstCutCopy;
|
|
NextNodeProc := GetNextCutCopy;
|
|
end;
|
|
tstVisible:
|
|
begin
|
|
Node := GetFirstVisible(nil, True);
|
|
NextNodeProc := GetNextVisible;
|
|
end;
|
|
tstChecked:
|
|
begin
|
|
Node := GetFirstChecked;
|
|
NextNodeProc := GetNextChecked;
|
|
end;
|
|
else // tstAll
|
|
Node := GetFirst;
|
|
NextNodeProc := GetNext;
|
|
end;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
procedure TCustomVirtualStringTree.GetDataFromGrid(const AStrings: TStringList;
|
|
const IncludeHeading: Boolean);
|
|
var
|
|
LColIndex : Integer;
|
|
LStartIndex : Integer;
|
|
LAddString : string;
|
|
LCellText : string;
|
|
LChildNode : PVirtualNode;
|
|
begin
|
|
{ Start from the First column. }
|
|
LStartIndex := 0;
|
|
|
|
{ Do it for Header first }
|
|
if IncludeHeading then
|
|
begin
|
|
LAddString := EmptyStr;
|
|
for LColIndex := LStartIndex to Pred(Header.Columns.Count) do
|
|
begin
|
|
if (LColIndex > LStartIndex) then
|
|
LAddString := LAddString + ',';
|
|
LAddString := LAddString + AnsiQuotedStr(Header.Columns.Items[LColIndex].Text, '"');
|
|
end;//for
|
|
AStrings.Add(LAddString);
|
|
end;//if
|
|
|
|
{ Loop thru the virtual tree for Data }
|
|
LChildNode := GetFirst;
|
|
while Assigned(LChildNode) do
|
|
begin
|
|
LAddString := EmptyStr;
|
|
|
|
{ Read for each column and then populate the text }
|
|
for LColIndex := LStartIndex to Pred(Header.Columns.Count) do
|
|
begin
|
|
LCellText := Text[LChildNode, LColIndex];
|
|
if (LCellText = EmptyStr) then
|
|
LCellText := ' ';
|
|
if (LColIndex > LStartIndex) then
|
|
LAddString := LAddString + ',';
|
|
LAddString := LAddString + AnsiQuotedStr(LCellText, '"');
|
|
end;//for - Header.Columns.Count
|
|
|
|
AStrings.Add(LAddString);
|
|
LChildNode := LChildNode.NextSibling;
|
|
end;//while Assigned(LChildNode);
|
|
end;
|
|
|
|
function TCustomVirtualStringTree.GetImageText(Node: PVirtualNode;
|
|
Kind: TVTImageKind; Column: TColumnIndex): string;
|
|
begin
|
|
Assert(Assigned(Node), 'Node must not be nil.');
|
|
|
|
if not (vsInitialized in Node.States) then
|
|
InitNode(Node);
|
|
Result := '';
|
|
|
|
DoGetImageText(Node, Kind, Column, Result);
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.GetOptions: TCustomStringTreeOptions;
|
|
|
|
begin
|
|
Result := inherited TreeOptions as TCustomStringTreeOptions;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.GetStaticText(Node: PVirtualNode; Column: TColumnIndex): string;
|
|
|
|
var
|
|
lEventArgs: TVSTGetCellTextEventArgs;
|
|
|
|
begin
|
|
Assert(Assigned(Node), 'Node must not be nil.');
|
|
lEventArgs := TVSTGetCellTextEventArgs.Create(Node, Column);
|
|
DoGetText(lEventArgs);
|
|
Exit(lEventArgs.StaticText);
|
|
end;
|
|
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.GetText(Node: PVirtualNode; Column: TColumnIndex): string;
|
|
|
|
var
|
|
lEventArgs: TVSTGetCellTextEventArgs;
|
|
|
|
begin
|
|
Assert(Assigned(Node), 'Node must not be nil.');
|
|
lEventArgs := TVSTGetCellTextEventArgs.Create(Node, Column);
|
|
lEventArgs.CellText := FDefaultText;
|
|
DoGetText(lEventArgs);
|
|
Exit(lEventArgs.CellText)
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
procedure TCustomVirtualStringTree.InitializeTextProperties(var PaintInfo: TVTPaintInfo);
|
|
|
|
// Initializes default values for customization in PaintNormalText.
|
|
|
|
begin
|
|
with PaintInfo do
|
|
begin
|
|
// Set default font values first.
|
|
Canvas.Font.Assign(Font);
|
|
if Enabled then // Otherwise only those colors are used, which are passed from Font to Canvas.Font.
|
|
Canvas.Font.Color := Colors.NodeFontColor
|
|
else
|
|
Canvas.Font.Color := Colors.DisabledColor;
|
|
|
|
if (toHotTrack in TreeOptions.PaintOptions) and (Node = HotNode) then
|
|
begin
|
|
if not (tsUseExplorerTheme in TreeStates) then
|
|
begin
|
|
Canvas.Font.Style := Canvas.Font.Style + [TFontStyle.fsUnderline];
|
|
Canvas.Font.Color := Colors.HotColor;
|
|
end;
|
|
end;
|
|
|
|
// Change the font color only if the node also is drawn in selected style.
|
|
if poDrawSelection in PaintOptions then
|
|
begin
|
|
if (Column = FocusedColumn) or (toFullRowSelect in TreeOptions.SelectionOptions) then
|
|
begin
|
|
if Node = DropTargetNode then
|
|
begin
|
|
if ((LastDropMode = dmOnNode) or (vsSelected in Node.States)) then
|
|
Canvas.Font.Color := Colors.GetSelectedNodeFontColor(True); // See #1083, since drop highlight color is chosen independent of the focus state, we need to choose Font color also independent of it.
|
|
end
|
|
else
|
|
if vsSelected in Node.States then
|
|
begin
|
|
Canvas.Font.Color := Colors.GetSelectedNodeFontColor(Focused or (toPopupMode in TreeOptions.PaintOptions));
|
|
end;
|
|
end;
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
procedure TCustomVirtualStringTree.PaintNormalText(var PaintInfo: TVTPaintInfo; TextOutFlags: Integer;
|
|
Text: string);
|
|
|
|
// This method is responsible for painting the given text to target canvas (under consideration of the given rectangles).
|
|
// The text drawn here is considered as the normal text in a node.
|
|
// Note: NodeWidth is the actual width of the text to be drawn. This does not necessarily correspond to the width of
|
|
// the node rectangle. The clipping rectangle comprises the entire node (including tree lines, buttons etc.).
|
|
|
|
var
|
|
TripleWidth: TDimension;
|
|
R: TRect;
|
|
DrawFormat: Cardinal;
|
|
Height: TDimension;
|
|
lNewNodeWidth: TDimension;
|
|
begin
|
|
InitializeTextProperties(PaintInfo);
|
|
with PaintInfo do
|
|
begin
|
|
R := ContentRect;
|
|
Canvas.TextFlags := 0;
|
|
InflateRect(R, -TextMargin, 0);
|
|
|
|
if (vsDisabled in Node.States) or not Enabled then
|
|
Canvas.Font.Color := Colors.DisabledColor;
|
|
// Multiline nodes don't need special font handling or text manipulation.
|
|
// Note: multiline support requires the Unicode version of DrawText, which is able to do word breaking.
|
|
// The emulation in this unit does not support this so we have to use the OS version. However
|
|
// DrawTextW is only available on NT/2000/XP and up. Hence there is only partial multiline support
|
|
// for 9x/Me.
|
|
if vsMultiline in Node.States then
|
|
begin
|
|
DoPaintText(Node, Canvas, Column, ttNormal);
|
|
Height := ComputeNodeHeight(Canvas, Node, Column);
|
|
|
|
// The edit control flag will ensure that no partial line is displayed, that is, only lines
|
|
// which are (vertically) fully visible are drawn.
|
|
DrawFormat := DT_NOPREFIX or DT_WORDBREAK or DT_END_ELLIPSIS or DT_EDITCONTROL or AlignmentToDrawFlag[Alignment];
|
|
if BidiMode <> bdLeftToRight then
|
|
DrawFormat := DrawFormat or DT_RTLREADING;
|
|
|
|
// Center the text vertically if it fits entirely into the content rect.
|
|
if R.Bottom - R.Top > Height then
|
|
InflateRect(R, 0, Divide(Height - R.Bottom - R.Top, 2));
|
|
end
|
|
else
|
|
begin
|
|
FFontChanged := False;
|
|
TripleWidth := FEllipsisWidth;
|
|
DoPaintText(Node, Canvas, Column, ttNormal);
|
|
if FFontChanged then
|
|
begin
|
|
// If the font has been changed then the ellipsis width must be recalculated.
|
|
TripleWidth := 0;
|
|
// Recalculate also the width of the normal text.
|
|
lNewNodeWidth := DoTextMeasuring(Canvas, Node, Column, Text).cx + 2 * TextMargin;
|
|
if lNewNodeWidth <> NodeWidth then
|
|
begin
|
|
NodeWidth := lNewNodeWidth;
|
|
InvalidateNode(Node); // repaint node and selection as the font chnaged, see #1084
|
|
end;//if
|
|
end;// if FFontChanged
|
|
|
|
DrawFormat := DT_NOPREFIX or DT_VCENTER or DT_SINGLELINE;
|
|
if BidiMode <> bdLeftToRight then
|
|
DrawFormat := DrawFormat or DT_RTLREADING;
|
|
// Check if the text must be shortend.
|
|
if (Column > NoColumn) and ((NodeWidth - 2 * TextMargin) > R.Width) then
|
|
begin
|
|
Text := DoShortenString(Canvas, Node, Column, Text, R.Right - R.Left, TripleWidth);
|
|
if Alignment = taRightJustify then
|
|
DrawFormat := DrawFormat or DT_RIGHT
|
|
else
|
|
DrawFormat := DrawFormat or DT_LEFT;
|
|
end
|
|
else
|
|
DrawFormat := DrawFormat or AlignmentToDrawFlag[Alignment];
|
|
end;
|
|
|
|
if Canvas.TextFlags and ETO_OPAQUE = 0 then
|
|
SetBkMode(Canvas.Handle, TRANSPARENT)
|
|
else
|
|
SetBkMode(Canvas.Handle, OPAQUE);
|
|
|
|
DoTextDrawing(PaintInfo, Text, R, DrawFormat);
|
|
end;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
procedure TCustomVirtualStringTree.PaintStaticText(const PaintInfo: TVTPaintInfo; pStaticTextAlignment: TAlignment; const Text: string);
|
|
|
|
// This method retrives and draws the static text bound to a particular node.
|
|
|
|
var
|
|
R: TRect;
|
|
DrawFormat: Cardinal;
|
|
|
|
begin
|
|
with PaintInfo do
|
|
begin
|
|
Canvas.Font.Assign(Font);
|
|
if toFullRowSelect in TreeOptions.SelectionOptions then
|
|
begin
|
|
if Node = DropTargetNode then
|
|
begin
|
|
if (LastDropMode = dmOnNode) or (vsSelected in Node.States) then
|
|
Canvas.Font.Color := Colors.GetSelectedNodeFontColor(Focused or (toPopupMode in TreeOptions.PaintOptions))
|
|
else
|
|
Canvas.Font.Color := Colors.NodeFontColor;
|
|
end
|
|
else
|
|
if vsSelected in Node.States then
|
|
begin
|
|
if Focused or (toPopupMode in TreeOptions.PaintOptions) then
|
|
Canvas.Font.Color := Colors.GetSelectedNodeFontColor(Focused or (toPopupMode in TreeOptions.PaintOptions))
|
|
else
|
|
Canvas.Font.Color := Colors.NodeFontColor;
|
|
end;
|
|
end;
|
|
|
|
DrawFormat := DT_NOPREFIX or DT_VCENTER or DT_SINGLELINE;
|
|
Canvas.TextFlags := 0;
|
|
DoPaintText(Node, Canvas, Column, ttStatic);
|
|
|
|
// Disabled node color overrides all other variants.
|
|
if (vsDisabled in Node.States) or not Enabled then
|
|
Canvas.Font.Color := Colors.DisabledColor;
|
|
|
|
R := ContentRect;
|
|
if pStaticTextAlignment = taRightJustify then begin
|
|
DrawFormat := DrawFormat or DT_RIGHT;
|
|
Dec(R.Right, TextMargin);
|
|
if PaintInfo.Alignment = taRightJustify then
|
|
Dec(R.Right, NodeWidth); // room for node text
|
|
end
|
|
else begin
|
|
Inc(R.Left, TextMargin);
|
|
if PaintInfo.Alignment = taLeftJustify then
|
|
Inc(R.Left, NodeWidth); // room for node text
|
|
end;
|
|
|
|
if Canvas.TextFlags and ETO_OPAQUE = 0 then
|
|
SetBkMode(Canvas.Handle, TRANSPARENT)
|
|
else
|
|
SetBkMode(Canvas.Handle, OPAQUE);
|
|
Winapi.Windows.DrawTextW(Canvas.Handle, PWideChar(Text), Length(Text), R, DrawFormat);
|
|
end;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
procedure TCustomVirtualStringTree.ReadText(Reader: TReader);
|
|
begin
|
|
SetDefaultText(Reader.ReadString);
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.SaveToCSVFile(
|
|
const FileNameWithPath: TFileName; const IncludeHeading: Boolean): Boolean;
|
|
var
|
|
LResultList : TStringList;
|
|
begin
|
|
Result := False;
|
|
if (FileNameWithPath = '') then
|
|
Exit;
|
|
|
|
LResultList := TStringList.Create;
|
|
try
|
|
{ Get the data from grid. }
|
|
GetDataFromGrid(LResultList, IncludeHeading);
|
|
{ Save File to Disk }
|
|
LResultList.SaveToFile(FileNameWithPath);
|
|
Result := True;
|
|
finally
|
|
FreeAndNil(LResultList);
|
|
end;//try-finally
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
procedure TCustomVirtualStringTree.SetDefaultText(const Value: string);
|
|
|
|
begin
|
|
if FDefaultText <> Value then
|
|
begin
|
|
FDefaultText := Value;
|
|
if not (csLoading in ComponentState) then
|
|
Invalidate;
|
|
end;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
procedure TCustomVirtualStringTree.SetOptions(const Value: TCustomStringTreeOptions);
|
|
|
|
begin
|
|
inherited TreeOptions.Assign(Value);
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
procedure TCustomVirtualStringTree.SetText(Node: PVirtualNode; Column: TColumnIndex; const Value: string);
|
|
|
|
begin
|
|
DoNewText(Node, Column, Value);
|
|
InvalidateNode(Node);
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
procedure TCustomVirtualStringTree.WMSetFont(var Msg: TWMSetFont);
|
|
|
|
// Whenever a new font is applied to the tree some default values are determined to avoid frequent
|
|
// determination of the same value.
|
|
|
|
var
|
|
MemDC: HDC;
|
|
Run: PVirtualNode;
|
|
TM: TTextMetric;
|
|
Size: TSize;
|
|
Data: PInteger;
|
|
|
|
begin
|
|
inherited;
|
|
|
|
MemDC := CreateCompatibleDC(0);
|
|
try
|
|
SelectObject(MemDC, Msg.Font);
|
|
WinApi.Windows.GetTextMetrics(MemDC, TM);
|
|
FTextHeight := TM.tmHeight;
|
|
|
|
GetTextExtentPoint32W(MemDC, '...', 3, Size);
|
|
FEllipsisWidth := Size.cx;
|
|
finally
|
|
DeleteDC(MemDC);
|
|
end;
|
|
|
|
// Have to reset all node widths.
|
|
Run := RootNode.FirstChild;
|
|
while Assigned(Run) do
|
|
begin
|
|
Data := InternalData(Run);
|
|
if Assigned(Data) then
|
|
Data^ := 0;
|
|
Run := GetNextNoInit(Run);
|
|
end;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.AddChild(Parent: PVirtualNode; UserData: Pointer): PVirtualNode;
|
|
var
|
|
NewNodeText: string;
|
|
begin
|
|
Result := inherited AddChild(Parent, UserData);
|
|
// Restore the prviously restored node if the caption of this node is knwon and no other node was selected
|
|
if (toRestoreSelection in TreeOptions.SelectionOptions) and Assigned(FPreviouslySelected) and Assigned(OnGetText) then
|
|
begin
|
|
// See if this was the previously selected node and restore it in this case
|
|
Self.OnGetText(Self, Result, Header.RestoreSelectionColumnIndex, ttNormal, NewNodeText);
|
|
if FPreviouslySelected.IndexOf(NewNodeText) >= 0 then
|
|
begin
|
|
// Select this node and make sure that the parent node is expanded
|
|
TreeStates:= TreeStates + [tsPreviouslySelectedLocked];
|
|
try
|
|
Self.Selected[Result] := True;
|
|
finally
|
|
TreeStates:= TreeStates - [tsPreviouslySelectedLocked];
|
|
end;
|
|
// if a there is a selected node now, then make sure that it is visible
|
|
if (Self.GetFirstSelected <> nil) then
|
|
Self.FullyVisible[Self.GetFirstSelected]:= True;
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
procedure TCustomVirtualStringTree.AdjustPaintCellRect(var PaintInfo: TVTPaintInfo; var NextNonEmpty: TColumnIndex);
|
|
|
|
// In the case a node spans several columns (if enabled) we need to determine how many columns.
|
|
// Note: the autospan feature can only be used with left-to-right layout.
|
|
|
|
begin
|
|
if (toAutoSpanColumns in TreeOptions.AutoOptions) and Header.UseColumns and (PaintInfo.BidiMode = bdLeftToRight) then
|
|
with Header.Columns, PaintInfo do
|
|
begin
|
|
// Start with the directly following column.
|
|
NextNonEmpty := GetNextVisibleColumn(Column);
|
|
|
|
// Auto spanning columns can only be used for left-to-right directionality because the tree is drawn
|
|
// from left to right. For RTL directionality it would be necessary to draw it from right to left.
|
|
// While this could be managed, it becomes impossible when directionality is mixed.
|
|
repeat
|
|
if (NextNonEmpty = InvalidColumn) or not ColumnIsEmpty(Node, NextNonEmpty) or
|
|
(Items[NextNonEmpty].BidiMode <> bdLeftToRight) then
|
|
Break;
|
|
Inc(CellRect.Right, Items[NextNonEmpty].Width);
|
|
NextNonEmpty := GetNextVisibleColumn(NextNonEmpty);
|
|
until False;
|
|
end
|
|
else
|
|
inherited;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.CalculateStaticTextWidth(Canvas: TCanvas; Node: PVirtualNode; Column: TColumnIndex; const Text: string): TDimension;
|
|
|
|
begin
|
|
Result := 0;
|
|
if (Length(Text) > 0) and (Alignment <> taCenter) and not (vsMultiline in Node.States) then
|
|
begin
|
|
DoPaintText(Node, Canvas, Column, ttStatic);
|
|
|
|
Inc(Result, DoTextMeasuring(Canvas, Node, Column, Text).cx);
|
|
Inc(Result, TextMargin);
|
|
end;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.CalculateTextWidth(Canvas: TCanvas; Node: PVirtualNode; Column: TColumnIndex;
|
|
const Text: string): TDimension;
|
|
|
|
// Determines the width of the given text.
|
|
|
|
begin
|
|
Result := 2 * TextMargin;
|
|
if Length(Text) > 0 then
|
|
begin
|
|
Canvas.Font.Assign(Font);
|
|
DoPaintText(Node, Canvas, Column, ttNormal);
|
|
|
|
Inc(Result, DoTextMeasuring(Canvas, Node, Column, Text).cx);
|
|
end;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.ColumnIsEmpty(Node: PVirtualNode; Column: TColumnIndex): Boolean;
|
|
|
|
// For hit tests it is necessary to consider cases where columns are empty and automatic column spanning is enabled.
|
|
// This method simply checks the given column's text and if this is empty then the column is considered as being empty.
|
|
|
|
begin
|
|
Result := Length(Text[Node, Column]) = 0;
|
|
// If there is no text then let the ancestor decide if the column is to be considered as being empty
|
|
// (e.g. by asking the application). If there is text then the column is never be considered as being empty.
|
|
if Result then
|
|
Result := inherited ColumnIsEmpty(Node, Column);
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
procedure TCustomVirtualStringTree.DefineProperties(Filer: TFiler);
|
|
|
|
begin
|
|
inherited;
|
|
|
|
// For backwards compatiblity
|
|
Filer.DefineProperty('WideDefaultText', ReadText, nil, False);
|
|
// Delphi does never store an empty string unless we define the property in code.
|
|
Filer.DefineProperty('DefaultText', ReadText, WriteText, IsDefaultTextStored);
|
|
Filer.DefineProperty('StringOptions', ReadOldStringOptions, nil, False);
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
destructor TCustomVirtualStringTree.Destroy;
|
|
begin
|
|
FreeAndNil(FPreviouslySelected);
|
|
inherited;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
procedure TCustomVirtualStringTree.DoAddToSelection(Node: PVirtualNode);
|
|
var
|
|
lSelectedNodeCaption: string;
|
|
begin
|
|
inherited;
|
|
if (toRestoreSelection in TreeOptions.SelectionOptions) and Assigned(Self.OnGetText) and not (tsPreviouslySelectedLocked in TreeStates) then
|
|
begin
|
|
if not Assigned(FPreviouslySelected) then
|
|
begin
|
|
FPreviouslySelected := TStringList.Create();
|
|
FPreviouslySelected.Duplicates := dupIgnore;
|
|
FPreviouslySelected.Sorted := True; //Improves performance, required to use Find()
|
|
FPreviouslySelected.CaseSensitive := False;
|
|
end;
|
|
if Self.SelectedCount = 1 then
|
|
FPreviouslySelected.Clear();
|
|
Self.OnGetText(Self, Node, Header.RestoreSelectionColumnIndex, ttNormal, lSelectedNodeCaption);
|
|
FPreviouslySelected.Add(lSelectedNodeCaption);
|
|
end;//if
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.DoCreateEditor(Node: PVirtualNode; Column: TColumnIndex): IVTEditLink;
|
|
begin
|
|
Result := inherited DoCreateEditor(Node, Column);
|
|
// Enable generic label editing support if the application does not have own editors.
|
|
if Result = nil then
|
|
Result := TStringEditLink.Create;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.DoGetNodeHint(Node: PVirtualNode; Column: TColumnIndex;
|
|
var LineBreakStyle: TVTTooltipLineBreakStyle): string;
|
|
|
|
begin
|
|
Result := inherited DoGetNodeHint(Node, Column, LineBreakStyle);
|
|
if Assigned(FOnGetHint) then
|
|
FOnGetHint(Self, Node, Column, LineBreakStyle, Result);
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.DoGetNodeTooltip(Node: PVirtualNode; Column: TColumnIndex;
|
|
var LineBreakStyle: TVTTooltipLineBreakStyle): string;
|
|
|
|
begin
|
|
Result := inherited DoGetNodeToolTip(Node, Column, LineBreakStyle);
|
|
if Assigned(FOnGetHint) then
|
|
FOnGetHint(Self, Node, Column, LineBreakStyle, Result)
|
|
else
|
|
Result := Text[Node, Column];
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.DoGetNodeExtraWidth(Node: PVirtualNode; Column: TColumnIndex;
|
|
Canvas: TCanvas = nil): TDimension;
|
|
|
|
begin
|
|
if not (toShowStaticText in TreeOptions.StringOptions) then
|
|
Exit(0);
|
|
if Canvas = nil then
|
|
Canvas := Self.Canvas;
|
|
Result := CalculateStaticTextWidth(Canvas, Node, Column, StaticText[Node, Column]);
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.DoGetNodeWidth(Node: PVirtualNode; Column: TColumnIndex; Canvas: TCanvas = nil): TDimension;
|
|
|
|
// Returns the text width of the given node in pixels.
|
|
// This width is stored in the node's data member to increase access speed.
|
|
|
|
var
|
|
Data: PDimension;
|
|
|
|
begin
|
|
if (Column > NoColumn) and (vsMultiline in Node.States) then
|
|
Result := Header.Columns[Column].Width
|
|
else
|
|
begin
|
|
if Canvas = nil then
|
|
Canvas := Self.Canvas;
|
|
|
|
if (Column = Header.MainColumn) or (Column = NoColumn) then
|
|
begin
|
|
// Primary column or no columns.
|
|
Data := InternalData(Node);
|
|
if Assigned(Data) then
|
|
begin
|
|
Result := Data^;
|
|
if (Result = 0)
|
|
or Header.doingAutoFitColumns then
|
|
begin
|
|
Data^ := CalculateTextWidth(Canvas, Node, Column, Text[Node, Column]);
|
|
Result := Data^;
|
|
end;
|
|
end
|
|
else
|
|
Result := 0;
|
|
end
|
|
else
|
|
// any other column
|
|
Result := CalculateTextWidth(Canvas, Node, Column, Text[Node, Column]);
|
|
end;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
procedure TCustomVirtualStringTree.DoGetText(var pEventArgs: TVSTGetCellTextEventArgs);
|
|
|
|
begin
|
|
if not (vsInitialized in pEventArgs.Node.States) then
|
|
InitNode(pEventArgs.Node);
|
|
if Assigned(OnGetCellText) then
|
|
begin
|
|
OnGetCellText(Self, pEventArgs);
|
|
end
|
|
else if Assigned(FOnGetText) then begin
|
|
FOnGetText(Self, pEventArgs.Node, pEventArgs.Column, TVSTTextType.ttNormal, pEventArgs.CellText);
|
|
if toShowStaticText in TreeOptions.StringOptions then
|
|
FOnGetText(Self, pEventArgs.Node, pEventArgs.Column, TVSTTextType.ttStatic, pEventArgs.StaticText);
|
|
end;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.DoIncrementalSearch(Node: PVirtualNode; const Text: string): Integer;
|
|
|
|
// Since the string tree has access to node text it can do incremental search on its own. Use the event to
|
|
// override the default behavior.
|
|
|
|
begin
|
|
Result := 0;
|
|
if Assigned(OnIncrementalSearch) then
|
|
OnIncrementalSearch(Self, Node, Text, Result)
|
|
else
|
|
// Default behavior is to match the search string with the start of the node text.
|
|
if not StartsText(Text, GetText(Node, FocusedColumn)) then
|
|
Result := 1;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
procedure TCustomVirtualStringTree.DoNewText(Node: PVirtualNode; Column: TColumnIndex; const Text: string);
|
|
|
|
begin
|
|
if Assigned(FOnNewText) then
|
|
FOnNewText(Self, Node, Column, Text);
|
|
|
|
// The width might have changed, so update the scrollbar.
|
|
if UpdateCount = 0 then
|
|
UpdateHorizontalScrollBar(True);
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
procedure TCustomVirtualStringTree.DoPaintNode(var PaintInfo: TVTPaintInfo);
|
|
|
|
// Main output routine to print the text of the given node using the space provided in PaintInfo.ContentRect.
|
|
|
|
var
|
|
lEventArgs: TVSTGetCellTextEventArgs;
|
|
TextOutFlags: Integer;
|
|
|
|
begin
|
|
// Set a new OnChange event for the canvas' font so we know if the application changes it in the callbacks.
|
|
// This long winded procedure is necessary because font changes (as well as brush and pen changes) are
|
|
// unfortunately not announced via the Canvas.OnChange event.
|
|
RedirectFontChangeEvent(PaintInfo.Canvas);
|
|
try
|
|
|
|
// Determine main text direction as well as other text properties.
|
|
TextOutFlags := ETO_CLIPPED or RTLFlag[PaintInfo.BidiMode <> bdLeftToRight];
|
|
lEventArgs := TVSTGetCellTextEventArgs.Create(PaintInfo.Node, PaintInfo.Column);
|
|
|
|
lEventArgs.CellText := FDefaultText;
|
|
lEventArgs.StaticTextAlignment := PaintInfo.Alignment;
|
|
DoGetText(lEventArgs);
|
|
|
|
// Paint the normal text first...
|
|
if not lEventArgs.CellText.IsEmpty then
|
|
PaintNormalText(PaintInfo, TextOutFlags, lEventArgs.CellText);
|
|
|
|
// ... and afterwards the static text if not centered and the node is not multiline enabled.
|
|
if (Alignment <> taCenter) and not (vsMultiline in PaintInfo.Node.States) and (toShowStaticText in TreeOptions.StringOptions) and not lEventArgs.StaticText.IsEmpty then
|
|
PaintStaticText(PaintInfo, lEventArgs.StaticTextAlignment, lEventArgs.StaticText);
|
|
finally
|
|
RestoreFontChangeEvent(PaintInfo.Canvas);
|
|
end;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.DoShortenString(Canvas: TCanvas; Node: PVirtualNode; Column: TColumnIndex;
|
|
const S: string; Width: TDimension; EllipsisWidth: TDimension = 0): string;
|
|
|
|
var
|
|
Done: Boolean;
|
|
|
|
begin
|
|
Done := False;
|
|
if Assigned(FOnShortenString) then
|
|
FOnShortenString(Self, Canvas, Node, Column, S, Width, Result, Done);
|
|
if not Done then
|
|
Result := ShortenString(Canvas.Handle, S, Width, EllipsisWidth);
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
procedure TCustomVirtualStringTree.DoTextDrawing(var PaintInfo: TVTPaintInfo; const Text: string; CellRect: TRect;
|
|
DrawFormat: Cardinal);
|
|
|
|
var
|
|
DefaultDraw: Boolean;
|
|
lText: string;
|
|
begin
|
|
DefaultDraw := True;
|
|
if Assigned(FOnDrawText) then
|
|
FOnDrawText(Self, PaintInfo.Canvas, PaintInfo.Node, PaintInfo.Column, Text, CellRect, DefaultDraw);
|
|
if ((DrawFormat and DT_RIGHT) > 0) and (TFontStyle.fsItalic in PaintInfo.Canvas.Font.Style) then
|
|
lText := Text + ' '
|
|
else
|
|
lText := Text;
|
|
if DefaultDraw then
|
|
Winapi.Windows.DrawTextW(PaintInfo.Canvas.Handle, PWideChar(lText), Length(lText), CellRect, DrawFormat);
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.DoTextMeasuring(Canvas: TCanvas; Node: PVirtualNode; Column: TColumnIndex;
|
|
const Text: string): TSize;
|
|
|
|
var
|
|
R: TRect;
|
|
DrawFormat: Integer;
|
|
|
|
begin
|
|
GetTextExtentPoint32W(Canvas.Handle, PWideChar(Text), Length(Text), Result);
|
|
if vsMultiLine in Node.States then
|
|
begin
|
|
DrawFormat := DT_CALCRECT or DT_NOPREFIX or DT_WORDBREAK or DT_END_ELLIPSIS or DT_EDITCONTROL or AlignmentToDrawFlag[Alignment];
|
|
if BidiMode <> bdLeftToRight then
|
|
DrawFormat := DrawFormat or DT_RTLREADING;
|
|
|
|
R := Rect(0, 0, Result.cx, MaxInt);
|
|
Winapi.Windows.DrawTextW(Canvas.Handle, PWideChar(Text), Length(Text), R, DrawFormat);
|
|
Result.cx := R.Right - R.Left;
|
|
end;
|
|
if Assigned(FOnMeasureTextWidth) then
|
|
FOnMeasureTextWidth(Self, Canvas, Node, Column, Text, Result.cx);
|
|
if Assigned(FOnMeasureTextHeight) then
|
|
FOnMeasureTextHeight(Self, Canvas, Node, Column, Text, Result.cy);
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.GetOptionsClass: TTreeOptionsClass;
|
|
|
|
begin
|
|
Result := TCustomStringTreeOptions;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.InternalData(Node: PVirtualNode): Pointer;
|
|
|
|
begin
|
|
if (Node = nil) or (FInternalDataOffset = 0) then
|
|
Result := nil
|
|
else if Node = RootNode then
|
|
Result := PByte(Node) + FInternalDataOffset
|
|
else
|
|
Result := PByte(Node) + Self.NodeDataSize + FInternalDataOffset;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
procedure TCustomVirtualStringTree.MainColumnChanged;
|
|
|
|
var
|
|
Run: PVirtualNode;
|
|
Data: PInteger;
|
|
|
|
begin
|
|
inherited;
|
|
|
|
// Have to reset all node widths.
|
|
Run := RootNode.FirstChild;
|
|
while Assigned(Run) do
|
|
begin
|
|
Data := InternalData(Run);
|
|
if Assigned(Data) then
|
|
Data^ := 0;
|
|
Run := GetNextNoInit(Run);
|
|
end;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.ReadChunk(Stream: TStream; Version: Integer; Node: PVirtualNode; ChunkType,
|
|
ChunkSize: Integer): Boolean;
|
|
|
|
// read in the caption chunk if there is one
|
|
|
|
var
|
|
NewText: string;
|
|
|
|
begin
|
|
case ChunkType of
|
|
CaptionChunk:
|
|
begin
|
|
NewText := '';
|
|
if ChunkSize > 0 then
|
|
begin
|
|
SetLength(NewText, ChunkSize div 2);
|
|
Stream.Read(PWideChar(NewText)^, ChunkSize);
|
|
end;
|
|
// Do a new text event regardless of the caption content to allow removing the default string.
|
|
Text[Node, Header.MainColumn] := NewText;
|
|
Result := True;
|
|
end;
|
|
else
|
|
Result := inherited ReadChunk(Stream, Version, Node, ChunkType, ChunkSize);
|
|
end;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
type
|
|
TOldVTStringOption = (soSaveCaptions, soShowStaticText);
|
|
|
|
procedure TCustomVirtualStringTree.ReadOldStringOptions(Reader: TReader);
|
|
|
|
// Migration helper routine to silently convert forms containing the old tree options member into the new
|
|
// sub-options structure.
|
|
|
|
var
|
|
OldOption: TOldVTStringOption;
|
|
EnumName: string;
|
|
|
|
begin
|
|
// If we are at design time currently then let the designer know we changed something.
|
|
UpdateDesigner;
|
|
|
|
// It should never happen at this place that there is something different than the old set.
|
|
if Reader.ReadValue = vaSet then
|
|
with TreeOptions do
|
|
begin
|
|
// Remove all default values set by the constructor.
|
|
StringOptions := [];
|
|
|
|
while True do
|
|
begin
|
|
// Sets are stored with their members as simple strings. Read them one by one and map them to the new option
|
|
// in the correct sub-option set.
|
|
EnumName := Reader.ReadStr;
|
|
if EnumName = '' then
|
|
Break;
|
|
OldOption := TOldVTStringOption(GetEnumValue(TypeInfo(TOldVTStringOption), EnumName));
|
|
case OldOption of
|
|
soSaveCaptions:
|
|
StringOptions := StringOptions + [toSaveCaptions];
|
|
soShowStaticText:
|
|
StringOptions := StringOptions + [toShowStaticText];
|
|
end;
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.RenderOLEData(const FormatEtcIn: TFormatEtc; out Medium: TStgMedium;
|
|
ForClipboard: Boolean): HResult;
|
|
|
|
// Returns string expressions of all currently selected nodes in the Medium structure.
|
|
|
|
begin
|
|
Result := inherited RenderOLEData(FormatEtcIn, Medium, ForClipboard);
|
|
if Failed(Result) then
|
|
try
|
|
if ForClipboard then
|
|
Medium.hGlobal := ContentToClipboard(FormatEtcIn.cfFormat, tstCutCopySet)
|
|
else
|
|
Medium.hGlobal := ContentToClipboard(FormatEtcIn.cfFormat, tstSelected);
|
|
|
|
// Fill rest of the Medium structure if rendering went fine.
|
|
if Medium.hGlobal <> 0 then
|
|
begin
|
|
Medium.tymed := TYMED_HGLOBAL;
|
|
Medium.unkForRelease := nil;
|
|
|
|
Result := S_OK;
|
|
end;
|
|
except
|
|
Result := E_FAIL;
|
|
end;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
procedure TCustomVirtualStringTree.WriteChunks(Stream: TStream; Node: PVirtualNode);
|
|
|
|
// Adds another sibling chunk for Node storing the label if the node is initialized.
|
|
// Note: If the application stores a node's caption in the node's data member (which will be quite common) and needs to
|
|
// store more node specific data then it should use the OnSaveNode event rather than the caption autosave function
|
|
// (take out soSaveCaption from StringOptions). Otherwise the caption is unnecessarily stored twice.
|
|
|
|
var
|
|
ChunkHeader: TChunkHeader;
|
|
S: string;
|
|
Len: Integer;
|
|
|
|
begin
|
|
inherited;
|
|
if (toSaveCaptions in TreeOptions.StringOptions) and (Node <> RootNode) and
|
|
(vsInitialized in Node.States) then
|
|
with Stream do
|
|
begin
|
|
// Read the node's caption (primary column only).
|
|
S := Text[Node, Header.MainColumn];
|
|
Len := 2 * Length(S);
|
|
if Len > 0 then
|
|
begin
|
|
// Write a new sub chunk.
|
|
ChunkHeader.ChunkType := CaptionChunk;
|
|
ChunkHeader.ChunkSize := Len;
|
|
Write(ChunkHeader, SizeOf(ChunkHeader));
|
|
Write(PWideChar(S)^, Len);
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
procedure TCustomVirtualStringTree.WriteText(Writer: TWriter);
|
|
begin
|
|
Writer.WriteString(DefaultText);
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.ComputeNodeHeight(Canvas: TCanvas; Node: PVirtualNode; Column: TColumnIndex;
|
|
S: string): TDimension;
|
|
|
|
// Default node height calculation for multi line nodes. This method can be used by the application to delegate the
|
|
// computation to the string tree.
|
|
// Canvas is used to compute that value by using its current font settings.
|
|
// Node and Column describe the cell to be used for the computation.
|
|
// S is the string for which the height must be computed. If this string is empty the cell text is used instead.
|
|
|
|
var
|
|
DrawFormat: Cardinal;
|
|
BidiMode: TBidiMode;
|
|
Alignment: TAlignment;
|
|
PaintInfo: TVTPaintInfo;
|
|
Dummy: TColumnIndex;
|
|
lOffsets: TVTOffsets;
|
|
begin
|
|
if Length(S) = 0 then
|
|
S := Text[Node, Column];
|
|
|
|
if Column <= NoColumn then
|
|
begin
|
|
BidiMode := Self.BidiMode;
|
|
Alignment := Self.Alignment;
|
|
end
|
|
else
|
|
begin
|
|
BidiMode := Header.Columns[Column].BidiMode;
|
|
Alignment := Header.Columns[Column].Alignment;
|
|
end;
|
|
|
|
if BidiMode <> bdLeftToRight then
|
|
ChangeBidiModeAlignment(Alignment);
|
|
|
|
if vsMultiline in Node.States then
|
|
DrawFormat := DT_NOPREFIX or DT_TOP or DT_WORDBREAK or DT_EDITCONTROL
|
|
else
|
|
DrawFormat := DT_NOPREFIX or DT_VCENTER or DT_SINGLELINE;
|
|
DrawFormat := DrawFormat or DT_CALCRECT;
|
|
|
|
// Allow for autospanning.
|
|
PaintInfo.Node := Node;
|
|
PaintInfo.BidiMode := BidiMode;
|
|
PaintInfo.Column := Column;
|
|
PaintInfo.CellRect := Rect(0, 0, 0, 0);
|
|
GetOffsets(Node, lOffsets, TVTElement.ofsEndOfClientArea, Column);
|
|
if Column > NoColumn then
|
|
begin
|
|
PaintInfo.CellRect.Right := Header.Columns[Column].Width - 2 * TextMargin;
|
|
PaintInfo.CellRect.Left := lOffsets[TVTElement.ofsLabel];
|
|
end
|
|
else
|
|
PaintInfo.CellRect.Right := ClientWidth;
|
|
AdjustPaintCellRect(PaintInfo, Dummy);
|
|
|
|
if BidiMode <> bdLeftToRight then
|
|
DrawFormat := DrawFormat or DT_RIGHT or DT_RTLREADING
|
|
else
|
|
DrawFormat := DrawFormat or DT_LEFT;
|
|
Winapi.Windows.DrawTextW(Canvas.Handle, PWideChar(S), Length(S), PaintInfo.CellRect, DrawFormat);
|
|
Result := PaintInfo.CellRect.Bottom - PaintInfo.CellRect.Top;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.ContentToClipboard(Format: Word; Source: TVSTTextSourceType): HGLOBAL;
|
|
|
|
// This method constructs a shareable memory object filled with string data in the required format. Supported are:
|
|
// CF_TEXT - plain ANSI text (Unicode text is converted using the user's current locale)
|
|
// CF_UNICODETEXT - plain Unicode text
|
|
// CF_CSV - comma separated plain ANSI text
|
|
// CF_VRTF + CF_RTFNOOBS - rich text (plain ANSI)
|
|
// CF_HTML - HTML text encoded using UTF-8
|
|
//
|
|
// Result is the handle to a globally allocated memory block which can directly be used for clipboard and drag'n drop
|
|
// transfers. The caller is responsible for freeing the memory. If for some reason the content could not be rendered
|
|
// the Result is 0.
|
|
|
|
begin
|
|
Result := VirtualTrees.Export.ContentToClipboard(Self, Format, Source);
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.ContentToHTML(Source: TVSTTextSourceType; const Caption: string = ''): String;
|
|
|
|
// Renders the current tree content (depending on Source) as HTML text encoded in UTF-8.
|
|
// If Caption is not empty then it is used to create and fill the header for the table built here.
|
|
// Based on ideas and code from Frank van den Bergh and Andreas Hörstemeier.
|
|
|
|
begin
|
|
Result := VirtualTrees.Export.ContentToHTML(Self, Source, Caption);
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.CanExportNode(Node: PVirtualNode): Boolean;
|
|
|
|
begin
|
|
case TreeOptions.ExportMode of
|
|
emChecked:
|
|
Result := CheckState[Node] = csCheckedNormal;
|
|
emUnchecked:
|
|
Result := CheckState[Node] = csUncheckedNormal;
|
|
emVisibleDueToExpansion: //Do not export nodes that are not visible because their parent is not expanded
|
|
Result := not Assigned(Node.Parent) or Self.Expanded[Node.Parent];
|
|
emSelected: // export selected nodes only
|
|
Result := Selected[Node];
|
|
else
|
|
Result := True;
|
|
end;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
procedure TCustomVirtualStringTree.RemoveFromSelection(Node: PVirtualNode);
|
|
var
|
|
lSelectedNodeCaption: string;
|
|
lIndex: Integer;
|
|
begin
|
|
inherited;
|
|
if (toRestoreSelection in TreeOptions.SelectionOptions) and Assigned(FPreviouslySelected) and not Self.Selected[Node] then
|
|
begin
|
|
if Self.SelectedCount = 0 then
|
|
FPreviouslySelected.Clear()
|
|
else
|
|
begin
|
|
Self.OnGetText(Self, Node, Header.RestoreSelectionColumnIndex, ttNormal, lSelectedNodeCaption);
|
|
if FPreviouslySelected.Find(lSelectedNodeCaption, lIndex) then
|
|
FPreviouslySelected.Delete(lIndex);
|
|
end;//else
|
|
end;//if
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.ContentToRTF(Source: TVSTTextSourceType): RawByteString;
|
|
|
|
// Renders the current tree content (depending on Source) as RTF (rich text).
|
|
// Based on ideas and code from Frank van den Bergh and Andreas Hörstemeier.
|
|
|
|
begin
|
|
Result := VirtualTrees.Export.ContentToRTF(Self, Source);
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
procedure TCustomVirtualStringTree.ContentToCustom(Source: TVSTTextSourceType);
|
|
|
|
// Generic export procedure which polls the application at every stage of the export.
|
|
|
|
begin
|
|
VirtualTrees.Export.ContentToCustom(Self, Source);
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.ContentToText(Source: TVSTTextSourceType; Separator: Char): String;
|
|
|
|
begin
|
|
Result := ContentToText(Source, string(Separator));
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
|
|
function TCustomVirtualStringTree.ContentToUnicode(Source: TVSTTextSourceType; Separator: Char): string;
|
|
|
|
begin
|
|
Result := Self.ContentToText(Source, string(Separator));
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.ContentToText(Source: TVSTTextSourceType; const Separator: string): string;
|
|
|
|
// Renders the current tree content (depending on Source) as Unicode text.
|
|
// If an entry contains the separator char then it is wrapped with double quotation marks.
|
|
|
|
begin
|
|
Result := VirtualTrees.Export.ContentToUnicodeString(Self, Source, Separator);
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
procedure TCustomVirtualStringTree.GetTextInfo(Node: PVirtualNode; Column: TColumnIndex; const AFont: TFont; var R: TRect;
|
|
var Text: string);
|
|
|
|
// Returns the font, the text and its bounding rectangle to the caller. R is returned as the closest
|
|
// bounding rectangle around Text.
|
|
|
|
var
|
|
NewHeight: TDimension;
|
|
TM: TTextMetric;
|
|
|
|
begin
|
|
// Get default font and initialize the other parameters.
|
|
inherited GetTextInfo(Node, Column, AFont, R, Text);
|
|
|
|
Canvas.Font.Assign(AFont);
|
|
|
|
FFontChanged := False;
|
|
RedirectFontChangeEvent(Canvas);
|
|
DoPaintText(Node, Canvas, Column, ttNormal);
|
|
if FFontChanged then
|
|
begin
|
|
AFont.Assign(Canvas.Font);
|
|
GetTextMetrics(Canvas, TM);
|
|
NewHeight := TM.tmHeight;
|
|
end
|
|
else // Otherwise the correct font is already there and we only need to set the correct height.
|
|
NewHeight := FTextHeight;
|
|
RestoreFontChangeEvent(Canvas);
|
|
|
|
// Alignment to the actual text.
|
|
Text := Self.Text[Node, Column];
|
|
R := GetDisplayRect(Node, Column, True, not (vsMultiline in Node.States));
|
|
if toShowHorzGridLines in TreeOptions.PaintOptions then
|
|
Dec(R.Bottom);
|
|
InflateRect(R, 0, -Divide(R.Bottom - R.Top - NewHeight, 2));
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.InvalidateNode(Node: PVirtualNode): TRect;
|
|
|
|
var
|
|
Data: PInteger;
|
|
|
|
begin
|
|
Result := inherited InvalidateNode(Node);
|
|
// Reset node width so changed text attributes are applied correctly.
|
|
Data := InternalData(Node);
|
|
if Assigned(Data) then
|
|
Data^ := 0;
|
|
end;
|
|
|
|
function TCustomVirtualStringTree.IsDefaultTextStored: Boolean;
|
|
begin
|
|
Exit(DefaultText <> cDefaultText);
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TCustomVirtualStringTree.Path(Node: PVirtualNode; Column: TColumnIndex; Delimiter: Char): string;
|
|
|
|
// Constructs a string containing the node and all its parents. The last character in the returned path is always the
|
|
// given delimiter.
|
|
|
|
begin
|
|
if (Node = nil) or (Node = RootNode) then
|
|
Result := Delimiter
|
|
else
|
|
begin
|
|
Result := '';
|
|
while Node <> RootNode do
|
|
begin
|
|
Result := Text[Node, Column] + Delimiter + Result;
|
|
Node := Node.Parent;
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
procedure TCustomVirtualStringTree.ResetInternalData(Node: PVirtualNode; Recursive: Boolean);
|
|
|
|
var
|
|
Data: PInteger;
|
|
Run: PVirtualNode;
|
|
|
|
begin
|
|
// Reset node width so changed text attributes are applied correctly.
|
|
if Assigned(Node) and (Node <> RootNode) then
|
|
begin
|
|
Data := InternalData(Node);
|
|
if Assigned(Data) then
|
|
Data^ := 0;
|
|
end;
|
|
|
|
if Recursive then
|
|
begin
|
|
if Assigned(Node) then
|
|
Run := Node.FirstChild
|
|
else
|
|
Run := RootNode.FirstChild;
|
|
|
|
while Assigned(Run) do
|
|
begin
|
|
ResetInternalData(Run, Recursive);
|
|
Run := Run.NextSibling;
|
|
end;
|
|
end;//if Recursive
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
procedure TCustomVirtualStringTree.ReinitNode(Node: PVirtualNode; Recursive: Boolean; ForceReinit: Boolean = False);
|
|
|
|
begin
|
|
inherited;
|
|
|
|
ResetInternalData(Node, False); // False because we are already in a loop inside ReinitNode
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
procedure TCustomVirtualStringTree.SetChildCount(Node: PVirtualNode; NewChildCount: Cardinal);
|
|
|
|
begin
|
|
inherited;
|
|
ResetInternalData(Node, False);
|
|
end;
|
|
|
|
//----------------- TVirtualStringTree ---------------------------------------------------------------------------------
|
|
|
|
|
|
function TVirtualStringTree.GetOptions: TStringTreeOptions;
|
|
|
|
begin
|
|
Result := inherited TreeOptions as TStringTreeOptions;
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
procedure TVirtualStringTree.SetOptions(const Value: TStringTreeOptions);
|
|
|
|
begin
|
|
inherited TreeOptions.Assign(Value);
|
|
end;
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
function TVirtualStringTree.GetOptionsClass: TTreeOptionsClass;
|
|
|
|
begin
|
|
Result := TStringTreeOptions;
|
|
end;
|
|
|
|
{ TVSTGetCellTextEventArgs }
|
|
|
|
//----------------------------------------------------------------------------------------------------------------------
|
|
|
|
constructor TVSTGetCellTextEventArgs.Create(pNode: PVirtualNode; pColumn: TColumnIndex; pExportType: TVTExportType);
|
|
begin
|
|
Self.Node := pNode;
|
|
Self.Column := pColumn;
|
|
Self.ExportType := pExportType;
|
|
end;
|
|
|
|
initialization
|
|
TCustomStyleEngine.RegisterStyleHook(TVirtualStringTree, TVclStyleScrollBarsHook);
|
|
|
|
finalization
|
|
TCustomStyleEngine.UnRegisterStyleHook(TVirtualStringTree, TVclStyleScrollBarsHook);
|
|
|
|
end.
|