1. 程式人生 > >Delphi中實現可以更改大小的對話方塊

Delphi中實現可以更改大小的對話方塊

關鍵字:Dialog、對話方塊、resizable

1、問題的提出
問題來自Stanley_Xu,希望得到只有關閉按鈕(還可以有幫助),左上也沒有程式的圖示並且能夠更改視窗大小的對話方塊。
VCL中為TForm設定了BorderStyle和BorderIcons屬性,用以簡化視窗樣式的設定(否則就要呼叫SetWindowLong和GetWindowLong等API函式)。TFormBorderStyle和TBorderIcon的定義和說明如下:

Value          Meaning
bsDialog       Not resizable; standard dialog box border//不能改大小
bsSingle       Not resizable; single-line border
bsNone Not     resizable; no visible border line
bsSizeable     Standard resizable border
bsToolWindow   like bsSingle but with a smaller caption
bsSizeToolWin  like bsSizeable with a smaller caption

type TBorderIcon = (biSystemMenu, biMinimize, biMaximize, biHelp);
TBorderIcons = set of TBorderIcon;

Value          Meaning
biSystemMenu   The form has a Control menu (also known as a System menu).
biMinimize     The form has a Minimize button
biMaximize     The form has a Maximize button
biHelp         If BorderStyle is bsDialog or biMinimize and biMaximize are excluded, a question mark appears in the form's title bar and when clicked, the cursor changes to crHelp; otherwise,no question mark appears.

顯然,通過BorderStyle和BorderIcons只能夠滿足一般的需要,要實現能夠修改大小的對話方塊就有所力不能及了。
一般情況下,我要得到不能最大最小化但又可以更改大小的視窗,就把BorderStyle設定為bsSizeable,把BorderIcons的biMinimize和biMaximize去掉,結果象這樣:視窗可以修改大小,但左上角有圖示,:

VcLSizableDlg
圖 1 帶圖示的對話方塊


注意左上角有圖示。

而我們的目標則是下面的兩種效果,左上角都沒有圖示,但視窗都可以修改大小。

OpenSaveDlg
圖 2 開啟檔案對話方塊

Browse4Folder

圖 3瀏覽資料夾對話方塊


2、問題解決一半
搜尋了一下MSDN,找到一篇教你如何設計可以可更改大小的屬性頁的文章(在MFC中CPropertySheet是作為CPropertyPage子頁出現的,後者從CDialog繼承而來,通常不能修改大小)《

How To Design a Resizable MFC Property Sheet》,文中介紹的方法是在屬性頁建立之前修改視窗樣式,然後手動處理WM_SIZE訊息。

int CALLBACK CMyPropertySheet::XmnPropSheetCallback(HWND hWnd, UINT message, LPARAM lParam)
{
    extern int CALLBACK AfxPropSheetCallback(HWND, UINT message, LPARAM lParam);
    // XMN: Call MFC's callback
    int nRes = AfxPropSheetCallback(hWnd, message, lParam);

    switch (message)
    {
    case PSCB_PRECREATE:
       
// Set our own window styles
        ((LPDLGTEMPLATE)lParam)->style |= (DS_3DLOOK | DS_SETFONT
  | WS_THICKFRAME | WS_SYSMENU | WS_POPUP | WS_VISIBLE | WS_CAPTION);
        break;
    }
    return nRes;
}

我試著將同樣的方法用到VCL的一個Form中。在設計時把BorderStyle設定為bsDialog,然後過載CreateParams方法。但結果是對話方塊確實變成了厚邊框(因為有WS_THICKFRAME樣式),滑鼠移動到各個邊框後能夠自動變化,左上角也沒有圖示,但視窗就是不能改變大小(新增的WM_SIZE訊息處理過程沒有觸發)。問題出在哪裡呢?

HalfZoCDlgResizable

圖 4 還不能完全令人滿意的對話方塊

