mirror of
				https://gitlab.com/freepascal.org/lazarus/lazarus.git
				synced 2025-10-26 11:01:50 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			1101 lines
		
	
	
		
			27 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
			
		
		
	
	
			1101 lines
		
	
	
		
			27 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
| {*******************************************************}
 | |
| {                                                       }
 | |
| {         Delphi VCL Extensions (RX)                    }
 | |
| {                                                       }
 | |
| {         Copyright (c) 1995, 1996 AO ROSNO             }
 | |
| {         Copyright (c) 1997 Master-Bank                }
 | |
| {                                                       }
 | |
| {*******************************************************}
 | |
| 
 | |
| {$mode objfpc}
 | |
| {$h+}
 | |
| 
 | |
| unit Placement;
 | |
| 
 | |
| interface
 | |
| 
 | |
| uses Controls, Classes, LazUTF8, Forms, IniFiles, Dialogs, RTTIUtils;
 | |
| 
 | |
|   
 | |
| type
 | |
|   TPlacementOption = (fpState, fpPosition, fpActiveControl);
 | |
|   TPlacementOptions = set of TPlacementOption;
 | |
|   TPlacementOperation = (poSave, poRestore);
 | |
| 
 | |
|   TIniLink       = Class;
 | |
|   TFormPlacement = Class;
 | |
|   TStoredValue   = Class;
 | |
|   TStoredValues  = Class;
 | |
|   
 | |
| 
 | |
| { TStoredValue }
 | |
| 
 | |
| {$ifdef storevariant}
 | |
|   TStoredType = Variant;
 | |
| {$else}
 | |
|   TStoredType = AnsiString;
 | |
| {$endif}  
 | |
| 
 | |
|   TStoredValueEvent = procedure(Sender: TStoredValue; var Value: TStoredType) of object;
 | |
| 
 | |
|   TStoredValue = class(TCollectionItem)
 | |
|   private
 | |
|     FName: string;
 | |
|     FValue: TStoredType;
 | |
|     FKeyString: string;
 | |
|     FOnSave: TStoredValueEvent;
 | |
|     FOnRestore: TStoredValueEvent;
 | |
|     function IsValueStored: Boolean;
 | |
|     function GetStoredValues: TStoredValues;
 | |
|   protected
 | |
|     function GetDisplayName: string; override;
 | |
|     procedure SetDisplayName(const Value: string); override;
 | |
|   public
 | |
|     constructor Create(ACollection: TCollection); override;
 | |
|     procedure Assign(Source: TPersistent); override;
 | |
|     procedure Clear;
 | |
|     procedure Save; virtual;
 | |
|     procedure Restore; virtual;
 | |
|     property StoredValues: TStoredValues read GetStoredValues;
 | |
|   published
 | |
|     property Name: string read FName write SetDisplayName;
 | |
|     property Value: TStoredType read FValue write FValue stored IsValueStored;
 | |
|     property KeyString: string read FKeyString write FKeyString;
 | |
|     property OnSave: TStoredValueEvent read FOnSave write FOnSave;
 | |
|     property OnRestore: TStoredValueEvent read FOnRestore write FOnRestore;
 | |
|   end;
 | |
| 
 | |
| { TStoredValues }
 | |
| 
 | |
|   TStoredValues = class(TOwnedCollection)
 | |
|   private
 | |
|     FStorage: TFormPlacement;
 | |
|     function GetValue(const AName: string): TStoredValue;
 | |
|     procedure SetValue(const AName: string; StoredValue: TStoredValue);
 | |
|     function GetStoredValue(const AName: string): TStoredType;
 | |
|     procedure SetStoredValue(const AName: string; Value: TStoredType);
 | |
|     function GetItem(Index: Integer): TStoredValue;
 | |
|     procedure SetItem(Index: Integer; StoredValue: TStoredValue);
 | |
|   public
 | |
|     constructor Create(AOwner: TPersistent);
 | |
|     function IndexOf(const AName: string): Integer;
 | |
|     procedure SaveValues; virtual;
 | |
|     procedure RestoreValues; virtual;
 | |
|     property Storage: TFormPlacement read FStorage write FStorage;
 | |
|     property Items[Index: Integer]: TStoredValue read GetItem write SetItem; default;
 | |
|     property Values[const Name: string]: TStoredValue read GetValue write SetValue;
 | |
|     property StoredValue[const Name: string]: TStoredType read GetStoredValue write SetStoredValue;
 | |
|   end;
 | |
| 
 | |
| { TFormPlacement }
 | |
| 
 | |
|   TFormPlacement = class(TComponent)
 | |
|   private
 | |
|     FActive: Boolean;
 | |
|     FIniFileName: String;
 | |
|     FIniSection: String;
 | |
|     FIniFile: TCustomIniFile;
 | |
|     FLinks: TList;
 | |
|     FOptions: TPlacementOptions;
 | |
|     FVersion: Integer;
 | |
|     FSaved: Boolean;
 | |
|     FRestored: Boolean;
 | |
|     FDestroying: Boolean;
 | |
|     //FDefMaximize: Boolean;
 | |
|     FSaveFormShow: TNotifyEvent;
 | |
|     FSaveFormDestroy: TNotifyEvent;
 | |
|     FSaveFormCloseQuery: TCloseQueryEvent;
 | |
|     FOnSavePlacement: TNotifyEvent;
 | |
|     FOnRestorePlacement: TNotifyEvent;
 | |
|     procedure SetEvents;
 | |
|     procedure RestoreEvents;
 | |
|     function  GetIniSection: string;
 | |
|     procedure SetIniSection(const Value: string);
 | |
|     function  GetIniFileName: string;
 | |
|     procedure SetIniFileName(const Value: string);
 | |
|     procedure AddLink(ALink: TIniLink);
 | |
