一、概述
MDI窗口包含一個框架窗口和若干子窗口。
實際上,框架窗口本身是一個普通主窗口,不過它的客戶去被一個特殊窗口覆蓋。
這個特殊窗口是系統預定義的“窗口類”,類名稱為:"MDICLIENT"。它負責各個MDI子窗口的管理。
二、窗口建立
1.注冊一個MDI框架窗口類,提供MDI框架窗口消息處理函數
MDI框架窗口消息處理函數中,將未處理消息交由DefFrameProc處理
//MDI框架窗口消息處理函數
2.注冊多個MDI子窗口類、對應提供各MDI子窗口的消息處理函數
LRESULT CALLBACK MDIFrameWndProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
{
//...
//其他消息交給由系統提供的缺省框架處理函數DefFrameProc
//其中,第二個參數是客戶區窗口句柄
return ::DefFrameProc (hwnd,hwndClient, message, wParam, lParam) ;
}
子窗口消息處理函數中,將未處理消息交由MDIDefMDIChildProc處理 //MDI子窗口消息處理函數
3.在框架窗口的客戶區建立MDI管理子窗口
LRESULT CALLBACK MDIChildWndProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
{
//...
//...
//其他消息交給由系統提供的缺省MDI子窗口處理函數
return ::DefMDIChildProc (hwnd, message, wParam, lParam) ;
}
MDI子窗口的管理實際上是由框架窗口客戶區的"MDILIENT"窗口完成的。
這是一個系統預定義的窗口。
在主窗口收到WM_CREATE消息後:
case WM_CREATE:
窗口的大小沒有關系,缺省的框架窗口消息處理函數為讓它覆蓋整個客戶區。
{
hinst=((LPCREATESTRUCT) lParam)->hInstance;
//填充CLIENTCREATESTRUCT結構
CLIENTCREATESTRUCT clientcreate ;
clientcreate.hWindowMenu = hMenuInitWindow ; //用於添加窗口列表的菜單句柄
clientcreate.idFirstChild = 50000 ; //起始ID
hwndClient =CreateWindowEx(0,
"MDICLIENT", //類名稱為"MDICLIENT"
NULL,
WS_CHILD |WS_CLIPCHILDREN| WS_VISIBLE,
0,
0,
0,
0,
hwnd,
(HMENU)1,//ID
hinst, //實例句柄
&clientcreate); //參數
}
return 0;
MDI客戶區窗口建立後,通過向它發送消息管理子窗口的建立、銷毀、排列等等。
4.MDI子窗口的建立
可以在菜單中添加命令項,以建立子窗口。
框架窗口的消息處理函數收到命令後,向MDI客戶區窗口發建立命令。
case ID_NEW:
三、消息循環中處理針對MDI的熱鍵
{
MDICREATESTRUCT mdicreate;
mdicreate.szClass = szMDIChildName ; //MDI子窗口的類名稱
mdicreate.szTitle = TEXT ("Hello") ;
mdicreate.hOwner = hinst ;
mdicreate.x = CW_USEDEFAULT ;
mdicreate.y = CW_USEDEFAULT ;
mdicreate.cx = CW_USEDEFAULT ;
mdicreate.cy = CW_USEDEFAULT ;
mdicreate.style = 0 ;
mdicreate.lParam = 0 ;
SendMessage (
hwndClient, //MDI客戶區窗口句柄
WM_MDICREATE, //創建MDI子窗口
0,
(LPARAM) (LPMDICREATESTRUCT) &mdicreate //創建參數
) ;
}
break;
在消息循環中,用TranslateMDISysAccel處理針對MDI的熱鍵。while (GetMessage (&msg, NULL, 0, 0))
四、命令的流向
{
if (!TranslateMDISysAccel (hwndClient, &msg) &&
!TranslateAccelerator (hwndFrame, hAccel, &msg))
{
TranslateMessage (&msg) ;
DispatchMessage (&msg) ;
}
}
框架窗口在收到WM_COMMAND等通知消息後,應該給當前激活的MDI窗口提供處理機會。case WM_COMMAND:
switch (LOWORD (wParam))
{
//針對框架的命令
case ID_ONE:
//...
return 0;
//針對MDI子窗口管理的命令
case IDM_WINDOW_TILE:
SendMessage (hwndClient, WM_MDITILE, 0, 0) ;
return 0 ;
//針對子窗口的命令又子窗口去處理
default:
hwndChild = (HWND) SendMessage (hwndClient,
WM_MDIGETACTIVE, 0, 0) ;
if (IsWindow (hwndChild))
SendMessage (hwndChild, WM_COMMAND, wParam, lParam) ;
break ; //..and then to DefFrameProc
}
break ; //跳出針對WM_COMMAND的case分支,又DefFrameProc處理剩下的命令
五、子窗口的管理
1.概述
給MDI客戶區窗口發控制消息即可
如:case WM_COMMAND:
2.當前子窗口的關閉
switch (LOWORD (wParam))
{
case IDM_WINDOW_TILE:
SendMessage (hwndClient, WM_MDITILE, 0, 0) ;
return 0 ;
case IDM_WINDOW_CASCADE:
SendMessage (hwndClient, WM_MDICASCADE, 0, 0) ;
return 0 ;
case IDM_WINDOW_ARRANGE:
SendMessage (hwndClient, WM_MDIICONARRANGE, 0, 0) ;
return 0;
//...
//...
}
break;
關閉當前激活窗口時,先向該窗口發送查詢消息:WM_QUERYENDSESSION。
子窗口的消息處理循環中響應此消息,作關閉前的一些處理,若能關閉,返回真
否則返回假。
框架窗口中根據此返回值決定是否關閉窗口。
如果用戶直接按下子窗口的關閉按鈕,則WM_CLOSE消息直接發送到了子窗口消息處理函數。
例如:
框架窗口命令處理中:
case IDM_FILE_CLOSE:
子窗口的消息處理函數中:
//獲得當前激活窗口
hwndChild = (HWND) SendMessage (hwndClient, WM_MDIGETACTIVE, 0, 0);
//詢問通過後,銷毀窗口
if (SendMessage (hwndChild, WM_QUERYENDSESSION, 0, 0))
SendMessage (hwndClient, WM_MDIDESTROY, (WPARAM) hwndChild, 0);
return 0;LRESULT CALLBACK HelloWndProc (HWND hwnd, UINT message,
3.關閉所有子窗口
WPARAM wParam, LPARAM lParam)
{
switch (message)
{
//...
//...
case WM_QUERYENDSESSION:
case WM_CLOSE:
if (IDOK != MessageBox (hwnd, TEXT ("OK to close window?"),
TEXT ("Hello"),
MB_ICONQUESTION | MB_OKCANCEL))
return 0 ;
break ; // i.e., call DefMDIChildProc
}
return DefMDIChildProc (hwnd, message, wParam, lParam) ;
}
當使用命令方式關閉所有子窗口時,需要枚舉所有子窗口進行關閉。
例:
框架窗口響應命令:
case IDM_WINDOW_CLOSEALL:
枚舉函數:
//針對所有子窗口執行CloseEnumProc
EnumChildWindows (hwndClient, CloseEnumProc, 0) ;
return 0 ;BOOL CALLBACK CloseEnumProc (HWND hwnd, LPARAM lParam)
六、菜單控制
{
if (GetWindow (hwnd, GW_OWNER)) // Check for icon title
return TRUE ;
SendMessage (GetParent (hwnd), WM_MDIRESTORE, (WPARAM) hwnd, 0) ;
if (!SendMessage (hwnd, WM_QUERYENDSESSION, 0, 0))
return TRUE ;
SendMessage (GetParent (hwnd), WM_MDIDESTROY, (WPARAM) hwnd, 0) ;
return TRUE ;
}
在MDI程序中,可以根據激活的子窗口而切換框架窗口的菜單。
並且,可以將窗口列表添加到菜單中去。所添加的菜單項命令是又框架對應的缺省消息處理函數完成的。
1.為每種窗口類准備一套菜單資源
2.裝載菜單,得到菜單句柄
3.框架在建立時,使用框架菜單的句柄作為參數。
4.子窗口在激活時,加載自己菜單到框架窗口
失去焦點時,還原框架菜單。
使用向MDI客戶區窗口發送WM_MDISETMENU或WM_MDISETMENU消息。
wParam為菜單句柄,lParam為欲添加窗口列表的子菜單句柄 case WM_MDIACTIVATE:
//激活時,設置框架菜單
if (lParam == (LPARAM) hwnd)
SendMessage (hwndClient, WM_MDISETMENU,
(WPARAM) hMenuHello, (LPARAM) hMenuHelloWindow) ;
//失去焦點時,將框架菜單還原
if (lParam != (LPARAM) hwnd)
SendMessage (hwndClient, WM_MDISETMENU, (WPARAM) hMenuInit,
(LPARAM) hMenuInitWindow) ;
DrawMenuBar (hwndFrame) ;
//注: hwndFrame的得到方法:
//hwndClient = GetParent (hwnd) ;
//hwndFrame = GetParent (hwndClient) ;
return 0 ;