(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;
}
~~~
';