在开发软件时,总希望软件界面越漂亮越好,我们在C++ Builder的SDI程序中比较容易实现漂亮的软件界面,但在MDI程序中,由于Windows的MDI软件的开发隐藏了许多技术细节,用SDI程序的方法,就不能实现了。下面笔者将详细讲述如何实现MDI程序背景图。
在MDI程序中是由两个窗口构成的,一个MDI主窗口和一个客户窗口。客户窗口覆盖了主窗口的客户区,并提供大量的MDI支持。在C++ Builder的MDI 的主窗口中提供了一个ClientHandle的客户窗口句柄,我们就是利用这个句柄来实现位图的显示。我们知道,在Windows的窗口中绘置位图,为使位图能够不断地刷新,必须响应WM_PAINT和WM_ERASEBKGND消息。我们可以利用钩子函数(钩子是Windows系统中一种特殊的消息处理机制,可以监视系统或进程中的各种事件消息,截获发往目标窗口的消息并进行处理)。截住Windows系统发送给MDI客户窗口的WM_PAINT和WM_ERASEBKGND消息,从而实现客户窗口的刷新和重绘。我们可以在系统中安装自定义的消息钩子,对发往客户窗口的消息进行过滤,只处理WM_PAINT和WM_ERASEBKGND,以实现我们的目标。
编写钩子函数
编写Windows钩子函数分为三步:定义钩子、安装钩子和卸载钩子。
定义钩子函数
钩子函数是一种特殊的回调函数,不同事件的钩子其函数头是不一样,本次用到的钩子函数如下所示:
LRESULT CALLBACK GetMsgProc(int nCode, WPARAM wParam, LPARAM lParam)
参数nCode的值表示本钩子函数是否必须处理该消息, wParam表明这次传递的消息是否已从Windows消息队列中删除, lParam参数用来传送消息。
在钩子函数中,必须将系统发送的消息继续回送给系统以使其它程序可以继续使用该消息,该函数为:
LRESULT CallNextHookEx(HHOOK hhk, int nCode, WPARAM wParam, LPARAM lParam )
参数hhk是安装钩子函数时安装函数返回的句柄,nCode、wParm和lParm参数是系统传给钩子函数的值。
安装钩子函数
定义完钩子函数后,必须将该钩子安装到Windows系统中才能生效,安装钩子的函数为:
HHOOK SetWindowsHookEx(int idHook, HOOKPROC lpfn, HINSTANCE hMod, DWORD dwThreadId)
参数idHook表示待安装的钩子函数类型,可以是键盘、鼠标或外壳等钩子,lpfn表示钩子函数的地址, hMod表示是全局钩子还是局部钩子,如果是全局钩子则钩子函数必须在DLL文件中,dwThreadId表示钩子将要起作用的程序ID。
卸载钩子函数
钩子函数使用完后必须卸载,这是一个良好
程序员必备的优良品质。卸载钩子的函数为:
BOOL UnhookWindowsHookEx(HHOOK hhk)
参数hhk表示待卸载的钩子句柄。
详细代码
根据上面所述,下面介绍详细代码:
//定义全局变量
HHOOK hMsgHook;
//钩子句柄
int iClientHeight, iClientWidth;
//待画的客户区高和宽
Graphics::TBitmap Face;
// 从文件调用位图的控件
HBITMAP hFaceBitmap;
//位图句柄
HWND hClientHandle, hMdiHandle;
//MDI主窗口和MDI客户窗口句柄
LRESULT CALLBACK GetMsgProc(int nCode, WPARAM wParam, LPARAM lParam ) ;
void __fastcall TMainForm::FormPaint(TObject Sender)
{ iClientHeight = ClientHeight;
iClientWidth = ClientWidth;
} //设置系统时, 在状态条上显示
void __fastcall TMainForm::FormShow(TObject Sender)
{ //从文件中调入位图
Face = new Graphics::TBitmap();
Face->LoadFromFile(“d:\\\\temp\\\\face.bmp”);
hFaceBitmap = Face->Handle;
//保存位图句柄
hClientHandle = ClientHandle;
//保存窗口句柄
hMdiHandle = Handle;
//保存MDI主窗口句柄
//安装截取程序消息的钩子函数
hMsgHook = SetWindowsHookEx(WH_GETMESSAGE, (HOOKPROC)GetMsgProc, NULL, GetCurrentThreadId() );
} //钩子函数,处理系统WM_PAINT和WM_ERASEBKGND消息
LRESULT CALLBACK GetMsgProc(int nCode, WPARAM wParam, LPARAM lParam )
{ LRESULT lReturn=0;
MSG cwMessage;
cwMessage = (MSG)lParam;
if ( cwMessage->hwnd == hClientHandle || cwMessage->hwnd == hMdiHandle)
//是发送给子窗口的消息则处理
{if ( cwMessage->message == WM_PAINT || cwMessage->message == WM_ERASEBKGND )
{ //重画用户窗口
DrawBitmap(hClientHandle, hFaceBitmap, iClientHeight, iClientWidth);
}
}
if ( hMsgHook != NULL) //将消息继续下传
lReturn = CallNextHookEx(hMsgHook, nCode, wParam, lParam );
return lReturn;
} //卸载钩子函数
void __fastcall TMainForm::FormClose(TObject Sender, TCloseAction &&Action)
{ if ( hMsgHook != NULL)
UnhookWindowsHookEx( hMsgHook );
if ( Face != NULL )
delete Face;
}
//在指定的窗口中,画位图,填充整个用户窗口
//Ture为绘制成功,false为绘制失败
BOOL DrawBitmap(HWND Handle, HBITMAP hBitmap, int iClientHeight, int iClientWidth)
{ if ( hBitmap == NULL )
return false;
BITMAP b;
int iBitmapH, iBitmapW;
GetObject( hBitmap, sizeof( BITMAP), &&b);
iBitmapH = b.bmHeight;
iBitmapW = b.bmWidth;
int x, y;
HDC hClientDC, hMemDC;
hClientDC = GetDC(Handle);
if ( hClientDC == NULL )
return false;
hMemDC = CreateCompatibleDC( hClientDC );
if ( hMemDC == NULL )
{ DeleteDC( hClientDC );
return false;
}
SelectObject( hMemDC, hBitmap );
x = 0;
while ( x < iClientWidth )
{ y = 0;
while ( y < iClientHeight )
{ClientCanvas->Draw(x, y, Face);
BitBlt( hClientDC, x, y,iBitmapW, iBitmapH, hMemDC, 0, 0,SRCCOPY );
y = y + iBitmapH; }
x = x + iBitmapW; }
DeleteDC( hMemDC );
DeleteDC( hClientDC );
return true; }
将上述C++ Builder代码片段加入用户的MDI软件中即可实现任意的MDI程序背景图