|     procedure NotifyLinks(Operation: TPlacementOperation);
 | |
|     procedure RemoveLink(ALink: TIniLink);
 | |
|     procedure FormShow(Sender: TObject);
 | |
|     procedure FormCloseQuery(Sender: TObject; var CanClose: Boolean);
 | |
|     procedure FormDestroy(Sender: TObject);
 | |
|     function GetForm: TForm;
 | |
|   protected
 | |
|     procedure IniNeeded(ReadOnly: Boolean);Virtual;
 | |
|     procedure IniFree;Virtual;
 | |
|     procedure Loaded; override;
 | |
|     procedure Save; dynamic;
 | |
|     procedure Restore; dynamic;
 | |
|     procedure SavePlacement; virtual;
 | |
|     procedure RestorePlacement; virtual;
 | |
|     function  DoReadString(const Section, Ident, Default: string): string; virtual;
 | |
|     procedure DoWriteString(const Section, Ident, Value: string); virtual;
 | |
|     property  Form: TForm read GetForm;
 | |
|   public
 | |
|     constructor Create(AOwner: TComponent); override;
 | |
|     destructor Destroy; override;
 | |
|     procedure SaveFormPlacement;
 | |
|     procedure RestoreFormPlacement;
 | |
|     function ReadString(const Ident, Default: string): string;
 | |
|     procedure WriteString(const Ident, Value: string);
 | |
|     function ReadInteger(const Ident: string; Default: Longint): Longint;
 | |
|     procedure WriteInteger(const Ident: string; Value: Longint);
 | |
|     procedure EraseSections;
 | |
|     property IniFile: TCustomIniFile read FIniFile;
 | |
|   published
 | |
|     property Active: Boolean read FActive write FActive default True;
 | |
|     property IniFileName: string read GetIniFileName write SetIniFileName;
 | |
|     property IniSection: string read GetIniSection write SetIniSection;
 | |
|     property Options: TPlacementOptions read FOptions write FOptions default [fpState, fpPosition];
 | |
|     property Version: Integer read FVersion write FVersion default 0;
 | |
|     property OnSavePlacement: TNotifyEvent read FOnSavePlacement write FOnSavePlacement;
 | |
|     property OnRestorePlacement: TNotifyEvent read FOnRestorePlacement  write FOnRestorePlacement;
 | |
|   end;
 | |
| 
 | |
| { TFormStorage }
 | |
| 
 | |
|   TFormStorage = class(TFormPlacement)
 | |
|   private
 | |
|     FStoredProps: TStrings;
 | |
|     FStoredValues: TStoredValues;
 | |
|     procedure SetStoredProps(Value: TStrings);
 | |
|     procedure SetStoredValues(Value: TStoredValues);
 | |
|     function GetStoredValue(const AName: string): TstoredType;
 | |
|     procedure SetStoredValue(const AName: string; Value: TStoredType);
 | |
|   protected
 | |
|     procedure Loaded; override;
 | |
|     procedure Notification(AComponent: TComponent; Operation: TOperation); override;
 | |
|     procedure SavePlacement; override;
 | |
|     procedure RestorePlacement; override;
 | |
|     procedure SaveProperties; virtual;
 | |
|     procedure RestoreProperties; virtual;
 | |
|     procedure WriteState(Writer: TWriter); override;
 | |
|   public
 | |
|     constructor Create(AOwner: TComponent); override;
 | |
|     destructor Destroy; override;
 | |
|     procedure SetNotification;
 | |
|     property StoredValue[const AName: string]: TStoredType read GetStoredValue write SetStoredValue;
 | |
|   published
 | |
|     property StoredProps: TStrings read FStoredProps write SetStoredProps;
 | |
|     property StoredValues: TStoredValues read FStoredValues write SetStoredValues;
 | |
|   end;
 | |
| 
 | |
| { TIniLink }
 | |
| 
 | |
|   TIniLink = class(TPersistent)
 | |
|   private
 | |
|     FStorage: TFormPlacement;
 | |
|     FOnSave: TNotifyEvent;
 | |
|     FOnLoad: TNotifyEvent;
 | |
|     function GetIniObject: TCustomIniFile;
 | |
|     function GetRootSection: string;
 | |
|     procedure SetStorage(Value: TFormPlacement);
 | |
|   protected
 | |
|     procedure SaveToIni; virtual;
 | |
|     procedure LoadFromIni; virtual;
 | |
|   public
 | |
|     destructor Destroy; override;
 | |
|     property IniObject: TCustomInifile read GetIniObject;
 | |
|     property Storage: TFormPlacement read FStorage write SetStorage;
 | |
|     property RootSection: string read GetRootSection;
 | |
|     property OnSave: TNotifyEvent read FOnSave write FOnSave;
 | |
|     property OnLoad: TNotifyEvent read FOnLoad write FOnLoad;
 | |
|   end;
 | |
| 
 | |
| 
 | |
| implementation
 | |
| 
 | |
| uses SysUtils, AppUtils, RTLConsts;
 | |
| 
 | |
| const
 | |
| { The following strings should not be localized }
 | |
|   siActiveCtrl = 'ActiveControl';
 | |
|   siVisible = 'Visible';
 | |
|   siVersion = 'FormVersion';
 | |
| 
 | |
| function XorEncode(const Key, Source: string): string;
 | |
| var
 | |
|   I: Integer;
 | |
|   C: Byte;
 | |
| begin
 | |
|   Result := '';
 | |
|   for I := 1 to Length(Source) do begin
 | |
|     if Length(Key) > 0 then
 | |
|       C := Byte(Key[1 + ((I - 1) mod Length(Key))]) xor Byte(Source[I])
 | |
|     else
 | |
|       C := Byte(Source[I]);
 | |
