主要是想如下的一个程序,设定了一个标记菜单,在选定不同菜单的时候,客户区的颜色会改变,以此说明win32对话框客户区的背景颜色设置与标记菜单使用。
比起MFC里面的背景颜色改变,WIN32的要复杂得多,而菜单的修改更是需要废一番功夫,但是通过这个程序揭示了窗体程序的本质。
一、菜单设置与修改
1、如下图,打开Resource.h,在#ifndef之前,已有的菜单项之后,为自己新设置的菜单选项,设置编号。
这里,501-503分别对应红色、绿色、蓝色三个菜单:
//自己新建的菜单项
#define IDM_RED 501
#define IDM_GREEN 502
#define IDM_BLUE 503
2、之后,如下图,对资源文件XX.rc查看代码。删去用快捷键、对话框的代码。因为VS2010自带的对话框,按ALT+/ ?会自动弹出对话框的。
对原有关于文件、帮助的菜单修改如下:
IDC_MENU_NOTE MENU
BEGIN
POPUP "背景颜色改变(&C)"
BEGIN
MENUITEM "红色(&R)", IDM_RED
MENUITEM "绿色(&G)", IDM_GREEN
MENUITEM "蓝色(&B)", IDM_BLUE
END
END
3、接下来,就真的可以回到Menu_Note.cpp的消息回调函数LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)中编程了。修改如下:
int menu_id=0;
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
int wmId, wmEvent;
PAINTSTRUCT ps;
HDC hdc;
// 获取窗口上的整个菜单栏的句柄
auto hmm = GetMenu(hWnd);
// 获取第一个弹出菜单,即[背景颜色改变]菜单的句柄
auto hfmn = GetSubMenu(hmm, 0);
HBRUSH brush;//画刷,用于改变客户区的背景色
RECT rect;//矩形,用于捕捉客户区的大小
switch (message)
{
case WM_COMMAND:
wmId = LOWORD(wParam);
wmEvent = HIWORD(wParam);
switch (wmId)//分析窗口的选择
{
case IDM_RED:
menu_id=IDM_RED;
CheckMenuRadioItem(hfmn, IDM_RED, IDM_BLUE, IDM_RED, MF_BYCOMMAND);//令此菜单单选
//第一个参数是菜单句柄、第二、三个参数是指明这个菜单从哪个编号到哪个编号是单选的,第四个参数是选择谁
hdc=GetDC(hWnd);
brush=CreateSolidBrush(RGB(255,0,0));//创建一个红色画刷
GetClientRect(hWnd,&rect);//获取客户区这个大矩形到rect中
FillRect(hdc,&rect,brush);//填充这个矩形
break;
case IDM_GREEN:
menu_id=IDM_GREEN;
CheckMenuRadioItem(hfmn, IDM_RED, IDM_BLUE, IDM_GREEN, MF_BYCOMMAND);
hdc=GetDC(hWnd);
brush=CreateSolidBrush(RGB(0,255,0));
GetClientRect(hWnd, &rect);
FillRect(hdc,&rect, brush);
break;
case IDM_BLUE:
menu_id=IDM_BLUE;
CheckMenuRadioItem(hfmn, IDM_RED, IDM_BLUE, IDM_BLUE, MF_BYCOMMAND);
hdc=GetDC(hWnd);
brush=CreateSolidBrush(RGB(0,0,255));
GetClientRect(hWnd, &rect);
FillRect(hdc,&rect, brush);
break;
}
break;
case WM_PAINT://如果用户移动窗口、调整窗口的大小会发生重绘,那又要重新上色
hdc = BeginPaint(hWnd, &ps);
switch(menu_id){
case 0:
brush=CreateSolidBrush(RGB(255,255,255));
GetClientRect(hWnd, &rect);
FillRect(hdc,&rect, brush);
break;
case IDM_RED:
brush=CreateSolidBrush(RGB(255,0,0));
GetClientRect(hWnd, &rect);
FillRect(hdc,&rect, brush);
break;
case IDM_GREEN:
brush=CreateSolidBrush(RGB(0,255,0));
GetClientRect(hWnd, &rect);
FillRect(hdc,&rect, brush);
break;
case IDM_BLUE:
brush=CreateSolidBrush(RGB(0,0,255));
GetClientRect(hWnd, &rect);
FillRect(hdc,&rect, brush);
break;
}
EndPaint(hWnd, &ps);
break;
case WM_DESTROY:
PostQuitMessage(0);
break;
default:
return DefWindowProc(hWnd, message, wParam, lParam);
}
return 0;
}
其实,对于菜单的核心在于调用CheckMenuRadioItem函数,第一个参数是要在其子项中设置的单选的菜单的句柄,第二个参数和第三个参数指定合并为一个组的ID范围,在这个范围内的菜单项被看人为同一组,这一组中,每一次只能有一项被checked,第四个参数就指定在这组项中哪一个被选中,最后一个参数决定是用ID来标识还用从0开始的索引。
至于设置对话框客户区的背景色,这里需要先设置画刷的颜色,再获取客户区的矩形,再进行填充,填充的时候,注意获取并初始化上下文HDC,HDC与HWND这个概念在《【win32】鼠标响应事件》(点击打开链接)已经提到过不在赘述。
这个程序同时需要设置一个全局变量,记录用户当前选择的是哪个标识菜单,用于发生窗口重绘的时候,给客户区的矩形重新上色。