3、問題的解決
查了一翻Forms.pas的原始碼,發現了問題所在。TCustomForm的WM_NCCREATE訊息處理過程中有一個ModifySystemMenu嵌入過程,用來修改Form的系統選單。注意下面紅色文字說的是“使系統選單看起來像對話方塊一樣”。接下來的幾句程式碼就把系統選單項刪得只剩下了“移動”和“關閉”。

procedure TCustomForm.WMNCCreate(var Message: TWMNCCreate);

procedure ModifySystemMenu;
var
    SysMenu: HMENU;
begin
    ……
   
{ Modify the system menu to look more like it's s'pose to }
    SysMenu := GetSystemMenu(Handle, False);
    if FBorderStyle = bsDialog then
    begin
        
{ Make the system menu look like a dialog which has only
        Move and Close }

        DeleteMenu(SysMenu, SC_TASKLIST, MF_BYCOMMAND);
        DeleteMenu(SysMenu, 7, MF_BYPOSITION);
        DeleteMenu(SysMenu, 5, MF_BYPOSITION);
        DeleteMenu(SysMenu, SC_MAXIMIZE, MF_BYCOMMAND);
        DeleteMenu(SysMenu, SC_MINIMIZE, MF_BYCOMMAND);
        DeleteMenu(SysMenu, SC_SIZE, MF_BYCOMMAND);
        DeleteMenu(SysMenu, SC_RESTORE, MF_BYCOMMAND);
    end else
    ……
end;

begin
    inherited;
    SetMenu(FMenu);
    if not (csDesigning in ComponentState) then ModifySystemMenu;
end;

所以,問題出在由於“SC_SIZE”被刪掉,視窗的樣式出現了畸形:有WS_THICKFRAME(可以修改視窗大小),但不響應WM_SIZE訊息(SC_SIZE被刪掉)。
解決的辦法很簡單:實現自己的WM_NCCREATE訊息處理過程,手動修改系統選單。


procedure TZoCDlgResizable.WMNCCreate(var Message: TWMNCCreate);

 //The following codes are copied from Form.pas line 4047, Delphi 7 sp1.
 procedure ModifySystemMenu;
 var
     SysMenu   : HMENU;
 begin
     SysMenu := GetSystemMenu(Handle, False);
     
{ Make the system menu look like a dialog which has only
  Move, Size and Close commands}
     DeleteMenu(SysMenu, SC_TASKLIST, MF_BYCOMMAND);
     DeleteMenu(SysMenu, 7, MF_BYPOSITION);
     
//Don't remove the separater before CLOSE command.
//   DeleteMenu(SysMenu, 5, MF_BYPOSITION);

     DeleteMenu(SysMenu, SC_MAXIMIZE, MF_BYCOMMAND);
     DeleteMenu(SysMenu, SC_MINIMIZE, MF_BYCOMMAND);
     
{ Don't remove the SIZE command, otherwise we'll lose the
  capability of resizing the Dialog. }
//   DeleteMenu(SysMenu, SC_SIZE, MF_BYCOMMAND);

     DeleteMenu(SysMenu, SC_RESTORE, MF_BYCOMMAND);
 end;

begin
   
{ Skip TCustomForm's WM_NCCREATE handler, which remove
    the SIZE command from the System Menu.}
    inherited DefaultHandler(Message);
   
//Dealing with the System Menu in our own way.
    ModifySystemMenu;
end;

4、TZoCDlgResizable類
最終的解決方案我封裝為一個繼承自TForm的類,效果如下,與圖1相同(如果想要圖2那樣的系統選單則把呼叫ModifySystemMenu的行刪掉),使用的時候從TZoCDlgResizable繼承一個即可。
BTW:我還順手給TZoCDlgResizable加了個SizeGrip屬性,具體情況可以看程式碼。


圖 5 沒有圖示、可以修改大小、帶有SizeGrip的對話方塊

5、參考資料: