📄 jvdockinfo.pas
字号:
{-----------------------------------------------------------------------------
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/MPL-1.1.html
Software distributed under the License is distributed on an "AS IS" basis,
WITHOUT WARRANTY OF ANY KIND, either expressed or implied. See the License for
the specific language governing rights and limitations under the License.
The Original Code is: JvDockInfo.pas, released on 2003-12-31.
The Initial Developer of the Original Code is luxiaoban.
Portions created by luxiaoban are Copyright (C) 2002,2003 luxiaoban.
All Rights Reserved.
Contributor(s):
You may retrieve the latest version of this file at the Project JEDI's JVCL home page,
located at http://jvcl.sourceforge.net
Known Issues:
-----------------------------------------------------------------------------}
// $Id: JvDockInfo.pas,v 1.39 2005/02/24 22:13:55 remkobonte Exp $
unit JvDockInfo;
{$I jvcl.inc}
interface
uses
{$IFDEF USEJVCL}
{$IFDEF UNITVERSIONING}
JclUnitVersioning,
{$ENDIF UNITVERSIONING}
{$ENDIF USEJVCL}
Windows, IniFiles, Registry, Classes, Controls, Forms,
{$IFDEF USEJVCL}
JvAppStorage,
{$ENDIF USEJVCL}
JvDockControlForm, JvDockSupportClass, JvDockSupportProc;
type
TJvDockInfoTree = class;
TJvDockFormStyle = (dsNormal, dsConjoin, dsTab, dsDockPanel);
TJvDockInfoZone = class(TJvDockBaseZone)
private
FDockFormName: string;
FParentName: string;
FDockRect: TRect;
FLastDockSiteName: string;
FUnDockLeft: Integer;
FUnDockTop: Integer;
FLRDockWidth: Integer;
FTBDockHeight: Integer;
FUnDockWidth: Integer;
FUnDockHeight: Integer;
FVSPaneWidth: Integer;
FVisible: Boolean;
FBorderStyle: TBorderStyle;
FFormStyle: TFormStyle;
FWindowState: TWindowState;
FCanDocked: Boolean;
FEachOtherDocked: Boolean;
FLeftDocked: Boolean;
FTopDocked: Boolean;
FRightDocked: Boolean;
FBottomDocked: Boolean;
FCustomDocked: Boolean; {NEW! Contains custom dock panel! }
FDockFormStyle: TJvDockFormStyle;
FDockClientData: string;
FDockControl: TWinControl;
function GetChildControlCount: Integer;
public
procedure SetDockInfoFromControlToNode(Control: TControl); virtual;
procedure SetDockInfoFromNodeToControl(Control: TControl); virtual;
procedure SetDockInfoFromDockControlToNode(DockControl: TJvDockBaseControl); virtual;
procedure SetDockInfoFromNodeToDockControl(DockControl: TJvDockBaseControl); virtual;
property DockFormName: string read FDockFormName write FDockFormName;
property ParentName: string read FParentName write FParentName;
property DockRect: TRect read FDockRect write FDockRect;
property LastDockSiteName: string read FLastDockSiteName write FLastDockSiteName;
property UnDockLeft: Integer read FUnDockLeft write FUnDockLeft;
property UnDockTop: Integer read FUnDockTop write FUnDockTop;
property LRDockWidth: Integer read FLRDockWidth write FLRDockWidth;
property TBDockHeight: Integer read FTBDockHeight write FTBDockHeight;
property UnDockWidth: Integer read FUnDockWidth write FUnDockWidth;
property UnDockHeight: Integer read FUnDockHeight write FUnDockHeight;
property VSPaneWidth: Integer read FVSPaneWidth write FVSPaneWidth;
property BorderStyle: TBorderStyle read FBorderStyle write FBorderStyle;
property FormStyle: TFormStyle read FFormStyle write FFormStyle;
property WindowState: TWindowState read FWindowState write FWindowState;
property Visible: Boolean read FVisible write FVisible;
property CanDocked: Boolean read FCanDocked write FCanDocked;
property EachOtherDocked: Boolean read FEachOtherDocked write FEachOtherDocked;
property LeftDocked: Boolean read FLeftDocked write FLeftDocked;
property TopDocked: Boolean read FTopDocked write FTopDocked;
property RightDocked: Boolean read FRightDocked write FRightDocked;
property BottomDocked: Boolean read FBottomDocked write FBottomDocked;
property CustomDocked: Boolean read FCustomDocked write FCustomDocked; {NEW! Contains custom dock panel! }
property DockFormStyle: TJvDockFormStyle read FDockFormStyle write FDockFormStyle;
property DockClientData: string read FDockClientData write FDockClientData;
property DockControl: TWinControl read FDockControl write FDockControl;
end;
// TJvDockInfoStyle enumerates the mode that is used when you call
// TJvDockInfoTree.ScanTreeZone. This is a part of the code used to
// implement persistence (loading and saving of docking layouts).
//
TJvDockInfoStyle =
(isNone, { No mode set }
{$IFDEF USEJVCL}
isJVCLReadInfo, { Mode for this scan is JVCL App Storage Load }
isJVCLWriteInfo, { Mode for this scan is JVCL App Storage Save }
{$ENDIF USEJVCL}
isReadFileInfo, { Mode for this scan is Text File Read. Backwards compatible. }
isWriteFileInfo, { Mode for this scan is Text File Write. Backwards compatible. }
isReadRegInfo, { Mode for this scan is registry Read }
isWriteRegInfo); { Mode for this scan is registry Write }
{ JvDockInfoTree contains information about the docking tree. It is created
as part of the persistence framework for the JvDocking components. In order
to save or load docking layout you must create one of these objects and use
it to store the information about the set of docked forms being managed by
JvDocking. }
TJvDockInfoTree = class(TJvDockBaseTree)
private
{$IFDEF USEJVCL}
FAppStorage: TJvCustomAppStorage;
FAppStoragePath: string;
{$ENDIF USEJVCL}
FDockInfoIni: TCustomIniFile;
FDockInfoReg: TRegistry;
FRegName: string;
FJvDockInfoStyle: TJvDockInfoStyle; { Which action to do when doing a ScanTreeZone() recursive operation over the document tree. }
FDataStream: TMemoryStream;
function FindDockForm(const FormName: string): TCustomForm;
function CreateHostControl(ATreeZone: TJvDockInfoZone): TWinControl;
protected
procedure ScanTreeZone(TreeZone: TJvDockBaseZone); override;
{$IFDEF USEJVCL}
procedure CreateZoneAndAddInfoFromAppStorage; virtual;
{$ENDIF USEJVCL}
procedure CreateZoneAndAddInfoFromIni; virtual;
procedure CreateZoneAndAddInfoFromReg; virtual;
procedure SetDockControlInfo(ATreeZone: TJvDockInfoZone); virtual;
public
constructor Create(TreeZone: TJvDockTreeZoneClass); override;
destructor Destroy; override;
// This is the most important function in this class, it basically
// puts the important information from the application form into this
// object.
procedure CreateZoneAndAddInfoFromApp(Control: TControl); virtual;
{$IFDEF USEJVCL}
procedure ReadInfoFromAppStorage;
procedure WriteInfoToAppStorage;
property AppStorage: TJvCustomAppStorage read FAppStorage write FAppStorage;
property AppStoragePath: string read FAppStoragePath write FAppStoragePath;
{$ENDIF USEJVCL}
procedure ReadInfoFromIni;
procedure ReadInfoFromReg(const RegName: string);
procedure WriteInfoToIni;
procedure WriteInfoToReg(const RegName: string);
property DockInfoIni: TCustomIniFile read FDockInfoIni write FDockInfoIni;
property DockInfoReg: TRegistry read FDockInfoReg write FDockInfoReg;
end;
{$IFDEF USEJVCL}
{$IFDEF UNITVERSIONING}
const
UnitVersioning: TUnitVersionInfo = (
RCSfile: '$RCSfile: JvDockInfo.pas,v $';
Revision: '$Revision: 1.39 $';
Date: '$Date: 2005/02/24 22:13:55 $';
LogPath: 'JVCL\run'
);
{$ENDIF UNITVERSIONING}
{$ENDIF USEJVCL}
implementation
uses
SysUtils,
JvDockGlobals, JvDockVSNetStyle;
//=== Local procedures =======================================================
function FindDockForm(const FormName: string): TCustomForm;
begin
if Pos(RsDockJvDockInfoSplitter, FormName) > 0 then
Result := nil
else
Result := JvDockFindDockFormWithName(FormName);
end;
function FindDockPanel(const ControlName: string): TWinControl;
var
Index: Word;
DockServer: TJvDockServer;
begin
Result := nil;
Index := Pos(RsDockJvDockInfoSplitter, ControlName);
if Index = 0 then
Exit;
Result := FindDockForm(Copy(ControlName, 1, Index - 1));
if Result <> nil then
begin
DockServer := FindDockServer(Result);
if DockServer <> nil then
with DockServer do
begin
if Pos('TopDockPanel', ControlName) > Index then
Result := TopDockPanel
else
if Pos('LeftDockPanel', ControlName) > Index then
Result := LeftDockPanel
else
if Pos('BottomDockPanel', ControlName) > Index then
Result := BottomDockPanel
else
if Pos('RightDockPanel', ControlName) > Index then
Result := RightDockPanel;
if (Result <> nil) and (Pos('PopupPanel', ControlName) > 20) then
Result := TJvDockVSNETPanel(Result).VSChannel.VSPopupPanel;
end;
end;
end;
function FindDockHost(const ControlName: string): TWinControl;
begin
Result := FindDockForm(ControlName);
if Result = nil then
Result := FindDockPanel(ControlName);
end;
//=== { TJvDockInfoTree } ====================================================
constructor TJvDockInfoTree.Create(TreeZone: TJvDockTreeZoneClass);
begin
inherited Create(TreeZone);
{$IFNDEF USEJVCL}
FDockInfoIni := nil;
FDockInfoReg := nil;
{$ENDIF !USEJVCL}
FJvDockInfoStyle := isNone;
FDataStream := TMemoryStream.Create;
end;
destructor TJvDockInfoTree.Destroy;
begin
inherited Destroy;
FreeAndNil(FDataStream);
end;
{ Create an TJvDockConjoinHostForm or TJvDockTabHostForm when restoring a docking layout }
function TJvDockInfoTree.CreateHostControl(ATreeZone: TJvDockInfoZone): TWinControl;
var
Form: TForm;
ADockClient: TJvDockClient;
begin
{ The dockinfo data that is saved, contains names of the values of ChildZone.DockControl
Thus on loading it can be that no form with that DockControl name can be found;
then DockControl will be nil
}
Result := nil;
case ATreeZone.DockFormStyle of
dsConjoin:
if Assigned(TJvDockInfoZone(ATreeZone.ChildZone).DockControl) then
begin
Form := TJvDockConjoinHostForm.Create(Application);
ADockClient := FindDockClient(TJvDockInfoZone(ATreeZone.ChildZone).DockControl);
Result := ADockClient.CreateConjoinPanelClass(Form).Parent;
end;
dsTab:
if Assigned(TJvDockInfoZone(ATreeZone.ChildZone).DockControl) then
begin
Form := TJvDockTabHostForm.Create(Application);
ADockClient := FindDockClient(TJvDockInfoZone(ATreeZone.ChildZone).DockControl);
Result := ADockClient.CreateTabDockClass(Form).Parent;
end;
end;
if Result <> nil then
Result.Name := ATreeZone.DockFormName;
end;
// CreateZoneAndAddInfoFromApp
//
// Control: TControl - note this is probably actually a TForm
// descendant, since this library only supports form docking.
//
// This is the most important function in this class, it basically
// puts the important information from the application form into this
// object.
//
// This is used to take a form that is docked somewhere and extract all the
// docking layout information contained inside it, and add it to this JvDockInfoTree
// object, which can then be iterated through, stored to disk, etc. }
procedure TJvDockInfoTree.CreateZoneAndAddInfoFromApp(Control: TControl);
var
I: TJvDockPosition; {was TAlign}
J: Integer;
TreeZone: TJvDockInfoZone;
DockBaseControl: TJvDockBaseControl;
TmpDockPanel: TJvDockPanel;
begin
TreeZone := TJvDockInfoZone(AddChildZone(CurrTreeZone, nil));
with TreeZone do
begin
ParentName := TJvDockInfoZone(CurrTreeZone).DockFormName;
SetDockInfoFromControlToNode(Control);
if Control is TJvDockPanel then
DockFormName := TJvDockInfoZone(CurrTreeZone).DockFormName +
RsDockJvDockInfoSplitter + Control.Name
else
DockFormName := Control.Name;
FDataStream.Clear;
if Control is TJvDockTabHostForm then
TJvDockTabHostForm(Control).PageControl.SaveToStream(FDataStream)
else
if Control is TJvDockConjoinHostForm then
TJvDockConjoinHostForm(Control).Panel.DockManager.SaveToStream(FDataStream)
else
if Control is TJvDockPanel then
TJvDockPanel(Control).DockManager.SaveToStream(FDataStream);
DockClientData := JvDockStreamDataToString(FDataStream);
DockBaseControl := FindDockBaseControl(Control);
if DockBaseControl <> nil then
begin
SetDockInfoFromDockControlToNode(DockBaseControl);
if Control is TJvDockTabHostForm then
DockFormStyle := dsTab
else
if Control is TJvDockConjoinHostForm then
DockFormStyle := dsConjoin
else
DockFormStyle := dsNormal;
if DockBaseControl is TJvDockClient then
begin
if Control is TJvDockableForm then
with TJvDockableForm(Control).DockableControl do
for J := 0 to DockClientCount - 1 do
begin
CurrTreeZone := TreeZone;
CreateZoneAndAddInfoFromApp(DockClients[J]);
CurrTreeZone := TreeZone.GetParentZone;
end;
end
else
begin
// Changed to persist ALL DockPanels, not just Top,Left,Right,Bottom.
// This is a hardcoded assumption throughout the component that is
// proving hard to overcome.
for I := Low(TJvDockPosition) to High(TJvDockPosition) do // There are 5 TJvDockPositions now ! {NEW!}
begin
CurrTreeZone := TreeZone;
TmpDockPanel := TJvDockServer(DockBaseControl).DockPanel[I];
if Assigned(TmpDockPanel) then
begin
CreateZoneAndAddInfoFromApp(TmpDockPanel);
if TmpDockPanel is TJvDockVSNETPanel then // JvDockVSNetStyle specific:
CreateZoneAndAddInfoFromApp(TJvDockVSNETPanel(TmpDockPanel).VSChannel.VSPopupPanel);
end;
CurrTreeZone := TreeZone.GetParentZone;
end;
end;
end;
if Control is TJvDockPanel then
begin
DockFormStyle := dsDockPanel;
if Control is TJvDockVSPopupPanel then
with TJvDockVSPopupPanel(Control) do
for J := 0 to DockClientCount - 1 do
begin
CurrTreeZone := TreeZone;
CreateZoneAndAddInfoFromApp(TWinControl(DockClients[J]));
CurrTreeZone := TreeZone.GetParentZone;
end
else
with TJvDockPanel(Control) do
for J := 0 to DockClientCount - 1 do
begin
CurrTreeZone := TreeZone;
CreateZoneAndAddInfoFromApp(TWinControl(DockClients[J]));
CurrTreeZone := TreeZone.GetParentZone;
end;
end;
end;
end;
{$IFDEF USEJVCL}
procedure TJvDockInfoTree.CreateZoneAndAddInfoFromAppStorage;
var
FormList: TStringList;
CP, CP1: PChar;
S: string;
I: Integer;
OldPath: string;
procedure CreateZoneAndAddInfo(Index: Integer);
var
I: Integer;
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -