(12):使用控件——单选按钮

最后更新于:2022-04-01 20:07:31

今天,咱们还是接着玩“控件斗地主”,这是我原创的超级游戏,有益身心健康,玩一朝,十年少。 哦,对,脑细胞极速运动了一下,想起了一个问题,这个破问题虽然网上有很多种解决方案,但是,并没有让所有人都解决问题。 不知道大家有没有调用过LoadIconMetric函数,这个函数在静态库Comctl32.lib中有定义(当然,动态库也有),不过,创建项目的时候,默认并没有引用它的,于是,大家知道,解决调用的方法就是在代码中加上: ~~~ #pragma comment(lib, "Comctl32.lib") ~~~ 我一般习惯这种方法,这样不必去修改项目属性。但是,很多朋友说过,在Win 7以后的系统,依然没有成功,我也尝到了调用失败的“甜头”,我一直在想,这是为什么呢? 于是,我又试了另一种方法,就是用LoadLibrary加载Comctl32.dll,然后通过函数指针去调用它: ~~~ typedef LRESULT (WINAPI * pLoadICMT)(.......); ~~~ 但结果还是没成功,GetProcAddress返回的地址为0,又一次尝到了失败带来的“刺激”感。 直到某一天,我在写某程序时,从上一文中大家都看到,那个按钮的视觉风格和Win9x/2000差不多,似乎没有XP那种充满美学水准的效果。其实,这是因为我们的程序没有启用视觉效果,默认情况下,使用版本5中的控件,而要有XP以上的风格,是在版本6的控件内库中才有。 当然方法可以很多人都知道,就是定义一个用于视觉效果的清单文件,本质是XML格式。不过我用的开发工具是VS 2005之后的版本,就不用弄个XML文件那么麻烦了,直接到MSDN上复制粘贴这段代码放到你的代码文件(.cpp)中,就是这个,直接抄过来就行了,适当的时候,要巧用MSDN上的资源。 ~~~ // 开启视觉效果 Copy from MSDN #pragma comment(linker,"\"/manifestdependency:type='win32' \ name='Microsoft.Windows.Common-Controls' version='6.0.0.0' \ processorArchitecture='*' publicKeyToken='6595b64144ccf1df' language='*'\"") ~~~ version是版本号,要有好看的效果,记得要6,不要写用低版本的。processorArchitecture是处理器平台,x86或amd64,用*号最好,通杀。 真是巧啊,原来这么一来,噗,LoadIconMetric也能调用了。总的来说,就是在代码文件加上以下内容: ~~~ #include //包含头文件 // 导入静态库 #pragma comment(lib, "Comctl32.lib") // 开启视觉效果 Copy from MSDN #pragma comment(linker,"\"/manifestdependency:type='win32' \ name='Microsoft.Windows.Common-Controls' version='6.0.0.0' \ processorArchitecture='*' publicKeyToken='6595b64144ccf1df' language='*'\"") ~~~ 这问题干掉了,开始今天的吹牛大行动。 单选按钮,是的,在WinForm里面你肯定知道,RadioButto,复选框就是CheckBox。不过那个时候.NET还没那么猛,那个时代,就是玩VB6,所以我知道VB里面,单选按钮叫Option吧。 然后找遍了Win32的控件库,怎么没见Radio和CheckBox,于是,我陷入了万分痛苦之中。不久后阅读MSDN文档,我就明白了,其实这两个玩意儿都是BUTTON类的,只是应用了不同的style罢了。 好的,咱们先来弄一个单选的吧。 ~~~ case WM_CREATE: { CreateWindow(L"Button",L"这玩意儿好", BS_RADIOBUTTON | WS_CHILD | WS_VISIBLE, 10,10,150,28, hwnd, (HMENU)IDC_RADBTN1, (HINSTANCE)GetWindowLongPtr(hwnd, GWLP_HINSTANCE), NULL); } return 0; ~~~ 应用BS_RADIOBUTTON样式,就可以使按钮变成单选按钮,不过,别忘了WS_CHILD | WS_VISIBLE。 运行一下。 ![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-06-14_575fd2d1cb270.PNG) 不过,当你点击它的时候,会发现它并不会选中,那是因为BS_RADIOBUTTON样式不会自动让它选上。我们换一种可以自动处理的样式,带AUTO打头的。 就是用 BS_AUTORADIOBUTTON | WS_CHILD | WS_VISIBLE,如下图的预览结果。 ![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-06-14_575fd2d1df489.PNG) 接下来,我们多创建几个吧。 ~~~ /*-----------------------------------------------------------------*/ //控件ID #define IDC_RADBTN1 50001 #define IDC_RADBTN2 50002 #define IDC_RADBTN3 50003 #define IDC_RADBTNBLUE 51001 #define IDC_RADBTNRED 51002 #define IDC_RADBTNGREEN 51003 /*-----------------------------------------------------------------*/ // 消息处理函数 LRESULT CALLBACK WindowProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam) { switch(msg) { case WM_CREATE: { // 获取当前实例句柄 HINSTANCE hthisapp = (HINSTANCE)GetWindowLongPtr(hwnd, GWLP_HINSTANCE); // 纵坐标,控件将以此作为基准, // 排列时依次增加 int yLoc = 0; // 用来显示文本 yLoc += 10; CreateWindow(L"Static",L"请问你的性别是:", SS_SIMPLE | WS_CHILD | WS_VISIBLE, 10,yLoc,160,18, hwnd, NULL, hthisapp, NULL); // 第一组单选按钮 yLoc += 22; CreateWindow(L"Button",L"男", WS_CHILD | WS_VISIBLE | BS_AUTORADIOBUTTON | WS_GROUP, 12, yLoc, 60, 16, hwnd, (HMENU)IDC_RADBTN1, hthisapp,NULL); yLoc += 20; CreateWindow(L"Button",L"女", WS_CHILD | WS_VISIBLE | BS_AUTORADIOBUTTON, 12,yLoc, 60, 16, hwnd,(HMENU)IDC_RADBTN2,hthisapp,NULL); yLoc += 20; CreateWindow(L"Button",L"人妖",WS_CHILD | WS_VISIBLE | BS_AUTORADIOBUTTON, 12,yLoc,60,16,hwnd,(HMENU)IDC_RADBTN3,hthisapp,NULL); // 显示文本 yLoc += 38; CreateWindow(L"Static",L"你喜欢哪一种颜色?", WS_CHILD | WS_VISIBLE | SS_SIMPLE, 10,yLoc,150,18,hwnd,NULL,hthisapp,NULL); //第二组单选按钮 yLoc += 22; CreateWindow(L"Button",L"蓝色",WS_CHILD | WS_VISIBLE | BS_AUTORADIOBUTTON | WS_GROUP, 12,yLoc,60,16,hwnd,(HMENU)IDC_RADBTNBLUE,hthisapp,NULL); yLoc += 20; CreateWindow(L"Button",L"红色",WS_CHILD | WS_VISIBLE | BS_AUTORADIOBUTTON, 12,yLoc,60,16,hwnd,(HMENU)IDC_RADBTNRED,hthisapp,NULL); yLoc += 20; CreateWindow(L"Button",L"绿色",WS_CHILD | WS_VISIBLE | BS_AUTORADIOBUTTON, 12,yLoc,60,16,hwnd,(HMENU)IDC_RADBTNGREEN,hthisapp,NULL); } return 0; case WM_DESTROY: PostQuitMessage(0); //平安退出 return 0; default: return DefWindowProc(hwnd,msg,wParam,lParam); } return 0; } ~~~ 在创建单选框中,为什么有些加了WS_GROUP样式,而有些未加呢? 其实是这样的,既然单选按钮是单选的,那么,任何一个单选按钮都与其他的单选按钮是互斥的关系。所以,在同一个容器(本例是同一个窗口)中就需要把单选按钮进行分组。 同一组中的单选按钮相互排斥,他的分组方法: 顺序,以Tab键顺序为参考,这个不用我介绍,你随便打开一个窗口,然后多按几下Tab键你就懂了,如果不懂,那你真的无可救药了。 凡是设置了WS_GROUP的单选框做为一组中的首元素,随后的所有单选按钮都和它在同一组,直到下一个设置了WS_GROUP样式的单选按钮。用上面的例子来说吧。 性别一组中,第一个应用了WS_GROUP的是“男”,随后的“女”和“人妖”都与“男”在同一组,因为后面一个“蓝色”设置了WS_GROUP样式。所以, 第一组为:男,女,人妖; 第二组为:蓝色,红色,绿色。 显然,用了BS_AUTORADIOBUTTON后,系统就会自动处理选择状态了。 ![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-06-14_575fd2d207d29.PNG) 完整的代码清单如下: ~~~ #include #include #include //包含头文件 // 导入静态库 #pragma comment(lib, "Comctl32.lib") // 开启视觉效果 Copy from MSDN #pragma comment(linker,"\"/manifestdependency:type='win32' \ name='Microsoft.Windows.Common-Controls' version='6.0.0.0' \ processorArchitecture='*' publicKeyToken='6595b64144ccf1df' language='*'\"") // 先声明一个WindowProc回调 LRESULT CALLBACK WindowProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam); // 入口点 int WINAPI wWinMain(HINSTANCE hTheApp, HINSTANCE hPrevApp, LPWSTR lpCmd, int nShow) { PCWSTR cn = L"My"; // 窗口名 PCWSTR tt = L"应用程序"; // 窗口标题 // 设计窗口类 WNDCLASS wc = { sizeof(WNDCLASS) }; wc.hbrBackground = (HBRUSH)COLOR_WINDOW; wc.lpfnWndProc = WindowProc; wc.style = CS_HREDRAW | CS_VREDRAW; LoadIconMetric(hTheApp, IDI_APPLICATION, LIM_SMALL, &wc.hIcon); wc.lpszClassName = cn; wc.hInstance = hTheApp; RegisterClass(&wc); // 注册窗口类 // 创建窗口 HWND hwnd = CreateWindow(cn, tt,WS_OVERLAPPEDWINDOW, 28,34,400,330,NULL,NULL,hTheApp,NULL); if( !hwnd) { /* 如果窗口创建失败, 那继续执行也没有意义 长痛不如短痛,结束吧。 */ return 0; } ShowWindow(hwnd,nShow); //显示窗口 UpdateWindow(hwnd); //更新窗口 // 消息循环 MSG msg; while(GetMessage(&msg,NULL,0,0)) { TranslateMessage(&msg); DispatchMessage(&msg); //调度消息到WindowProc回调 } return 0; } /*-----------------------------------------------------------------*/ //控件ID #define IDC_RADBTN1 50001 #define IDC_RADBTN2 50002 #define IDC_RADBTN3 50003 #define IDC_RADBTNBLUE 51001 #define IDC_RADBTNRED 51002 #define IDC_RADBTNGREEN 51003 #define IDC_BTN_OK 1107 //确定按钮ID /*-----------------------------------------------------------------*/ // 消息处理函数 LRESULT CALLBACK WindowProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam) { switch(msg) { case WM_CREATE: { // 获取当前实例句柄 HINSTANCE hthisapp = (HINSTANCE)GetWindowLongPtr(hwnd, GWLP_HINSTANCE); // 纵坐标,控件将以此作为基准, // 排列时依次增加 int yLoc = 0; // 用来显示文本 yLoc += 10; CreateWindow(L"Static",L"请问你的性别是:", SS_SIMPLE | WS_CHILD | WS_VISIBLE, 10,yLoc,160,18, hwnd, NULL, hthisapp, NULL); // 第一组单选按钮 yLoc += 22; CreateWindow(L"Button",L"男", WS_CHILD | WS_VISIBLE | BS_AUTORADIOBUTTON | WS_GROUP, 12, yLoc, 60, 16, hwnd, (HMENU)IDC_RADBTN1, hthisapp,NULL); yLoc += 20; CreateWindow(L"Button",L"女", WS_CHILD | WS_VISIBLE | BS_AUTORADIOBUTTON, 12,yLoc, 60, 16, hwnd,(HMENU)IDC_RADBTN2,hthisapp,NULL); yLoc += 20; CreateWindow(L"Button",L"人妖",WS_CHILD | WS_VISIBLE | BS_AUTORADIOBUTTON, 12,yLoc,60,16,hwnd,(HMENU)IDC_RADBTN3,hthisapp,NULL); // 显示文本 yLoc += 38; CreateWindow(L"Static",L"你喜欢哪一种颜色?", WS_CHILD | WS_VISIBLE | SS_SIMPLE, 10,yLoc,150,18,hwnd,NULL,hthisapp,NULL); //第二组单选按钮 yLoc += 22; CreateWindow(L"Button",L"蓝色",WS_CHILD | WS_VISIBLE | BS_AUTORADIOBUTTON | WS_GROUP, 12,yLoc,60,16,hwnd,(HMENU)IDC_RADBTNBLUE,hthisapp,NULL); yLoc += 20; CreateWindow(L"Button",L"红色",WS_CHILD | WS_VISIBLE | BS_AUTORADIOBUTTON, 12,yLoc,60,16,hwnd,(HMENU)IDC_RADBTNRED,hthisapp,NULL); yLoc += 20; CreateWindow(L"Button",L"绿色",WS_CHILD | WS_VISIBLE | BS_AUTORADIOBUTTON, 12,yLoc,60,16,hwnd,(HMENU)IDC_RADBTNGREEN,hthisapp,NULL); // 创建一个确定按钮 CreateWindow(L"Button",L"确定",WS_VISIBLE | WS_CHILD | BS_DEFPUSHBUTTON, 230,180,80,27,hwnd,(HMENU)IDC_BTN_OK,hthisapp,NULL); } return 0; case WM_DESTROY: PostQuitMessage(0); //平安退出 return 0; default: return DefWindowProc(hwnd,msg,wParam,lParam); } return 0; } ~~~
';