lazarus/components/codetools/codetoolsstructs.pas
mattias a9a7b211e5 codetools: started TGenericStringMap
git-svn-id: trunk@33926 -
2011-12-03 00:18:08 +00:00

1391 lines
37 KiB
ObjectPascal

{
***************************************************************************
* *
* This source is free software; you can redistribute it and/or modify *
* it under the terms of the GNU General Public License as published by *
* the Free Software Foundation; either version 2 of the License, or *
* (at your option) any later version. *
* *
* This code is distributed in the hope that it will be useful, but *
* WITHOUT ANY WARRANTY; without even the implied warranty of *
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU *
* General Public License for more details. *
* *
* A copy of the GNU General Public License is available on the World *
* Wide Web at <http://www.gnu.org/copyleft/gpl.html>. You can also *
* obtain it by writing to the Free Software Foundation, *
* Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. *
* *
***************************************************************************
Author: Mattias Gaertner
Abstract:
Most codetools returns simple values like a single code position or a
string. But some creates lists of data.
This unit provides structures for complex results.
}
unit CodeToolsStructs;
{$mode objfpc}{$H+}
interface
uses
Classes, SysUtils, FileProcs, AVL_Tree, BasicCodeTools;
type
TResourcestringInsertPolicy = (
rsipNone, // do not add/insert
rsipAppend, // append at end
rsipAlphabetically,// insert alphabetically
rsipContext // insert context sensitive
);
TPascalClassSection = (
pcsPrivate,
pcsProtected,
pcsPublic,
pcsPublished
);
TPascalClassSections = set of TPascalClassSection;
const
AllPascalClassSections = [low(TPascalClassSection)..high(TPascalClassSection)];
const
PascalClassSectionKeywords: array[TPascalClassSection] of string = (
'private',
'protected',
'public',
'published'
);
type
{ TMTAVLTreeNodeMemManager }
TMTAVLTreeNodeMemManager = class(TAVLTreeNodeMemManager)
public
procedure DisposeNode(ANode: TAVLTreeNode); override;
function NewNode: TAVLTreeNode; override;
end;
{ TMTAVLTree - TAVLTree with a multithreaded node manager }
TMTAVLTree = class(TAVLTree)
protected
fNodeManager: TAVLTreeNodeMemManager;
public
constructor Create(OnCompareMethod: TListSortCompare);
destructor Destroy; override;
end;
{ TGenericAVLTreeEnumerator }
generic TGenericAVLTreeEnumerator<TData> = class
private
FTree: TAVLTree;
FCurrent: TAVLTreeNode;
function GetCurrent: TData;
public
constructor Create(Tree: TAVLTree);
function MoveNext: boolean;
property Current: TData read GetCurrent;
end;
{ TBaseStringMap }
TBaseStringMap = class
private
type
TBaseStringMapItem = record
Name: string;
end;
PBaseStringMapItem = ^TBaseStringMapItem;
private
FCompareKeyItemFunc: TListSortCompare;
FTree: TMTAVLTree;// tree of PGenericStringMapItem
FCaseSensitive: boolean;
function GetCompareItemsFunc: TListSortCompare;
function FindNode(const s: string): TAVLTreeNode;
procedure DisposeItem(NodeData: Pointer); virtual; abstract;
function IsDataEqual(NodeData1, NodeData2: Pointer): boolean; virtual; abstract;
public
constructor Create(TheCaseSensitive: boolean);
destructor Destroy; override;
procedure Clear;
function Contains(const s: string): boolean;
function ContainsIdentifier(P: PChar): boolean;
function FindNodeWithIdentifierAsPrefix(P: PChar): TAVLTreeNode;
procedure GetNames(List: TStrings);
procedure Remove(const Name: string);
property CaseSensitive: boolean read FCaseSensitive;
property Tree: TMTAVLTree read FTree; // tree of PGenericStringMapItem
function Equals(OtherTree: TBaseStringMap): boolean; reintroduce;
procedure WriteDebugReport;
function CalcMemSize: PtrUint;
property CompareItemsFunc: TListSortCompare read GetCompareItemsFunc;
property CompareKeyItemFunc: TListSortCompare read FCompareKeyItemFunc;
procedure SetCompareFuncs(
const NewCompareItemsFunc, NewCompareKeyItemFunc: TListSortCompare);
end;
{ TGenericStringMap }
generic TGenericStringMap<TData> = class(TBaseStringMap)
public
type
TGenericStringMapItem = record
Name: string;
Value: TData;
end;
PGenericStringMapItem = ^TGenericStringMapItem;
TSpecStringMapEnumerator = specialize TGenericAVLTreeEnumerator<TData>;
private
function GetItems(const s: string): TData;
procedure SetItems(const s: string; const AValue: TData);
procedure DisposeItem(NodeData: Pointer); override;
function IsDataEqual(NodeData1, NodeData2: Pointer): boolean; override;
public
function Get(const Name: string; out Value: TData): boolean;
procedure Add(const Name: string; const Value: TData);
property Items[const s: string]: TData read GetItems write SetItems; default;
procedure Assign(Source: TGenericStringMap);
function GetEnumerator: TSpecStringMapEnumerator;
end;
TStringMap = class;
TStringMapItem = record
Name: string;
end;
PStringMapItem = ^TStringMapItem;
{ TStringMapEnumerator }
TStringMapEnumerator = class
protected
FTree: TAVLTree;
FCurrent: TAVLTreeNode;
public
constructor Create(Tree: TAVLTree);
function MoveNext: boolean;
// "Current" is implemented by the descendant classes
end;
{ TStringMap }
TStringMap = class
private
FCompareKeyItemFunc: TListSortCompare;
FTree: TAVLTree;// tree of PStringMapItem
FCaseSensitive: boolean;
function GetCompareItemsFunc: TListSortCompare;
function FindNode(const s: string): TAVLTreeNode;
protected
procedure DisposeItem(p: Pointer); virtual;
function ItemsAreEqual(p1, p2: Pointer): boolean; virtual;
procedure AssignItem(Src, Dest: Pointer); virtual;
public
constructor Create(TheCaseSensitive: boolean);
destructor Destroy; override;
procedure Clear; virtual;
function Contains(const s: string): boolean;
function ContainsIdentifier(P: PChar): boolean;
function FindNodeWithIdentifierAsPrefix(P: PChar): TAVLTreeNode;
procedure GetNames(List: TStrings);
procedure Remove(const Name: string); virtual;
property CaseSensitive: boolean read FCaseSensitive;
property Tree: TAVLTree read FTree; // tree of PStringMapItem
function Equals(OtherTree: TStringMap): boolean; reintroduce;
procedure Assign(Source: TStringMap); virtual; abstract;
procedure WriteDebugReport; virtual;
function CalcMemSize: PtrUint; virtual;
property CompareItemsFunc: TListSortCompare read GetCompareItemsFunc;
property CompareKeyItemFunc: TListSortCompare read FCompareKeyItemFunc;
procedure SetCompareFuncs(
const NewCompareItemsFunc, NewCompareKeyItemFunc: TListSortCompare);
end;
TStringToStringTreeItem = record
Name: string;
Value: string;
end;
PStringToStringTreeItem = ^TStringToStringTreeItem;
TStringToStringTree = class;
{ TStringToStringTreeEnumerator }
TStringToStringTreeEnumerator = class(TStringMapEnumerator)
private
function GetCurrent: PStringToStringTreeItem;
public
property Current: PStringToStringTreeItem read GetCurrent;
end;
{ TStringToStringTree }
TStringToStringTree = class(TStringMap)
private
function GetStrings(const s: string): string;
procedure SetStrings(const s: string; const AValue: string);
protected
procedure DisposeItem(p: Pointer); override;
function ItemsAreEqual(p1, p2: Pointer): boolean; override;
procedure AssignItem(Src, Dest: Pointer); override;
public
function GetString(const Name: string; out Value: string): boolean;
procedure Add(const Name, Value: string); virtual;
property Strings[const s: string]: string read GetStrings write SetStrings; default;
function AsText: string;
procedure Assign(Source: TStringMap); override;
procedure WriteDebugReport; override;
function CalcMemSize: PtrUint; override;
function GetEnumerator: TStringToStringTreeEnumerator;
end;
TStringToPointerTree = class;
TStringToPointerTreeItem = record
Name: string;
Value: Pointer;
end;
PStringToPointerTreeItem = ^TStringToPointerTreeItem;
{ TStringToPointerTreeEnumerator }
TStringToPointerTreeEnumerator = class(TStringMapEnumerator)
private
function GetCurrent: PStringToPointerTreeItem;
public
property Current: PStringToPointerTreeItem read GetCurrent;
end;
{ TStringToPointerTree }
TStringToPointerTree = class(TStringMap)
private
FFreeValues: boolean;
function GetItems(const s: string): Pointer;
procedure SetItems(const s: string; AValue: Pointer);
protected
procedure DisposeItem(p: Pointer); override;
function ItemsAreEqual(p1, p2: Pointer): boolean; override;
procedure AssignItem(Src, Dest: Pointer); override;
public
function GetItem(const Name: string; out Value: Pointer): boolean;
procedure Add(const Name: string; const Value: Pointer); virtual;
property Items[const s: string]: Pointer read GetItems write SetItems; default;
procedure Assign(Source: TStringMap); override;
function GetEnumerator: TStringToPointerTreeEnumerator;
property FreeValues: boolean read FFreeValues write FFreeValues;
end;
{ TFilenameToStringTree }
TFilenameToStringTree = class(TStringToStringTree)
public
constructor Create(CaseInsensitive: boolean); // false = system default
end;
{ TFilenameToPointerTree }
TFilenameToPointerTree = class(TStringToPointerTree)
public
constructor Create(CaseInsensitive: boolean); // false = system default
end;
TStringTree = class;
{ TStringTreeEnumerator }
TStringTreeEnumerator = class
private
FTree: TStringTree;
FCurrent: TAVLTreeNode;
function GetCurrent: string;
public
constructor Create(Tree: TStringTree);
function MoveNext: boolean;
property Current: string read GetCurrent;
end;
{ TStringTree }
TStringTree = class
public
Tree: TAVLTree;
constructor Create;
destructor Destroy; override;
procedure Clear;
function FindNode(const s: string): TAVLTreeNode; inline;
procedure ReplaceString(var s: string);
function CalcMemSize: PtrUInt;
function GetEnumerator: TStringTreeEnumerator;
end;
type
TCTComponentAccess = class(TComponent);
{ TComponentChildCollector }
TComponentChildCollector = class
private
FChildren: TFPList;
FRoot: TComponent;
procedure AddChildComponent(Child: TComponent);
public
constructor Create;
destructor Destroy; override;
function GetComponents(RootComponent: TComponent; AddRoot: boolean = true): TFPList;
property Children: TFPList read FChildren;
property Root: TComponent read FRoot;
end;
// case sensitive
function CompareStringToStringItems(Data1, Data2: Pointer): integer;
function CompareStringAndStringToStringTreeItem(Key, Data: Pointer): integer;
function CompareIdentifierAndStringToStringTreeItem(Identifier, Data: Pointer): integer;
function CompareIdentifierPrefixAndStringToStringTreeItem(Identifier, Data: Pointer): integer;
// case insensitive
function CompareStringToStringItemsI(Data1, Data2: Pointer): integer;
function CompareStringAndStringToStringTreeItemI(Key, Data: Pointer): integer;
function CompareIdentifierAndStringToStringTreeItemI(Identifier, Data: Pointer): integer;
function CompareIdentifierPrefixAndStringToStringTreeItemI(Identifier, Data: Pointer): integer;
function CompareFilenameToStringItems(Data1, Data2: Pointer): integer;
function CompareFilenameAndFilenameToStringTreeItem(Key, Data: Pointer): integer;
function CompareFilenameToStringItemsI(Data1, Data2: Pointer): integer;
function CompareFilenameAndFilenameToStringTreeItemI(Key, Data: Pointer): integer;
function CompareAnsiStringPtrs(Data1, Data2: Pointer): integer;
{$IF FPC_FULLVERSION<20701}
{$DEFINE EnableAVLFindPointerFix}
{$ENDIF}
function AVLFindPointer(Tree: TAVLTree; Data: Pointer): TAVLTreeNode; {$IFDEF EnableAVLFindPointerFix}inline;{$ENDIF}
implementation
function CompareStringToStringItems(Data1, Data2: Pointer): integer;
begin
Result:=CompareStr(PStringToStringTreeItem(Data1)^.Name,
PStringToStringTreeItem(Data2)^.Name);
end;
function CompareStringToStringItemsI(Data1, Data2: Pointer): integer;
begin
Result:=CompareText(PStringToStringTreeItem(Data1)^.Name,
PStringToStringTreeItem(Data2)^.Name);
end;
function CompareFilenameToStringItems(Data1, Data2: Pointer): integer;
begin
Result:=CompareFilenames(PStringToStringTreeItem(Data1)^.Name,
PStringToStringTreeItem(Data2)^.Name);
end;
function CompareStringAndStringToStringTreeItem(Key, Data: Pointer): integer;
begin
Result:=CompareStr(String(Key),PStringToStringTreeItem(Data)^.Name);
end;
function CompareIdentifierAndStringToStringTreeItem(Identifier, Data: Pointer
): integer;
var
Id: PChar absolute Identifier;
Item: PStringToStringTreeItem absolute Data;
IdLen: LongInt;
ItemLen: PtrInt;
begin
Result:=-CompareIdentifiersCaseSensitive(Id,PChar(Item^.Name));
if Result=0 then begin
IdLen:=GetIdentLen(Id);
ItemLen:=length(Item^.Name);
if IdLen=Itemlen then
Result:=0
else if IdLen>ItemLen then
Result:=1
else
Result:=-1;
end;
end;
function CompareIdentifierPrefixAndStringToStringTreeItem(Identifier,
Data: Pointer): integer;
var
Id: PChar absolute Identifier;
Item: PStringToStringTreeItem absolute Data;
begin
Result:=-CompareIdentifiersCaseSensitive(Id,PChar(Item^.Name));
end;
function CompareStringAndStringToStringTreeItemI(Key, Data: Pointer): integer;
begin
Result:=CompareText(String(Key),PStringToStringTreeItem(Data)^.Name);
end;
function CompareIdentifierAndStringToStringTreeItemI(Identifier, Data: Pointer
): integer;
var
Id: PChar absolute Identifier;
Item: PStringToStringTreeItem absolute Data;
IdLen: LongInt;
ItemLen: PtrInt;
begin
Result:=-CompareIdentifiers(Id,PChar(Item^.Name));
if Result=0 then begin
IdLen:=GetIdentLen(Id);
ItemLen:=length(Item^.Name);
if IdLen=Itemlen then
Result:=0
else if IdLen>ItemLen then
Result:=1
else
Result:=-1;
end;
end;
function CompareIdentifierPrefixAndStringToStringTreeItemI(Identifier,
Data: Pointer): integer;
var
Id: PChar absolute Identifier;
Item: PStringToStringTreeItem absolute Data;
begin
Result:=-CompareIdentifiers(Id,PChar(Item^.Name));
end;
function CompareFilenameAndFilenameToStringTreeItem(Key, Data: Pointer
): integer;
begin
Result:=CompareFilenames(String(Key),PStringToStringTreeItem(Data)^.Name);
end;
function CompareFilenameToStringItemsI(Data1, Data2: Pointer): integer;
begin
Result:=CompareFilenamesIgnoreCase(PStringToStringTreeItem(Data1)^.Name,
PStringToStringTreeItem(Data2)^.Name);
end;
function CompareFilenameAndFilenameToStringTreeItemI(Key, Data: Pointer
): integer;
begin
Result:=CompareFilenamesIgnoreCase(String(Key),
PStringToStringTreeItem(Data)^.Name);
end;
function CompareAnsiStringPtrs(Data1, Data2: Pointer): integer;
begin
Result:=CompareStr(AnsiString(Data1),AnsiString(Data2));
end;
function AVLFindPointer(Tree: TAVLTree; Data: Pointer): TAVLTreeNode;
begin
{$IFDEF EnableAVLFindPointerFix}
Result:=Tree.FindLeftMost(Data);
while (Result<>nil) do begin
if Result.Data=Data then break;
Result:=Tree.FindSuccessor(Result);
if Result=nil then exit;
if Tree.OnCompare(Data,Result.Data)<>0 then exit(nil);
end;
{$ELSE}
Result:=Tree.FindPointer(Data);
{$ENDIF}
end;
{ TGenericAVLTreeEnumerator }
function TGenericAVLTreeEnumerator.GetCurrent: TData;
begin
Result:=TData(FCurrent.Data);
end;
constructor TGenericAVLTreeEnumerator.Create(Tree: TAVLTree);
begin
FTree:=Tree;
end;
function TGenericAVLTreeEnumerator.MoveNext: boolean;
begin
if FCurrent=nil then
FCurrent:=FTree.FindLowest
else
FCurrent:=FTree.FindSuccessor(FCurrent);
Result:=FCurrent<>nil;
end;
{ TMTAVLTree }
constructor TMTAVLTree.Create(OnCompareMethod: TListSortCompare);
begin
inherited Create(OnCompareMethod);
fNodeManager:=TMTAVLTreeNodeMemManager.Create;
SetNodeManager(fNodeManager);
end;
destructor TMTAVLTree.Destroy;
begin
inherited Destroy;
FreeAndNil(fNodeManager);
end;
{ TMTAVLTreeNodeMemManager }
procedure TMTAVLTreeNodeMemManager.DisposeNode(ANode: TAVLTreeNode);
begin
ANode.Free;
end;
function TMTAVLTreeNodeMemManager.NewNode: TAVLTreeNode;
begin
Result:=TAVLTreeNode.Create;
end;
{ TBaseStringMap }
function TBaseStringMap.GetCompareItemsFunc: TListSortCompare;
begin
Result:=Tree.OnCompare;
end;
function TBaseStringMap.FindNode(const s: string): TAVLTreeNode;
begin
Result:=FTree.FindKey(Pointer(s),FCompareKeyItemFunc)
end;
constructor TBaseStringMap.Create(TheCaseSensitive: boolean);
begin
FCaseSensitive:=TheCaseSensitive;
if CaseSensitive then begin
FCompareKeyItemFunc:=@CompareStringAndStringToStringTreeItem;
FTree:=TMTAVLTree.Create(@CompareStringToStringItems);
end else begin
FCompareKeyItemFunc:=@CompareStringAndStringToStringTreeItemI;
FTree:=TMTAVLTree.Create(@CompareStringToStringItemsI);
end;
end;
destructor TBaseStringMap.Destroy;
begin
Clear;
FTree.Free;
FTree:=nil;
inherited Destroy;
end;
procedure TBaseStringMap.Clear;
var
Node: TAVLTreeNode;
begin
Node:=FTree.FindLowest;
while Node<>nil do begin
DisposeItem(Node.Data);
Node:=FTree.FindSuccessor(Node);
end;
FTree.Clear;
end;
function TBaseStringMap.Contains(const s: string): boolean;
begin
Result:=FindNode(s)<>nil;
end;
function TBaseStringMap.ContainsIdentifier(P: PChar): boolean;
begin
if CaseSensitive then
Result:=FTree.FindKey(p,@CompareIdentifierAndStringToStringTreeItem)<>nil
else
Result:=FTree.FindKey(p,@CompareIdentifierAndStringToStringTreeItemI)<>nil;
end;
function TBaseStringMap.FindNodeWithIdentifierAsPrefix(P: PChar): TAVLTreeNode;
begin
if CaseSensitive then
Result:=FTree.FindKey(p,@CompareIdentifierPrefixAndStringToStringTreeItem)
else
Result:=FTree.FindKey(p,@CompareIdentifierPrefixAndStringToStringTreeItemI);
end;
procedure TBaseStringMap.GetNames(List: TStrings);
var
Node: TAVLTreeNode;
Item: PBaseStringMapItem;
begin
Node:=Tree.FindLowest;
while Node<>nil do begin
Item:=PBaseStringMapItem(Node.Data);
List.Add(Item^.Name);
Node:=Tree.FindSuccessor(Node);
end;
end;
procedure TBaseStringMap.Remove(const Name: string);
var
Node: TAVLTreeNode;
Item: Pointer;
begin
Node:=FindNode(Name);
if Node<>nil then begin
Item:=Node.Data;
FTree.Delete(Node);
DisposeItem(Item);
end;
end;
function TBaseStringMap.Equals(OtherTree: TBaseStringMap): boolean;
var
Node: TAVLTreeNode;
OtherNode: TAVLTreeNode;
OtherItem: PBaseStringMapItem;
Item: PBaseStringMapItem;
begin
Result:=false;
if OtherTree=nil then exit;
if OtherTree.ClassType<>ClassType then exit;
if Tree.Count<>OtherTree.Tree.Count then exit;
Node:=Tree.FindLowest;
OtherNode:=OtherTree.Tree.FindLowest;
while Node<>nil do begin
if OtherNode=nil then exit;
Item:=PBaseStringMapItem(Node.Data);
OtherItem:=PBaseStringMapItem(OtherNode.Data);
if (Item^.Name<>OtherItem^.Name)
or (not IsDataEqual(Item,OtherItem)) then exit;
OtherNode:=OtherTree.Tree.FindSuccessor(OtherNode);
Node:=Tree.FindSuccessor(Node);
end;
if OtherNode<>nil then exit;
Result:=true;
end;
procedure TBaseStringMap.WriteDebugReport;
var
Node: TAVLTreeNode;
Item: PStringToStringTreeItem;
begin
DebugLn(['TGenericStringMap.WriteDebugReport ',Tree.Count]);
Node:=Tree.FindLowest;
while Node<>nil do begin
Item:=PStringToStringTreeItem(Node.Data);
DebugLn([Item^.Name]);
Node:=Tree.FindSuccessor(Node);
end;
end;
function TBaseStringMap.CalcMemSize: PtrUint;
var
Node: TAVLTreeNode;
Item: PBaseStringMapItem;
begin
Result:=PtrUInt(InstanceSize)
+PtrUInt(FTree.InstanceSize)
+PtrUint(FTree.Count)*SizeOf(TAVLTreeNode);
Node:=FTree.FindLowest;
while Node<>nil do begin
Item:=PBaseStringMapItem(Node.Data);
inc(Result,MemSizeString(Item^.Name)
+SizeOf(TBaseStringMapItem));
Node:=FTree.FindSuccessor(Node);
end;
end;
procedure TBaseStringMap.SetCompareFuncs(const NewCompareItemsFunc,
NewCompareKeyItemFunc: TListSortCompare);
begin
FCompareKeyItemFunc:=NewCompareKeyItemFunc;
Tree.OnCompare:=NewCompareItemsFunc;
end;
{ TGenericStringMap }
function TGenericStringMap.GetItems(const s: string): TData;
var
Node: TAVLTreeNode;
begin
Node:=FindNode(s);
if Node<>nil then
Result:=PGenericStringMapItem(Node.Data)^.Value
else
Result:=''
end;
procedure TGenericStringMap.SetItems(const s: string; const AValue: TData);
var
Node: TAVLTreeNode;
NewItem: PStringToStringTreeItem;
begin
Node:=FindNode(s);
if Node<>nil then begin
PGenericStringMapItem(Node.Data)^.Value:=AValue;
end else begin
New(NewItem);
NewItem^.Name:=s;
NewItem^.Value:=AValue;
FTree.Add(NewItem);
end;
end;
procedure TGenericStringMap.DisposeItem(NodeData: Pointer);
var
Item: PGenericStringMapItem absolute NodeData;
begin
DisposeItem(Item);
end;
function TGenericStringMap.IsDataEqual(NodeData1, NodeData2: Pointer): boolean;
var
Item1: PGenericStringMapItem absolute NodeData1;
Item2: PGenericStringMapItem absolute NodeData2;
begin
Result:=Item1^.Value=Item2^.Value;
end;
function TGenericStringMap.Get(const Name: string; out Value: TData): boolean;
var
Node: TAVLTreeNode;
begin
Node:=FindNode(Name);
if Node<>nil then begin
Value:=PGenericStringMapItem(Node.Data)^.Value;
Result:=true;
end else begin
Result:=false;
end;
end;
procedure TGenericStringMap.Add(const Name: string; const Value: TData);
begin
Items[Name]:=Value;
end;
procedure TGenericStringMap.Assign(Source: TGenericStringMap);
var
Node: TAVLTreeNode;
Item: PGenericStringMapItem;
begin
Clear;
Node:=Source.Tree.FindLowest;
while Node<>nil do begin
Item:=PGenericStringMapItem(Node.Data);
Items[Item^.Name]:=Item^.Value;
Node:=Source.Tree.FindSuccessor(Node);
end;
end;
function TGenericStringMap.GetEnumerator: TSpecStringMapEnumerator;
begin
Result:=TSpecStringMapEnumerator.Create(Tree);
end;
{ TFilenameToPointerTree }
constructor TFilenameToPointerTree.Create(CaseInsensitive: boolean);
begin
inherited Create(true);
if CaseInsensitive then
SetCompareFuncs(@CompareFilenameToStringItemsI,
@CompareFilenameAndFilenameToStringTreeItemI)
else
SetCompareFuncs(@CompareFilenameToStringItems,
@CompareFilenameAndFilenameToStringTreeItem);
end;
{ TStringToPointerTree }
function TStringToPointerTree.GetItems(const s: string): Pointer;
var
Node: TAVLTreeNode;
begin
Node:=FindNode(s);
if Node<>nil then
Result:=PStringToPointerTreeItem(Node.Data)^.Value
else
Result:=nil;
end;
procedure TStringToPointerTree.SetItems(const s: string; AValue: Pointer);
var
Node: TAVLTreeNode;
NewItem: PStringToPointerTreeItem;
begin
Node:=FindNode(s);
if Node<>nil then begin
NewItem:=PStringToPointerTreeItem(Node.Data);
if FreeValues then
TObject(NewItem^.Value).Free;
NewItem^.Value:=AValue;
end else begin
New(NewItem);
NewItem^.Name:=s;
NewItem^.Value:=AValue;
FTree.Add(NewItem);
end;
end;
procedure TStringToPointerTree.DisposeItem(p: Pointer);
var
Item: PStringToPointerTreeItem absolute p;
begin
if FreeValues then
TObject(Item^.Value).Free;
Dispose(Item);
end;
function TStringToPointerTree.ItemsAreEqual(p1, p2: Pointer): boolean;
var
Item1: PStringToPointerTreeItem absolute p1;
Item2: PStringToPointerTreeItem absolute p2;
begin
Result:=(Item1^.Name=Item2^.Name)
and (Item1^.Value=Item2^.Value);
end;
procedure TStringToPointerTree.AssignItem(Src, Dest: Pointer);
var
SrcItem: PStringToPointerTreeItem absolute Src;
DestItem: PStringToPointerTreeItem absolute Dest;
begin
DestItem^.Name:=SrcItem^.Name;
DestItem^.Value:=SrcItem^.Value;
end;
function TStringToPointerTree.GetItem(const Name: string; out Value: Pointer
): boolean;
var
Node: TAVLTreeNode;
begin
Node:=FindNode(Name);
if Node<>nil then begin
Value:=PStringToPointerTreeItem(Node.Data)^.Value;
Result:=true;
end else begin
Result:=false;
end;
end;
procedure TStringToPointerTree.Add(const Name: string; const Value: Pointer);
begin
Items[Name]:=Value;
end;
procedure TStringToPointerTree.Assign(Source: TStringMap);
var
Node: TAVLTreeNode;
Item: PStringToPointerTreeItem;
begin
if (Source=nil) or (Source.ClassType<>ClassType) then
raise Exception.Create('invalid class');
Clear;
Node:=Source.Tree.FindLowest;
while Node<>nil do begin
Item:=PStringToPointerTreeItem(Node.Data);
Items[Item^.Name]:=Item^.Value;
Node:=Source.Tree.FindSuccessor(Node);
end;
end;
function TStringToPointerTree.GetEnumerator: TStringToPointerTreeEnumerator;
begin
Result:=TStringToPointerTreeEnumerator.Create(FTree);
end;
{ TStringMapEnumerator }
constructor TStringMapEnumerator.Create(Tree: TAVLTree);
begin
FTree:=Tree;
end;
function TStringMapEnumerator.MoveNext: boolean;
begin
if FCurrent=nil then
FCurrent:=FTree.FindLowest
else
FCurrent:=FTree.FindSuccessor(FCurrent);
Result:=FCurrent<>nil;
end;
{ TStringToPointerTreeEnumerator }
function TStringToPointerTreeEnumerator.GetCurrent: PStringToPointerTreeItem;
begin
Result:=PStringToPointerTreeItem(FCurrent.Data);
end;
{ TStringMap }
function TStringMap.GetCompareItemsFunc: TListSortCompare;
begin
Result:=Tree.OnCompare;
end;
function TStringMap.FindNode(const s: string): TAVLTreeNode;
begin
Result:=FTree.FindKey(Pointer(s),FCompareKeyItemFunc);
end;
procedure TStringMap.DisposeItem(p: Pointer);
var
Item: PStringMapItem absolute p;
begin
Dispose(Item);
end;
function TStringMap.ItemsAreEqual(p1, p2: Pointer): boolean;
var
Item1: PStringMapItem absolute p1;
Item2: PStringMapItem absolute p2;
begin
Result:=Item1^.Name=Item2^.Name;
end;
procedure TStringMap.AssignItem(Src, Dest: Pointer);
var
SrcItem: PStringMapItem absolute Src;
DestItem: PStringMapItem absolute Dest;
begin
DestItem^.Name:=SrcItem^.Name;
end;
constructor TStringMap.Create(TheCaseSensitive: boolean);
begin
FCaseSensitive:=TheCaseSensitive;
if CaseSensitive then begin
FCompareKeyItemFunc:=@CompareStringAndStringToStringTreeItem;
FTree:=TAVLTree.Create(@CompareStringToStringItems);
end else begin
FCompareKeyItemFunc:=@CompareStringAndStringToStringTreeItemI;
FTree:=TAVLTree.Create(@CompareStringToStringItemsI);
end;
end;
destructor TStringMap.Destroy;
begin
Clear;
FTree.Free;
FTree:=nil;
inherited Destroy;
end;
procedure TStringMap.Clear;
var
Node: TAVLTreeNode;
begin
Node:=FTree.FindLowest;
while Node<>nil do begin
DisposeItem(PStringMapItem(Node.Data));
Node:=FTree.FindSuccessor(Node);
end;
FTree.Clear;
end;
function TStringMap.Contains(const s: string): boolean;
begin
Result:=FindNode(s)<>nil;
end;
function TStringMap.ContainsIdentifier(P: PChar): boolean;
begin
if CaseSensitive then
Result:=FTree.FindKey(p,@CompareIdentifierAndStringToStringTreeItem)<>nil
else
Result:=FTree.FindKey(p,@CompareIdentifierAndStringToStringTreeItemI)<>nil;
end;
function TStringMap.FindNodeWithIdentifierAsPrefix(P: PChar): TAVLTreeNode;
begin
if CaseSensitive then
Result:=FTree.FindKey(p,@CompareIdentifierPrefixAndStringToStringTreeItem)
else
Result:=FTree.FindKey(p,@CompareIdentifierPrefixAndStringToStringTreeItemI);
end;
procedure TStringMap.GetNames(List: TStrings);
var
Node: TAVLTreeNode;
Item: PStringMapItem;
begin
Node:=Tree.FindLowest;
while Node<>nil do begin
Item:=PStringMapItem(Node.Data);
List.Add(Item^.Name);
Node:=Tree.FindSuccessor(Node);
end;
end;
procedure TStringMap.Remove(const Name: string);
var
Node: TAVLTreeNode;
Item: PStringMapItem;
begin
Node:=FindNode(Name);
if Node<>nil then begin
Item:=PStringMapItem(Node.Data);
FTree.Delete(Node);
Dispose(Item);
end;
end;
function TStringMap.Equals(OtherTree: TStringMap): boolean;
var
Node: TAVLTreeNode;
OtherNode: TAVLTreeNode;
OtherItem: PStringMapItem;
Item: PStringMapItem;
begin
Result:=false;
if (OtherTree=nil) or (OtherTree.ClassType<>ClassType) then exit;
if Tree.Count<>OtherTree.Tree.Count then exit;
Node:=Tree.FindLowest;
OtherNode:=OtherTree.Tree.FindLowest;
while Node<>nil do begin
if OtherNode=nil then exit;
Item:=PStringMapItem(Node.Data);
OtherItem:=PStringMapItem(OtherNode.Data);
if not ItemsAreEqual(Item,OtherItem) then exit;
OtherNode:=OtherTree.Tree.FindSuccessor(OtherNode);
Node:=Tree.FindSuccessor(Node);
end;
if OtherNode<>nil then exit;
Result:=true;
end;
procedure TStringMap.WriteDebugReport;
var
Node: TAVLTreeNode;
Item: PStringMapItem;
begin
DebugLn(['TStringMap.WriteDebugReport ',Tree.Count]);
Node:=Tree.FindLowest;
while Node<>nil do begin
Item:=PStringMapItem(Node.Data);
DebugLn([Item^.Name]);
Node:=Tree.FindSuccessor(Node);
end;
end;
function TStringMap.CalcMemSize: PtrUint;
var
Node: TAVLTreeNode;
Item: PStringMapItem;
begin
Result:=PtrUInt(InstanceSize)
+PtrUInt(FTree.InstanceSize)
+PtrUint(FTree.Count)*SizeOf(TAVLTreeNode);
Node:=FTree.FindLowest;
while Node<>nil do begin
Item:=PStringMapItem(Node.Data);
inc(Result,MemSizeString(Item^.Name)
+SizeOf(TStringMapItem));
Node:=FTree.FindSuccessor(Node);
end;
end;
procedure TStringMap.SetCompareFuncs(const NewCompareItemsFunc,
NewCompareKeyItemFunc: TListSortCompare);
begin
FCompareKeyItemFunc:=NewCompareKeyItemFunc;
Tree.OnCompare:=NewCompareItemsFunc;
end;
{ TStringToStringTreeEnumerator }
function TStringToStringTreeEnumerator.GetCurrent: PStringToStringTreeItem;
begin
Result:=PStringToStringTreeItem(FCurrent.Data);
end;
{ TStringTreeEnumerator }
function TStringTreeEnumerator.GetCurrent: string;
begin
Result:=AnsiString(FCurrent.Data);
end;
constructor TStringTreeEnumerator.Create(Tree: TStringTree);
begin
FTree:=Tree;
end;
function TStringTreeEnumerator.MoveNext: boolean;
begin
if FCurrent=nil then
FCurrent:=FTree.Tree.FindLowest
else
FCurrent:=FTree.Tree.FindSuccessor(FCurrent);
Result:=FCurrent<>nil;
end;
{ TStringToStringTree }
function TStringToStringTree.GetStrings(const s: string): string;
var
Node: TAVLTreeNode;
begin
Node:=FindNode(s);
if Node<>nil then
Result:=PStringToStringTreeItem(Node.Data)^.Value
else
Result:=''
end;
procedure TStringToStringTree.SetStrings(const s: string; const AValue: string);
var
Node: TAVLTreeNode;
NewItem: PStringToStringTreeItem;
begin
Node:=FindNode(s);
if Node<>nil then begin
PStringToStringTreeItem(Node.Data)^.Value:=AValue;
end else begin
New(NewItem);
NewItem^.Name:=s;
NewItem^.Value:=AValue;
FTree.Add(NewItem);
end;
end;
procedure TStringToStringTree.DisposeItem(p: Pointer);
var
Item: PStringToStringTreeItem absolute p;
begin
Dispose(Item);
end;
function TStringToStringTree.ItemsAreEqual(p1, p2: Pointer): boolean;
var
Item1: PStringToStringTreeItem absolute p1;
Item2: PStringToStringTreeItem absolute p2;
begin
Result:=(Item1^.Name=Item2^.Name)
and (Item1^.Value=Item2^.Value);
end;
procedure TStringToStringTree.AssignItem(Src, Dest: Pointer);
var
SrcItem: PStringToStringTreeItem absolute Src;
DestItem: PStringToStringTreeItem absolute Dest;
begin
DestItem^.Name:=SrcItem^.Name;
DestItem^.Value:=SrcItem^.Value;
end;
function TStringToStringTree.GetString(const Name: string; out Value: string
): boolean;
var
Node: TAVLTreeNode;
begin
Node:=FindNode(Name);
if Node<>nil then begin
Value:=PStringToStringTreeItem(Node.Data)^.Value;
Result:=true;
end else begin
Result:=false;
end;
end;
procedure TStringToStringTree.Add(const Name, Value: string);
begin
Strings[Name]:=Value;
end;
function TStringToStringTree.AsText: string;
var
Node: TAVLTreeNode;
Item: PStringToStringTreeItem;
begin
Result:='';
Node:=Tree.FindLowest;
while Node<>nil do begin
Item:=PStringToStringTreeItem(Node.Data);
Result:=Result+Item^.Name+'='+Item^.Value+LineEnding;
Node:=Tree.FindSuccessor(Node);
end;
end;
procedure TStringToStringTree.Assign(Source: TStringMap);
var
Node: TAVLTreeNode;
Item: PStringToStringTreeItem;
begin
if (Source=nil) or (Source.ClassType<>ClassType) then
raise Exception.Create('invalid class');
Clear;
Node:=Source.Tree.FindLowest;
while Node<>nil do begin
Item:=PStringToStringTreeItem(Node.Data);
Strings[Item^.Name]:=Item^.Value;
Node:=Source.Tree.FindSuccessor(Node);
end;
end;
procedure TStringToStringTree.WriteDebugReport;
var
Node: TAVLTreeNode;
Item: PStringToStringTreeItem;
begin
DebugLn(['TStringToStringTree.WriteDebugReport ',Tree.Count]);
Node:=Tree.FindLowest;
while Node<>nil do begin
Item:=PStringToStringTreeItem(Node.Data);
DebugLn([Item^.Name,'=',Item^.Value]);
Node:=Tree.FindSuccessor(Node);
end;
end;
function TStringToStringTree.CalcMemSize: PtrUint;
var
Node: TAVLTreeNode;
Item: PStringToStringTreeItem;
begin
Result:=PtrUInt(InstanceSize)
+PtrUInt(FTree.InstanceSize)
+PtrUint(FTree.Count)*SizeOf(TAVLTreeNode);
Node:=FTree.FindLowest;
while Node<>nil do begin
Item:=PStringToStringTreeItem(Node.Data);
inc(Result,MemSizeString(Item^.Name)
+MemSizeString(Item^.Value)
+SizeOf(TStringToStringTreeItem));
Node:=FTree.FindSuccessor(Node);
end;
end;
function TStringToStringTree.GetEnumerator: TStringToStringTreeEnumerator;
begin
Result:=TStringToStringTreeEnumerator.Create(FTree);
end;
{ TFilenameToStringTree }
constructor TFilenameToStringTree.Create(CaseInsensitive: boolean);
begin
inherited Create(true);
if CaseInsensitive then
SetCompareFuncs(@CompareFilenameToStringItemsI,
@CompareFilenameAndFilenameToStringTreeItemI)
else
SetCompareFuncs(@CompareFilenameToStringItems,
@CompareFilenameAndFilenameToStringTreeItem);
end;
{ TStringTree }
constructor TStringTree.Create;
begin
Tree:=TAVLTree.Create(@CompareAnsiStringPtrs);
end;
destructor TStringTree.Destroy;
begin
Clear;
FreeAndNil(Tree);
inherited Destroy;
end;
procedure TStringTree.Clear;
var
Node: TAVLTreeNode;
begin
Node:=Tree.FindLowest;
while Node<>nil do begin
AnsiString(Node.Data):='';
Node:=Tree.FindSuccessor(Node);
end;
Tree.Clear;
end;
function TStringTree.FindNode(const s: string): TAVLTreeNode;
begin
Result:=Tree.Find(Pointer(s));
end;
procedure TStringTree.ReplaceString(var s: string);
var
Node: TAVLTreeNode;
h: String;
begin
if GetStringRefCount(s)<=0 then exit;
Node:=FindNode(s);
if Node=nil then begin
// increase refcount
h:=s;
Tree.Add(Pointer(h));
Pointer(h):=nil; // keep refcount
//debugln(['TStringTree.ReplaceString new string: refcount=',GetStringRefCount(s)]);
//debugln(['TStringTree.ReplaceString NewString="',dbgstr(s),'"']);
end else begin
s:=AnsiString(Node.Data);
//debugln(['TStringTree.ReplaceString old string: refcount=',GetStringRefCount(s)]);
//debugln(['TStringTree.ReplaceString OldString="',dbgstr(s),'"']);
end;
//debugln(['TStringTree.ReplaceString ',GetStringRefCount(s),' ',Node<>nil]);
end;
function TStringTree.CalcMemSize: PtrUInt;
var
Node: TAVLTreeNode;
begin
Result:=PtrUInt(InstanceSize)
+PtrUInt(Tree.InstanceSize)
+PtrUInt(TAVLTreeNode.InstanceSize)*PtrUInt(Tree.Count);
Node:=Tree.FindLowest;
while Node<>nil do begin
inc(Result,MemSizeString(AnsiString(Node.Data)));
Node:=Tree.FindSuccessor(Node);
end;
end;
function TStringTree.GetEnumerator: TStringTreeEnumerator;
begin
Result:=TStringTreeEnumerator.Create(Self);
end;
{ TComponentChildCollector }
procedure TComponentChildCollector.AddChildComponent(Child: TComponent);
var
OldRoot: TComponent;
begin
//debugln(['TComponentChildCollector.AddChildComponent ',DbgSName(Child)]);
Children.Add(Child);
OldRoot := Root;
try
if csInline in Child.ComponentState then
FRoot := Child;
TCTComponentAccess(Child).GetChildren(@AddChildComponent,Root);
finally
FRoot := OldRoot;
end;
end;
constructor TComponentChildCollector.Create;
begin
FChildren:=TFPList.Create;
end;
destructor TComponentChildCollector.Destroy;
begin
FreeAndNil(FChildren);
inherited Destroy;
end;
function TComponentChildCollector.GetComponents(RootComponent: TComponent;
AddRoot: boolean): TFPList;
begin
Children.Clear;
if AddRoot then
Children.Add(RootComponent);
FRoot:=RootComponent;
TCTComponentAccess(RootComponent).GetChildren(@AddChildComponent,FRoot);
Result:=Children;
end;
end.