|     Result := Result + AnsiLowerCase(IntToHex(C, 2));
 | |
|   end;
 | |
| end;
 | |
| 
 | |
| function XorDecode(const Key, Source: string): string;
 | |
| var
 | |
|   I: Integer;
 | |
|   C: Char;
 | |
|   
 | |
| begin
 | |
|   Result := '';
 | |
|   for I := 0 to Length(Source) div 2 - 1 do begin
 | |
|     C := Chr(StrToIntDef('$' + Copy(Source, (I * 2) + 1, 2), Ord(' ')));
 | |
|     if Length(Key) > 0 then
 | |
|       C := Chr(Byte(Key[1 + (I mod Length(Key))]) xor Byte(C));
 | |
|     Result := Result + C;
 | |
|   end;
 | |
| end;
 | |
|                             
 | |
| 
 | |
| Function GetDefaultIniName : String;
 | |
| 
 | |
| begin
 | |
| {$ifdef unix}
 | |
|   Result:=IncludeTrailingPathDelimiter(GetEnvironmentVariableUTF8('HOME'))
 | |
|           +'.'+ExtractFileName(Application.ExeName)
 | |
| 
 | |
| {$else}
 | |
|   Result:=ChangeFileExt(Application.ExeName,'.ini');
 | |
| {$endif}
 | |
| end;
 | |
| 
 | |
| function FindPart(const HelpWilds, InputStr: string): Integer;
 | |
| 
 | |
| var
 | |
|   I, J: Integer;
 | |
|   Diff: Integer;
 | |
|   
 | |
| begin
 | |
|   I := Pos('?', HelpWilds);
 | |
|   if I = 0 then begin
 | |
|     { if no '?' in HelpWilds }
 | |
|     Result := Pos(HelpWilds, InputStr);
 | |
|     Exit;
 | |
|   end;
 | |
|   { '?' in HelpWilds }
 | |
|   Diff := Length(InputStr) - Length(HelpWilds);
 | |
|   if Diff < 0 then begin
 | |
|     Result := 0;
 | |
|     Exit;
 | |
|   end;
 | |
|   { now move HelpWilds over InputStr }
 | |
|   for I := 0 to Diff do begin
 | |
|     for J := 1 to Length(HelpWilds) do begin
 | |
|       if (InputStr[I + J] = HelpWilds[J]) or
 | |
|         (HelpWilds[J] = '?') then
 | |
|       begin
 | |
|         if J = Length(HelpWilds) then begin
 | |
|           Result := I + 1;
 | |
|           Exit;
 | |
|         end;
 | |
|       end
 | |
|       else Break;
 | |
|     end;
 | |
|   end;
 | |
|   Result := 0;
 | |
| end;
 | |
| 
 | |
| 
 | |
| 
 | |
| function IsWild(InputStr, Wilds: string; IgnoreCase: Boolean): Boolean;
 | |
| 
 | |
|  function SearchNext(var Wilds: string): Integer;
 | |
|  { looking for next *, returns position and string until position }
 | |
|  begin
 | |
|    Result := Pos('*', Wilds);
 | |
|    if Result > 0 then Wilds := Copy(Wilds, 1, Result - 1);
 | |
|  end;
 | |
| 
 | |
| var
 | |
|   CWild, CInputWord: Integer; { counter for positions }
 | |
|   I, LenHelpWilds: Integer;
 | |
|   MaxInputWord, MaxWilds: Integer; { Length of InputStr and Wilds }
 | |
|   HelpWilds: string;
 | |
| begin
 | |
|   if Wilds = InputStr then begin
 | |
|     Result := True;
 | |
|     Exit;
 | |
|   end;
 | |
|   repeat { delete '**', because '**' = '*' }
 | |
|     I := Pos('**', Wilds);
 | |
|     if I > 0 then
 | |
|       Wilds := Copy(Wilds, 1, I - 1) + '*' + Copy(Wilds, I + 2, MaxInt);
 | |
|   until I = 0;
 | |
|   if Wilds = '*' then begin { for fast end, if Wilds only '*' }
 | |
|     Result := True;
 | |
|     Exit;
 | |
|   end;
 | |
|   MaxInputWord := Length(InputStr);
 | |
|   MaxWilds := Length(Wilds);
 | |
|   if IgnoreCase then begin { upcase all letters }
 | |
|     InputStr := AnsiUpperCase(InputStr);
 | |
|     Wilds := AnsiUpperCase(Wilds);
 | |
|   end;
 | |
|   if (MaxWilds = 0) or (MaxInputWord = 0) then begin
 | |
|     Result := False;
 | |
|     Exit;
 | |
|   end;
 | |
|   CInputWord := 1;
 | |
|   CWild := 1;
 | |
|   Result := True;
 | |
|   repeat
 | |
|     if InputStr[CInputWord] = Wilds[CWild] then begin { equal letters }
 | |
|       { goto next letter }
 | |
|       Inc(CWild);
 | |
|       Inc(CInputWord);
 | |
|       Continue;
 | |
|     end;
 | |
|     if Wilds[CWild] = '?' then begin { equal to '?' }
 | |
|       { goto next letter }
 | |
|       Inc(CWild);
 | |
|       Inc(CInputWord);
 | |
|       Continue;
 | |
|     end;
 | |
|     if Wilds[CWild] = '*' then begin { handling of '*' }
 | |
|       HelpWilds := Copy(Wilds, CWild + 1, MaxWilds);
 | |
|       I := SearchNext(HelpWilds);
 | |
|       LenHelpWilds := Length(HelpWilds);
 | |
|       if I = 0 then begin
 | |
|         { no '*' in the rest, compare the ends }
 | |
|         if HelpWilds = '' then Exit; { '*' is the last letter }
 | |
|         { check the rest for equal Length and no '?' }
 | |
