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去掉,结果象这样:窗口可以修改大小,但左上角有图标,:

Delphi中实现可以更改大小的对话框

图 1 带图标的对话框

注意左上角有图标。

而我们的目标则是下面的两种效果,左上角都没有图标,但窗口都可以修改大小。

Delphi中实现可以更改大小的对话框

图 2 打开文件对话框

Delphi中实现可以更改大小的对话框

图 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消息处理过程没有触发)。问题出在哪里呢?

Delphi中实现可以更改大小的对话框

图 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属性,具体情况可以看代码。

Delphi中实现可以更改大小的对话框

图 5 没有图标、可以修改大小、带有SizeGrip的对话框

下载(exe和源代码)

http://www.zocsoft.com/temp/Resizable_Dialog.rar

5、参考资料:

MSDN: How To Design a Resizable MFC Property Sheet

引用地址:《Delphi中实现可以更改大小的对话框》

· 震惊!原来酒驾处罚这么严重

酒驾处罚标准 不要有任何一丝侥幸心理! http://image....

· 把年龄相仿的狮虎熊放一起,谁更厉害?结果出人意料

很多人都想知道狮子、老虎和熊打起来谁最厉害,于是便有好事之人把这三种动物关在一起...

· 湖北宜昌三峡坝区水面惊现神秘动物

近日,湖北宜昌,一段视频在当地热传:有网友在三峡坝区拍到神秘动物,体型硕大数米长...

 
Delphi中实现可以更改大小的对话框
Delphi中实现可以更改大小的对话框
关键字:Dialog、对话框、resizable1、问题的提出问题来自Stanley_Xu,希望得到只有关闭按钮(还可以有帮助),左上也没有程序的图标并且能够更改窗口大小的对话框。VCL中为TForm设置了BorderStyle和BorderIcons属性...查看完整版>>Delphi中实现可以更改大小的对话框
 
实现控件的移动、改变大小(DELPHI实现)
摘要:实现控件的移动、改变大小(DELPHI实现)实现控件的移动,改变大小(DELPHI实现)主要使用Perform方法 function Perform(Msg: Cardinal; WParam, LParam: Longint): Longint; 只要能够使用类似于win32API的函...查看完整版>>实现控件的移动、改变大小(DELPHI实现)
 
用Delphi实现控制IE窗体的大小 ——IESizer
  做过网页的朋友都知道,在设计网页的时候有一个很重要的问题,就是要使网页在不同的分辨率(体现在可视屏幕大小)下都能有良好的显示效果;另外有些时候可能要做一些特殊用途的网页(例如广告或通告等页面),当...查看完整版>>用Delphi实现控制IE窗体的大小 ——IESizer
 
实现控件的移动、改变大小(DELPHI实现)
实现控件的移动,改变大小(DELPHI实现)主要使用Perform方法 function Perform(Msg: Cardinal; WParam, LParam: Longint): Longint; 只要能够使用类似于win32API的函数SendMessage(),本方法同样可在其他环境里应...查看完整版>>实现控件的移动、改变大小(DELPHI实现)
 
DELPHI也可以实现控件数组,用定义数组变量实现控件数组
Delphi也可以实现控件数组,用定义数组变量实现控件数组  小弟表达能力有限,此文章又是本人第一次发表文档,所以不周到之处请各位仁兄多多包涵。  我们在使用Delphi时有时会发现一个问题,就是Delphi没有像VB或...查看完整版>>DELPHI也可以实现控件数组,用定义数组变量实现控件数组