【win32】标记菜单与对话框背景色

主要是想如下的一个程序,设定了一个标记菜单,在选定不同菜单的时候,客户区的颜色会改变,以此说明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】鼠标响应事件》(点击打开链接)已经提到过不在赘述。

这个程序同时需要设置一个全局变量,记录用户当前选择的是哪个标识菜单,用于发生窗口重绘的时候,给客户区的矩形重新上色。

  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值