|         for I := 0 to LenHelpWilds - 1 do begin
 | |
|           if (HelpWilds[LenHelpWilds - I] <> InputStr[MaxInputWord - I]) and
 | |
|             (HelpWilds[LenHelpWilds - I]<> '?') then
 | |
|           begin
 | |
|             Result := False;
 | |
|             Exit;
 | |
|           end;
 | |
|         end;
 | |
|         Exit;
 | |
|       end;
 | |
|       { handle all to the next '*' }
 | |
|       Inc(CWild, 1 + LenHelpWilds);
 | |
|       I := FindPart(HelpWilds, Copy(InputStr, CInputWord, MaxInt));
 | |
|       if I= 0 then begin
 | |
|         Result := False;
 | |
|         Exit;
 | |
|       end;
 | |
|       CInputWord := I + LenHelpWilds;
 | |
|       Continue;
 | |
|     end;
 | |
|     Result := False;
 | |
|     Exit;
 | |
|   until (CInputWord > MaxInputWord) or (CWild > MaxWilds);
 | |
|   { no completed evaluation }
 | |
|   if CInputWord <= MaxInputWord then Result := False;
 | |
|   if (CWild <= MaxWilds) and (Wilds[MaxWilds] <> '*') then Result := False;
 | |
| end;
 | |
| 
 | |
| { TFormPlacement }
 | |
| 
 | |
| constructor TFormPlacement.Create(AOwner: TComponent);
 | |
| begin
 | |
|   inherited Create(AOwner);
 | |
|   FActive := True;
 | |
|   if (AOwner is TForm) then
 | |
|     FOptions := [fpState, fpPosition]
 | |
|   else
 | |
|     FOptions := [];
 | |
|   FLinks := TList.Create;
 | |
| end;
 | |
| 
 | |
| destructor TFormPlacement.Destroy;
 | |
| begin
 | |
|   IniFree;
 | |
|   while FLinks.Count > 0 do
 | |
|     RemoveLink(TiniLink(FLinks.Last));
 | |
|   FreeAndNil(FLinks);
 | |
|   if not (csDesigning in ComponentState) then
 | |
|     RestoreEvents;
 | |
|   inherited Destroy;
 | |
| end;
 | |
| 
 | |
| procedure TFormPlacement.Loaded;
 | |
| var
 | |
|   IsLoading: Boolean;
 | |
| begin
 | |
|   IsLoading := csLoading in ComponentState;
 | |
|   inherited Loaded;
 | |
|   if not (csDesigning in ComponentState) then
 | |
|     begin
 | |
|     if IsLoading then
 | |
|       SetEvents;
 | |
|     end;
 | |
| end;
 | |
| 
 | |
| procedure TFormPlacement.AddLink(ALink: TIniLink);
 | |
| begin
 | |
|   FLinks.Add(ALink);
 | |
|   ALink.FStorage := Self;
 | |
| end;
 | |
| 
 | |
| procedure TFormPlacement.NotifyLinks(Operation: TPlacementOperation);
 | |
| var
 | |
|   I: Integer;
 | |
| begin
 | |
|   for I := 0 to FLinks.Count - 1 do
 | |
|     with TIniLink(FLinks[I]) do
 | |
|       case Operation of
 | |
|         poSave: SaveToIni;
 | |
|         poRestore: LoadFromIni;
 | |
|       end;
 | |
| end;
 | |
| 
 | |
| procedure TFormPlacement.RemoveLink(ALink: TIniLink);
 | |
| begin
 | |
|   ALink.FStorage := nil;
 | |
|   FLinks.Remove(ALink);
 | |
| end;
 | |
| 
 | |
| function TFormPlacement.GetForm: TForm;
 | |
| begin
 | |
|   if (Owner is TCustomForm) then
 | |
|     Result := TForm(Owner as TCustomForm)
 | |
|   else
 | |
|     Result := nil;
 | |
| end;
 | |
| 
 | |
| procedure TFormPlacement.SetEvents;
 | |
| begin
 | |
|   if (Owner is TCustomForm) then
 | |
|     begin
 | |
|     with TForm(Form) do
 | |
|       begin
 | |
|       FSaveFormShow := OnShow;
 | |
|       OnShow := @FormShow;
 | |
|       FSaveFormCloseQuery := OnCloseQuery;
 | |
|       OnCloseQuery := @FormCloseQuery;
 | |
|       FSaveFormDestroy := OnDestroy;
 | |
|       OnDestroy := @FormDestroy;
 | |
|       end;
 | |
|     end;
 | |
| end;
 | |
| 
 | |
| procedure TFormPlacement.RestoreEvents;
 | |
| begin
 | |
|   if (Owner <> nil) and (Owner is TCustomForm) then
 | |
|     with TForm(Form) do
 | |
|       begin
 | |
|       OnShow := FSaveFormShow;
 | |
|       OnCloseQuery := FSaveFormCloseQuery;
 | |
|       OnDestroy := FSaveFormDestroy;
 | |
|       end;
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure TFormPlacement.FormShow(Sender: TObject);
 | |
| begin
 | |
|   if Active then
 | |
|     try
 | |
|       RestoreFormPlacement;
 | |
|     except
 | |
|       Application.HandleException(Self);
 | |
|     end;
 | |
|   if Assigned(FSaveFormShow) then FSaveFormShow(Sender);
 | |
| end;
 | |
| 
 | |
| procedure TFormPlacement.FormCloseQuery(Sender: TObject; var CanClose: Boolean);
 | |
| begin
 | |
|   if Assigned(FSaveFormCloseQuery) then
 | |
|     FSaveFormCloseQuery(Sender, CanClose);
 | |
|   if CanClose and Active and (Owner is TCustomForm) and (Form.Handle <> 0) then
 | |
|     try
 | |
