使用Delphi自带的TDockTabSet组件实现停靠功能(Jeremy North)
源地址:http://edn.embarcadero.com/article/33446
摘要: Instructions on how to use the TDockTabSet component to make advanced docking user interfaces.
Introduction
?
This article discusses the use of the TDockTabSet component that was originally introduced in Delphi 2005.
Creating the applications main form
Create a File | New | VCL Forms Application - Delphi for Win32.
Setting up the main form
Locate the TDockTabSet component??on the additional page in the Tool Palette and drop it onto the main form.
?
| Align | alLeft |
| DockSite | False |
| ShrinkToFit | True |
| Style | tsModernTabs |
| TabPosition | tpLeft |
| Width | 25 |
You may want to use the tsModernPopup tab style instead of the tsModernTabs. The download that accompanies this article does.
To keep the TDockTabSet component company on the form, drop the following components and modify the properties as indicated.
| Align | alTop |
| BevelKind | bkTile |
| BevelOuter | bvNone |
| Caption |
| Align | alLeft |
| BevelOuter | bvNone |
| DockSite | True |
| Name | pDockLeft |
| Width | 0 |
| Align | alLeft |
| Align | Client |
?
?
If you name your components differently remember to reference the correct name when adding the source code.
The main form of your application should look something like the following.
The setting of the all important property
?
There is one more property that needs to be set on the TDockTabSet before we can continue. Set the?DestinationDockSite?to be the left aligned panel. Called pDockLeft in this article.
?
Now it's time to create another form for the application. There are now 3 ways this can be done in Delphi 2006 and I'll mention all of them:
Name the new form frmDock and save the unit as DockForm.
In the code editor for this unit do the following.
| 1 2 3 4 5 6 7 8 9 | procedure?TfrmDock.FormClose(Sender: TObject; var?Action: TCloseAction); begin ??ManualFloat(Rect(0, 0, 0, 0)); ??Action := caFree; end; |
4.Add an OnStartDock event that contains the following code
| 1 | procedure?TfrmDock.FormStartDock(Sender: TObject; var?DragObject:<br data-filtered="filtered"><br data-filtered="filtered">? TDragDockObject);<br data-filtered="filtered"><br data-filtered="filtered">begin<br data-filtered="filtered"><br data-filtered="filtered">? DragObject := TDragDockObjectEx.Create(Self);<br data-filtered="filtered"><br data-filtered="filtered">? DragObject.Brush.Color := clAqua; // this will display a red outline<br data-filtered="filtered"><br data-filtered="filtered">end;<br data-filtered="filtered"> |
| 1 | class?function?TfrmDock.CreateDockForm(const?aColor: TColor): TCustomForm;<br data-filtered="filtered"><br data-filtered="filtered">begin<br data-filtered="filtered"><br data-filtered="filtered">? result := TfrmDock.Create(Application);<br data-filtered="filtered"><br data-filtered="filtered">? result.Color := aColor;<br data-filtered="filtered"><br data-filtered="filtered">? result.Caption := ColorToString(aColor);<br data-filtered="filtered"><br data-filtered="filtered">? result.Show;<br data-filtered="filtered"><br data-filtered="filtered">end;<br data-filtered="filtered"> |
6.Finally modify the following properties on the dock form
| BorderStyle | bsSizeToolWin |
| DragKind | dkDock |
| DragMode | dmAutomatic |
You may also want to modify the size of the frmDock form to not be so wide.
That is the form that will be docked completed. Time to write some in the main unit!
?
Switch to the MainForm unit now and make the following changes.
1.Invoke the Use Unit dialog (Alt+F11 or File | Use Unit) and select the DockForm unit.
2.Create an OnClick event for the TButton with the following code
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 | procedure?TfrmMain.Button1Click(Sender: TObject); var ??i: Integer; begin ??// close all previously dockable forms before recreating ??for?i := 0?to?Screen.FormCount - 1?do ????if?Screen.Forms[i] is?TfrmDock then ??????Screen.Forms[i].Close; ??// dock to the component called pDockLeft ??TfrmDock.CreateDockForm(clBlue).ManualDock(pDockLeft); ??// dock to the top on the pDockLeft panel ??TfrmDock.CreateDockForm(clGreen).ManualDock(pDockLeft, nil, alTop); ??// dock to the right on the pDockLeft panel ??TfrmDock.CreateDockForm(clRed).ManualDock(pDockLeft, nil, alRight); ??// dock directly to the DockTabSet ??TfrmDock.CreateDockForm(clWhite).ManualDock(DockTabSet1); end; |
The remaining code is required to get the docking behavior to play nice amongst each other.
?
1.Create an OnDockDrop event for the pDockLeft panel and add the following code
This OnDockDrop event makes sure that the width of the pDockLeft panel is sufficient for when the frmDock form is dropped on it. If the panel previously had a width of 0 (which means you can't see it) then set the width to 150. This value is hardcoded for the demo but I could have easily read the current width of the control being dropped and using that value. You can get the width of the control being dropped from the TDragDockObject passed in as the?Source?parameter. You would use the following to get the width of the control that is being dropped:
?
| 1 | pDockLeft.Width := Source.Control.Width; |
The OnDockDrop code also makes sure that the splitter is visible when a form is being docked and that it is in the correct position.
?
| 1 2 3 4 5 6 7 | procedure?TfrmMain.pDockLeftDockDrop(Sender: TObject; Source: TDragDockObject; X, Y: Integer); begin ??if?pDockLeft.Width = 0?then ????pDockLeft.Width := 150; ??Splitter1.Visible := True; ??Splitter1.Left := pDockLeft.Width; end; |
2.Create an OnUndock event for the pDockLeft panel and add the following code
If the form being undocked is the last one on it then we need to set the Panels width back to 0 and hide the Splitter.
?
| 1 2 3 4 5 6 7 8 | procedure?TfrmMain.pDockLeftUnDock(Sender: TObject; Client: TControl; NewTarget: TWinControl; var?Allow: Boolean); begin ??if?pDockLeft.DockClientCount = 1?then ??begin ????pDockLeft.Width := 0; ????Splitter1.Visible := False; ??end; end; |
The DockOver event is the event responsible for drawing the forms outline at the dock site.
| 1 2 3 4 5 6 7 8 9 10 11 12 13 | procedure?TfrmMain.pDockLeftDockOver(Sender: TObject; Source: TDragDockObject; X, Y: Integer; State: TDragState; ??var?Accept: Boolean); var ??lRect: TRect; begin ??Accept := Source.Control is?TfrmDock; ??if?Accept then ??begin ????lRect.TopLeft := pDockLeft.ClientToScreen(Point(0, 0)); ????lRect.BottomRight := pDockLeft.ClientToScreen(Point(150, pDockLeft.Height)); ????Source.DockRect := lRect; ??end; end; |
If a control is being docked to the DockTabSet component we want to make sure it is visible.
?
| 1 | procedure?TfrmMain.DockTabSet1DockDrop(Sender: TObject; Source: TDragDockObject; X, Y: Integer);<br data-filtered="filtered">begin<br data-filtered="filtered">? DockTabSet1.Visible := True;<br data-filtered="filtered">end;<br data-filtered="filtered"> |
The DockTabSet1 component should only be visible if there is a control docked to it.
| 1 2 3 4 | procedure?TfrmMain.DockTabSet1TabRemoved(Sender: TObject); begin ??DockTabSet1.Visible := DockTabSet1.Tabs.Count > 0; end; |
More action shots
Application appearance on startup
Tab selected. Note you can click on the pin button to pin the green form to the docksite or click on the cross to close the form. To not do anything, click in the memo to change the focus away from the green form to hide it.
Dragging an undocked form, note the red drag outline
Hide image
All windows unpinned
What about VCL.NET support?
?
Unfortunately there is a bug in the VCL.NET implementation of the TDockTabSet component. It is currently logged in QualityCentral and hopefully it will be addressed in the next major Delphi release.
?
Report No: 24640 ( RAID: 238759 ) Status: Open
TDockTabSet control doesn't work correctly in VCL.NET
http://qc.borland.com/wc/qcmain.aspx?d=24640
?
The bug cannot be fixed in an update as it requires an interface change to the TDockTabSet class. If you wish to fix the bug yourself you can use the following workaround:
?
Move the?DoAddDockClient?method from?strict private?to?protected?and?override?it.
previously...
TDockTabSet = class(TTabSet)strict private
procedure DoAddDockClient(Client: TControl; const ARect: TRect);
should become... TDockTabSet = class(TTabSet)
protected
procedure DoAddDockClient(Client: TControl; const ARect: TRect); override;
You also have to take the necessary steps to ensure that the modified unit is compiled into your assembly as well.
?
An alternate approach might be:
begin
RegisterComponents('Fixed DockTabSet', [TFixedDockTabSet]);
end;
Build the new package Install the package using the Installed .NET Components menu item from the components menu, making sure to add it to the VCL.NET components tab. If all worked well when you create a new VCL.NET Application you will get a?Fixed DockTabSet?palette page that has the new TDockTabSet control on it.
?
Closing comments
With the help of the TDockTabSet component you can now create more advanced user interfaces with dockable windows which is sure to annoy most users. With a lot more hair pulling and sleepless nights, you can use the code in this article as a basis to create your our docking framework. I've done this as part for my QualityCentral windows client I created.
I encourage you to download and try it out from my website?www.jed-software.com.
?
Hide image
Hide image
While doing so, don't forget to add any issues or enhancements to QualityCentral.
Source Download
?
You can download the source to this article from Code Central:
??Download Source Code
本文轉自 不得閑 博客園博客,原文鏈接:http://www.cnblogs.com/DxSoft/archive/2010/12/15/1906408.html?? ,如需轉載請自行聯系原作者
總結
以上是生活随笔為你收集整理的使用Delphi自带的TDockTabSet组件实现停靠功能(Jeremy North)的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 配置Exchange 2010 服务器(
- 下一篇: 《Zabbix-ICMP ping监控添