|       SaveFormPlacement;
 | |
|     except
 | |
|       Application.HandleException(Self);
 | |
|     end;
 | |
| end;
 | |
| 
 | |
| procedure TFormPlacement.FormDestroy(Sender: TObject);
 | |
| begin
 | |
|   if Active and not FSaved then
 | |
|     begin
 | |
|     FDestroying := True;
 | |
|     try
 | |
|       SaveFormPlacement;
 | |
|     except
 | |
|       Application.HandleException(Self);
 | |
|     end;
 | |
|     FDestroying := False;
 | |
|     end;
 | |
|   if Assigned(FSaveFormDestroy) then
 | |
|     FSaveFormDestroy(Sender);
 | |
| end;
 | |
| 
 | |
| 
 | |
| 
 | |
| function TFormPlacement.GetIniFileName: string;
 | |
| begin
 | |
|   Result := FIniFileName;
 | |
|   if (Result = '') and not (csDesigning in ComponentState) then
 | |
|     Result := GetDefaultIniName;
 | |
| end;
 | |
| 
 | |
| procedure TFormPlacement.SetIniFileName(const Value: string);
 | |
| begin
 | |
|   FIniFileName:=Value;
 | |
| end;
 | |
| 
 | |
| function TFormPlacement.GetIniSection: string;
 | |
| begin
 | |
|   Result := FIniSection;
 | |
|   if (Result = '') and not (csDesigning in ComponentState) then
 | |
|     Result := GetDefaultSection(Owner);
 | |
| end;
 | |
| 
 | |
| procedure TFormPlacement.SetIniSection(const Value: string);
 | |
| begin
 | |
|   FIniSection:=Value;
 | |
| end;
 | |
| 
 | |
| procedure TFormPlacement.Save;
 | |
| begin
 | |
|   if Assigned(FOnSavePlacement) then
 | |
|     FOnSavePlacement(Self);
 | |
| end;
 | |
| 
 | |
| procedure TFormPlacement.Restore;
 | |
| begin
 | |
|   if Assigned(FOnRestorePlacement) then FOnRestorePlacement(Self);
 | |
| end;
 | |
| 
 | |
| procedure TFormPlacement.SavePlacement;
 | |
| begin
 | |
|   if (Owner is TCustomForm) then
 | |
|     begin
 | |
|     if (Options * [fpState, fpPosition] <> []) then
 | |
|       begin
 | |
|       WriteFormPlacement(Form, IniFile, IniSection);
 | |
|       IniFile.WriteBool(IniSection, siVisible, FDestroying);
 | |
|       end;
 | |
|     if (fpActiveControl in Options) and (Form.ActiveControl <> nil) then
 | |
|       IniFile.WriteString(IniSection, siActiveCtrl, Form.ActiveControl.Name);
 | |
|     end;
 | |
|   NotifyLinks(poSave);
 | |
| end;
 | |
| 
 | |
| procedure TFormPlacement.RestorePlacement;
 | |
| begin
 | |
|   if Owner is TCustomForm then
 | |
|     ReadFormPlacement(Form, IniFile, IniSection, fpState in Options, fpPosition in Options);
 | |
|   NotifyLinks(poRestore);
 | |
| end;
 | |
| 
 | |
| procedure TFormPlacement.IniNeeded(ReadOnly: Boolean);
 | |
| begin
 | |
|   if ReadOnly then ;
 | |
|   if IniFile = nil then
 | |
|     FIniFile := TIniFile.Create(UTF8ToSys(IniFileName));
 | |
| end;
 | |
| 
 | |
| procedure TFormPlacement.IniFree;
 | |
| begin
 | |
|   if IniFile <> nil then
 | |
|     FreeAndNil(FIniFile);
 | |
| end;
 | |
| 
 | |
| function TFormPlacement.DoReadString(const Section, Ident,
 | |
|   Default: string): string;
 | |
| begin
 | |
|   if IniFile <> nil then
 | |
|     Result := IniFile.ReadString(Section, Ident, Default)
 | |
|   else
 | |
|     begin
 | |
|     IniNeeded(True);
 | |
|     try
 | |
|       Result := Inifile.ReadString(Section, Ident, Default);
 | |
|     finally
 | |
|       IniFree;
 | |
|     end;
 | |
|   end;
 | |
| end;
 | |
| 
 | |
| function TFormPlacement.ReadString(const Ident, Default: string): string;
 | |
| begin
 | |
|   Result := DoReadString(IniSection, Ident, Default);
 | |
| end;
 | |
| 
 | |
| procedure TFormPlacement.DoWriteString(const Section, Ident, Value: string);
 | |
| begin
 | |
|   if IniFile<>nil then
 | |
|     IniFile.WriteString(Section, Ident, Value)
 | |
|   else begin
 | |
|     IniNeeded(False);
 | |
|     try
 | |
|       IniFile.WriteString(Section, Ident, Value);
 | |
|     finally
 | |
|       IniFree;
 | |
|     end;
 | |
|   end;
 | |
| end;
 | |
| 
 | |
| procedure TFormPlacement.WriteString(const Ident, Value: string);
 | |
| begin
 | |
|   DoWriteString(IniSection, Ident, Value);
 | |
| end;
 | |
| 
 | |
| function TFormPlacement.ReadInteger(const Ident: string; Default: Longint): Longint;
 | |
| begin
 | |
|   if (IniFile<>nil) then
 | |
|     Result := IniFile.ReadInteger(IniSection, Ident, Default)
 | |
|   else
 | |
|     begin
 | |
|     IniNeeded(True);
 | |
|     try
 | |
|       Result := Inifile.ReadInteger(IniSection, Ident, Default);
 | |
|     finally
 | |
|       IniFree;
 | |
|     end;
 | |
|   end;
 | |
| end;
 | |
| 
 | |
| procedure TFormPlacement.WriteInteger(const Ident: string; Value: Longint);
 | |
| begin
 | |
|   if IniFile<>nil then
 | |
|     IniFile.WriteInteger(IniSection, Ident, Value)
 | |
|   else begin
 | |
|     IniNeeded(False);
 | |
|     try
 | |
|       Inifile.WriteInteger(IniSection, Ident, Value);
 | |
|     finally
 | |
|       IniFree;
 | |
|     end;
 | |
|   end;
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure TFormPlacement.EraseSections;
 | |
| var
 | |
|   Lines: TStrings;
 | |
|   I: Integer;
 | |
| begin
 | |
|   if IniFile= nil then begin
 | |
|     IniNeeded(False);
 | |
|     try
 | |
|       Lines := TStringList.Create;
 | |
|       try
 | |
|         Inifile.ReadSections(Lines);
 | |
|         for I := 0 to Lines.Count - 1 do begin
 | |
|           if (Lines[I] = IniSection) or
 | |
|             (IsWild(Lines[I], IniSection + '.*', False) or
 | |
|             IsWild(Lines[I], IniSection + '\*', False)) then
 | |
|             Inifile.EraseSection(Lines[I]);
 | |
|         end;
 | |
|       finally
 | |
|         Lines.Free;
 | |
|       end;
 | |
|     finally
 | |
|       IniFree;
 | |
|     end;
 | |
|   end;
 | |
| end;
 | |
| 
 | |
| procedure TFormPlacement.SaveFormPlacement;
 | |
| begin
 | |
|   if FRestored or not Active then begin
 | |
|     IniNeeded(False);
 | |
|     try
 | |
|       WriteInteger(siVersion, FVersion);
 | |
|       SavePlacement;
 | |
|       Save;
 | |
|       FSaved := True;
 | |
|     finally
 | |
|       IniFree;
 | |
|     end;
 | |
|   end;
 | |
| end;
 | |
| 
 | |
| procedure TFormPlacement.RestoreFormPlacement;
 | |
| var
 | |
|   cActive: TComponent;
 | |
| begin
 | |
|   FSaved := False;
 | |
|   IniNeeded(True);
 | |
|   try
 | |
|     if ReadInteger(siVersion, 0) >= FVersion then begin
 | |
|       RestorePlacement;
 | |
|       FRestored := True;
 | |
|       Restore;
 | |
|       if (fpActiveControl in Options) and (Owner is TCustomForm) then
 | |
|         begin
 | |
|         cActive := Form.FindComponent(Inifile.ReadString(IniSection, siActiveCtrl, ''));
 | |
|         if (cActive <> nil) and (cActive is TWinControl) and
 | |
|           TWinControl(cActive).CanFocus then
 | |
|             Form.ActiveControl := TWinControl(cActive);
 | |
|       end;
 | |
|     end;
 | |
|     FRestored := True;
 | |
|   finally
 | |
|     IniFree;
 | |
|   end;
 | |
| end;
 | |
| 
 | |
| { TFormStorage }
 | |
| 
 | |
| constructor TFormStorage.Create(AOwner: TComponent);
 | |
| begin
 | |
|   inherited Create(AOwner);
 | |
|   FStoredProps:=TStringList.Create;
 | |
|   FStoredValues:=TStoredValues.Create(Self);
 | |
|   FStoredValues.Storage := Self;
 | |
| end;
 | |
| 
 | |
| destructor TFormStorage.Destroy;
 | |
| begin
 | |
|   FreeAndNil(FStoredValues);
 | |
|   FreeAndNil(FStoredProps);
 | |
|   inherited Destroy;
 | |
| end;
 | |
| 
 | |
| procedure TFormStorage.SetNotification;
 | |
| var
 | |
|   I: Integer;
 | |
|   Component: TComponent;
 | |
| begin
 | |
|   for I := FStoredProps.Count - 1 downto 0 do begin
 | |
|     Component := TComponent(FStoredProps.Objects[I]);
 | |
|     if Component <> nil then Component.FreeNotification(Self);
 | |
|   end;
 | |
| end;
 | |
| 
 | |
| procedure TFormStorage.SetStoredProps(Value: TStrings);
 | |
| begin
 | |
|   FStoredProps.Assign(Value);
 | |
|   SetNotification;
 | |
| end;
 | |
| 
 | |
| procedure TFormStorage.SetStoredValues(Value: TStoredValues);
 | |
| begin
 | |
|   FStoredValues.Assign(Value);
 | |
| end;
 | |
|   
 | |
| function TFormStorage.GetStoredValue(const AName: string): TStoredType;
 | |
| begin
 | |
|   Result := StoredValues.StoredValue[AName];
 | |
| end;
 | |
|     
 | |
| procedure TFormStorage.SetStoredValue(const AName: string; Value: TStoredType);
 | |
| begin
 | |
|   StoredValues.StoredValue[AName] := Value;
 | |
| end;
 | |
|       
 | |
| 
 | |
| procedure TFormStorage.Loaded;
 | |
| begin
 | |
|   inherited Loaded;
 | |
|   UpdateStoredList(Owner, FStoredProps, True);
 | |
| end;
 | |
| 
 | |
| procedure TFormStorage.WriteState(Writer: TWriter);
 | |
| begin
 | |
|   UpdateStoredList(Owner, FStoredProps, False);
 | |
|   inherited WriteState(Writer);
 | |
| end;
 | |
| 
 | |
| procedure TFormStorage.Notification(AComponent: TComponent; Operation: TOperation);
 | |
| var
 | |
|   I: Integer;
 | |
|   Component: TComponent;
 | |
| begin
 | |
|   inherited Notification(AComponent, Operation);
 | |
|   if not (csDestroying in ComponentState) and (Operation = opRemove) and
 | |
|     (FStoredProps <> nil) then
 | |
|     for I := FStoredProps.Count - 1 downto 0 do begin
 | |
|       Component := TComponent(FStoredProps.Objects[I]);
 | |
|       if Component = AComponent then FStoredProps.Delete(I);
 | |
|     end;
 | |
| end;
 | |
| 
 | |
| procedure TFormStorage.SaveProperties;
 | |
| begin
 | |
|   with TPropsStorage.Create do
 | |
|   try
 | |
|     Section := IniSection;
 | |
|     OnWriteString := @DoWriteString;
 | |
|     OnEraseSection := @IniFile.EraseSection;
 | |
|     StoreObjectsProps(Owner, FStoredProps);
 | |
|   finally
 | |
|     Free;
 | |
|   end;
 | |
| end;
 | |
| 
 | |
| procedure TFormStorage.RestoreProperties;
 | |
| begin
 | |
|   with TPropsStorage.Create do
 | |
|   try
 | |
|     Section := IniSection;
 | |
|     OnReadString := @DoReadString;
 | |
|     try
 | |
|       LoadObjectsProps(Owner, FStoredProps);
 | |
|     except
 | |
|       { ignore any exceptions }
 | |
|     end;
 | |
|   finally
 | |
|     Free;
 | |
|   end;
 | |
| end;
 | |
| 
 | |
| procedure TFormStorage.SavePlacement;
 | |
| begin
 | |
|   inherited SavePlacement;
 | |
|   SaveProperties;
 | |
| {$IFDEF RX_D3}
 | |
|   StoredValues.SaveValues;
 | |
| {$ENDIF}
 | |
| end;
 | |
| 
 | |
| procedure TFormStorage.RestorePlacement;
 | |
| begin
 | |
|   inherited RestorePlacement;
 | |
|   FRestored := True;
 | |
|   RestoreProperties;
 | |
| {$IFDEF RX_D3}
 | |
|   StoredValues.RestoreValues;
 | |
| {$ENDIF}
 | |
| end;
 | |
| 
 | |
| { TIniLink }
 | |
| 
 | |
| destructor TIniLink.Destroy;
 | |
| begin
 | |
|   FOnSave := nil;
 | |
|   FOnLoad := nil;
 | |
|   SetStorage(nil);
 | |
|   inherited Destroy;
 | |
| end;
 | |
| 
 | |
| function TIniLink.GetIniObject: TCustomInifile;
 | |
| begin
 | |
|   if Assigned(FStorage) then
 | |
|     Result := FStorage.IniFile
 | |
|   else Result := nil;
 | |
| end;
 | |
| 
 | |
| function TIniLink.GetRootSection: string;
 | |
| begin
 | |
|   if Assigned(FStorage) then
 | |
|      Result := FStorage.FIniSection
 | |
|   else
 | |
|     Result := '';
 | |
|   if Result <> '' then
 | |
|     Result := Result + '\';
 | |
| end;
 | |
| 
 | |
| procedure TIniLink.SetStorage(Value: TFormPlacement);
 | |
| begin
 | |
|   if FStorage <> Value then
 | |
|     begin
 | |
|     if FStorage <> nil then
 | |
|       FStorage.RemoveLink(Self);
 | |
|     if Value <> nil then
 | |
|       Value.AddLink(Self);
 | |
|   end;
 | |
| end;
 | |
| 
 | |
| procedure TIniLink.SaveToIni;
 | |
| begin
 | |
|   if Assigned(FOnSave) then FOnSave(Self);
 | |
| end;
 | |
| 
 | |
| procedure TIniLink.LoadFromIni;
 | |
| begin
 | |
|   if Assigned(FOnLoad) then FOnLoad(Self);
 | |
| end;
 | |
| 
 | |
| { TStoredValue }
 | |
| 
 | |
| constructor TStoredValue.Create(ACollection: TCollection);
 | |
| begin
 | |
|   inherited Create(ACollection);
 | |
| {$ifdef storevariant}
 | |
|   FValue := Unassigned;
 | |
| {$else}
 | |
|   FValue:='';
 | |
| {$endif}
 | |
| end;
 | |
| 
 | |
| procedure TStoredValue.Assign(Source: TPersistent);
 | |
| begin
 | |
|   if (Source is TStoredValue) and (Source <> nil) then
 | |
|     begin
 | |
| {$ifdef storevariant}
 | |
|     if VarIsEmpty(TStoredValue(Source).FValue) then
 | |
|       Clear
 | |
|     else
 | |
| {$endif}
 | |
|       Value := TStoredValue(Source).FValue;
 | |
|     Name := TStoredValue(Source).Name;
 | |
|     KeyString := TStoredValue(Source).KeyString;
 | |
|     end;
 | |
| end;
 | |
| 
 | |
| function TStoredValue.GetDisplayName: string;
 | |
| begin
 | |
|   if FName = '' then
 | |
|     Result := inherited GetDisplayName
 | |
|   else
 | |
|     Result := FName;
 | |
| end;
 | |
| 
 | |
| procedure TStoredValue.SetDisplayName(const Value: string);
 | |
| begin
 | |
|   if (Value <> '') and (AnsiCompareText(Value, FName) <> 0) and
 | |
|     (Collection is TStoredValues) and (TStoredValues(Collection).IndexOf(Value) >= 0) then
 | |
|     raise Exception.Create(SDuplicateString);
 | |
|   FName := Value;
 | |
|   inherited;
 | |
| end;
 | |
| 
 | |
| function TStoredValue.GetStoredValues: TStoredValues;
 | |
| begin
 | |
|   if Collection is TStoredValues then
 | |
|     Result := TStoredValues(Collection)
 | |
|   else
 | |
|     Result := nil;
 | |
| end;
 | |
| 
 | |
| procedure TStoredValue.Clear;
 | |
| begin
 | |
| {$ifdef storevariant}
 | |
|   FValue := Unassigned;
 | |
| {$else}
 | |
|   FValue := '';
 | |
| {$endif}
 | |
| end;
 | |
| 
 | |
| function TStoredValue.IsValueStored: Boolean;
 | |
| begin
 | |
| {$ifdef storevariant}
 | |
|   Result := not VarIsEmpty(FValue);
 | |
| {$else}
 | |
|   Result := (FValue<>'');
 | |
| {$endif}
 | |
| end;
 | |
| 
 | |
| procedure TStoredValue.Save;
 | |
| var
 | |
|   SaveValue: TStoredType;
 | |
|   SaveStrValue: string;
 | |
| begin
 | |
|   SaveValue := Value;
 | |
|   if Assigned(FOnSave) then
 | |
|     FOnSave(Self, SaveValue);
 | |
| {$ifdef storevariant}
 | |
|   SaveStrValue := VarToStr(SaveValue);
 | |
| {$else}
 | |
|   SaveStrValue := SaveValue;
 | |
| {$endif}
 | |
|   if KeyString <> '' then
 | |
|     SaveStrValue := XorEncode(KeyString, SaveStrValue);
 | |
|   StoredValues.Storage.WriteString(Name, SaveStrValue);
 | |
| end;
 | |
| 
 | |
| procedure TStoredValue.Restore;
 | |
| var
 | |
|   RestoreValue: TStoredType;
 | |
|   RestoreStrValue, DefaultStrValue: string;
 | |
| begin
 | |
| {$ifdef storevariant}
 | |
|   DefaultStrValue := VarToStr(Value);
 | |
| {$else}
 | |
|   DefaultStrValue := Value;
 | |
| {$endif}
 | |
|   if KeyString <> '' then
 | |
|     DefaultStrValue := XorEncode(KeyString, DefaultStrValue);
 | |
|   RestoreStrValue := StoredValues.Storage.ReadString(Name, DefaultStrValue);
 | |
|   if KeyString <> '' then
 | |
|     RestoreStrValue := XorDecode(KeyString, RestoreStrValue);
 | |
|   RestoreValue := RestoreStrValue;
 | |
|   if Assigned(FOnRestore) then
 | |
|     FOnRestore(Self, RestoreValue);
 | |
|   Value := RestoreValue;
 | |
| end;
 | |
| 
 | |
| { TStoredValues }
 | |
| 
 | |
| constructor TStoredValues.Create(AOwner: TPersistent);
 | |
| begin
 | |
|   inherited Create(AOwner, TStoredValue);
 | |
| end;
 | |
| 
 | |
| function TStoredValues.IndexOf(const AName: string): Integer;
 | |
| begin
 | |
|   for Result := 0 to Count - 1 do
 | |
|     if AnsiCompareText(Items[Result].Name, AName) = 0 then Exit;
 | |
|   Result := -1;
 | |
| end;
 | |
| 
 | |
| function TStoredValues.GetItem(Index: Integer): TStoredValue;
 | |
| begin
 | |
|   Result := TStoredValue(inherited Items[Index]);
 | |
| end;
 | |
| 
 | |
| procedure TStoredValues.SetItem(Index: Integer; StoredValue: TStoredValue);
 | |
| begin
 | |
|   inherited SetItem(Index, TCollectionItem(StoredValue));
 | |
| end;
 | |
| 
 | |
| function TStoredValues.GetStoredValue(const AName: string): TStoredType;
 | |
| var
 | |
|   AStoredValue: TStoredValue;
 | |
| begin
 | |
|   AStoredValue := GetValue(AName);
 | |
|   if AStoredValue = nil then
 | |
| {$ifdef storevariant}
 | |
|     Result := Null
 | |
| {$else}
 | |
|     Result := ''
 | |
| {$endif}
 | |
|   else
 | |
|     Result := AStoredValue.Value;
 | |
| end;
 | |
| 
 | |
| procedure TStoredValues.SetStoredValue(const AName: string; Value: TStoredType);
 | |
| var
 | |
|   AStoredValue: TStoredValue;
 | |
| begin
 | |
|   AStoredValue := GetValue(AName);
 | |
|   if AStoredValue = nil then begin
 | |
|     AStoredValue := TStoredValue(Add);
 | |
|     AStoredValue.Name := AName;
 | |
|     AStoredValue.Value := Value;
 | |
|   end
 | |
|   else AStoredValue.Value := Value;
 | |
| end;
 | |
| 
 | |
| function TStoredValues.GetValue(const AName: string): TStoredValue;
 | |
| var
 | |
|   I: Integer;
 | |
| begin
 | |
|   I := IndexOf(AName);
 | |
|   if I < 0 then
 | |
|     Result := nil
 | |
|   else
 | |
|     Result := Items[I];
 | |
| end;
 | |
| 
 | |
| procedure TStoredValues.SetValue(const AName: string; StoredValue: TStoredValue);
 | |
| var
 | |
|   I: Integer;
 | |
| begin
 | |
|   I := IndexOf(AName);
 | |
|   if I >= 0 then
 | |
|     Items[I].Assign(StoredValue);
 | |
| end;
 | |
| 
 | |
| procedure TStoredValues.SaveValues;
 | |
| var
 | |
|   I: Integer;
 | |
| begin
 | |
|   for I := 0 to Count - 1 do
 | |
|     Items[I].Save;
 | |
| end;
 | |
| 
 | |
| procedure TStoredValues.RestoreValues;
 | |
| var
 | |
|   I: Integer;
 | |
| begin
 | |
|   for I := 0 to Count - 1 do
 | |
|     Items[I].Restore;
 | |
| end;
 | |
| 
 | |
| end.
 | 
