WINDOWS 提供了几个预定义的窗口类以方便我们的使用。大多数时间内,我们把它们用在对话框中,所以我们一般就它们叫做子窗口控件。子窗口控件会自己处理消息,并在自己状态发生改变时通知父窗口。这样就大大地减轻了我们的编程工作,所以我们应尽可能地利用它们。本课中我们把这些控件放在窗口中以简化程序,但是大多数时间内子窗口控件都是放在对话框中的。我们示例中演示的子窗口控件包括:按钮、下拉菜单、检查框、单选按钮、编辑框等。使用子窗口控件时,先调用 CreateWindow 或 CreateWindowEx。在这里由于WINDOWS 已经注册了这些子控件,所以无须我们再注册。当然我们不能改变它们的类名称。譬如:如果您想产生一个按钮,在调用上述两个函数时就必须指定类名为"button"。其他必须指定的参数还有父窗口的句柄和将要产生的子控件的ID号。子控件的ID号是用来标识子控件的,故也必须是唯一 的。子控件产生后,当其状态改变时将会向父窗口发送消息。一般我们应在父窗口的WM_CREATE消息中产生字控件。子控件向父窗口发送的消息是 WM_COMMAND,并在传递的参数wPara的底位中包括控件的ID号,消息号在wParam的高位,lParam中则包括了子控件的窗口的句柄。各类控件有不同的消息代码集,详情请参见WIN32 API参考手册。父窗口也可以通过调用函数SendMessage向子控件发送消息,其中第一个参数是子控件的窗口句柄,第二个参数是要发送的消息号,附加的参数可以在wParam和lParam中传递,其实只要知道了某个窗口的句柄就可以用该函数向其发送相关消息。所以产生了子窗口后必须处理 WM_COMMAND消息以便可以接收到子控件的消息。
取父窗口的窗口句柄:hwndParent = GetParent (hwnd) ;发送消息:SendMessage (hwndParent, message, wParam, lParam) ;预定义的控件有:按钮、复选框、编辑方块、清单方块、下拉式清单方块、静态字符串标签和滚动条。当使用预定义的控件时,不必为其注册窗口类,窗口类已经存在于Windows中,并有一个预先定义的名字。您只需在CreateWindow()的参数中指出窗口类名字。CreateWindow()的窗口样式参数准确地定义了子窗口控件的外形和功能。Windows内建了这些控件的窗口消息处理程序。======================================== 按钮控件 创建子窗口时指定窗口类为"button".显示窗口的CreateWindow参数如下:Class name(类别名称) TEXT ("button")Window text(窗口文字) 一个c字符串szTextWindow style(窗口样式) WS_CHILD | WS_VISIBLE | 按钮样式(下边有说明)x position(x位置)y position(y位置)Width(宽度)Height(高度)Parent window(父窗口)Child window ID(子窗口ID) 要转换为HMENU类型.如(HMENU) i Instance handle(执行实体句柄) 执行实例句柄Extra parameters(附加参数) NULL其中的按钮样式为:BS_PUSHBUTTONBS_DEFPUSHBUTTONBS_CHECKBOX (带有复选框.复选框的状态要手动发送BM_SETCHECK消息设置)BS_AUTOCHECKBOX (带有复选框.复选框的状态自动设置)BS_RADIOBUTTON (带单选按钮.其状态要手动发送BM_SETCHECK消息设置)BS_3STATEBS_AUTO3STATEBS_GROUPBOX (分组方块. 它只是一个标题框. 不处理输入.)BS_AUTORADIOBUTTONBS_OWNERDRAW按钮会向父窗口发送WM_COMMAND消息.参数:LOWORD(wParam) 为子窗口ID.HIWORD(wParam) 为通知码.lParam 为子窗口句柄.通知码HIWORD(wParam)有:BN_CLICKEDBN_PAINTBN_HILITE or BN_PUSHEDBN_UNHILITE or BN_UNPUSHEDBN_DISABLEBN_DOUBLECLICKED or BN_DBLCLKBN_SETFOCUSBN_KILLFOCUS父窗口也可以向按钮发送消息.BM_GETCHECK (复选框的选定标记)BM_SETCHECKBM_GETSTATE ("正常状态" 还是"按下状态" )BM_SETSTATEBM_SETSTYLE (改变按钮样式)BM_CLICKBM_GETIMAGEBM_SETIMAGE要得到控件ID用GetDlgCtrlId(hwndChild);要得到控件的子窗口句柄用GetDlgItem(hwndParent, id); 改变按钮的文字用SetWindowText (hwnd, pszString) ;取按钮的当前文字用iLength = GetWindowText (hwnd, pszBuffer, iMaxLength) ;======================================显示/隐藏子窗口如果子窗口的窗口类的样式中没有WS_VISIBLE. 则在没有呼叫ShowWindow之前不会显示窗口.显示子窗口用:ShowWindow (hwndChild, SW_SHOWNORMAL) ;隐藏子窗口用:ShowWindow (hwndChild, SW_HIDE) ;查看子窗口是否可见用:IsWindowVisible (hwndChild) ;不可用/启用 子窗口要使按钮不可用(文字变为灰色).用:EnableWindow (hwndChild, FALSE) ;恢复为可用:EnableWindow (hwndChild, TRUE) ;判断是否被启用:IsWindowEnabled (hwndChild) ;输入焦点用户使用按钮时.按钮获得输入焦点而其父窗口失去输入焦点.这时父窗口先收到WM_KILLFOCUS消息(wParam参数为获得输入焦点的窗口的句柄).然后获得输入焦点的窗口(按钮子窗口)收到一个WM_SETFOCUS消息(wParam参数为失去输入焦点的窗口的句柄).控件与颜色系统颜色Windows保留了29种系统颜色以供各种显示使用(例如:菜单颜色.菜单文字颜色.窗口颜色等等)。您可以使用GetSysColor和SetSysColors来获得和设定这些颜色。设定的系统颜色只在目前Windows对话过程中有效。要在按钮中显示图标或位图,您可以用BS_ICON或BS_BITMAP样式,并用BM_SETIMAGE消息设定位图。对于BS_OWNERDRAW样式的按钮,它允许完全自由地绘制按钮。====================================
静态控件
创建子窗口时指定窗口类为"static".它既不接收鼠标或键盘输入,也不向父窗口发送WM_COMMAND消息。设置静态控件的文字用SetWindowText.================================ 滚动条控件 创建子窗口时指定窗口类为"scrollbar".它不向父窗口发送WM_COMMAND消息,而是像窗口滚动条那样发送WM_VSCROLL和WM_HSCROLL消息。可以通过lParam参数来区分窗口滚动条与滚动条控件。对子窗口滚动条其值为0,对于滚动条控件其值为滚动条子窗口的句柄。wParam对窗口滚动条和滚动条控件来说含义相同。设置滚动条控件用的函数和设置窗口滚动条一样:SetScrollRange (hwndScroll, SB_CTL, iMin, iMax, bRedraw) ;SetScrollPos (hwndScroll, SB_CTL, iPos, bRedraw) ;SetScrollInfo (hwndScroll, SB_CTL, &si, bRedraw) ;区别是设置滚动条控件时.第一个参数是控件子窗口句柄而不是父窗口句柄.第二个参数是SB_CTL而不是SB_VERT或SB_HORZ.===========================为控件指定窗口函数控件的窗口函数是windows内部的. 但可以用GWL_WNDPROC做参数通过GetWindowLong函数得到它的函数地址. 而且可以用SetWindowLong给它重新指定一个新的窗口函数(新的函数也要是callback函数).如:OldScroll = (WNDPROC) SetWindowLong (hwndScroll, GWL_WNDPROC, (LONG) ScrollProc)) ; //返回值是原来的窗口函数的地址.在需要用Tab键在控件之间切换输入焦点时. 由于控件获得输入焦点后所有的键盘消息都发送给控件的窗口函数了.而控件原来的窗口函数并不处理Tab键按下的消息.这时就可以用上边的方法给控件重新设定一个窗口消息处理函数并在其中处理Tab键.最后再在新的窗口函数中呼叫原来的窗口函数处理其它消息:return CallWindowProc (OldScroll, hwnd, message, wParam,lParam) ;这样就可以处理Tab键了.就像可以通过SetWindowLong给窗口重新设置窗口函数一样.可以用SetClassLong来设置某个窗口的窗口类的一些东西.例如下边重新设置窗口类的画刷: SetClassLong (hwnd, GCL_HBRBACKGROUND, (LONG)CreateSolidBrush (RGB (color[0], color[1], color[2])));=============================== 编辑控件 创建子窗口时用"edit".如:hwndEdit = CreateWindow (TEXT ("edit"), NULL, WS_CHILD | WS_VISIBLE | WS_HSCROLL | WS_VSCROLL | WS_BORDER | ES_LEFT | ES_MULTILINE | ES_AUTOHSCROLL | ES_AUTOVSCROLL, 0, 0, 0, 0, hwnd, (HMENU) ID_EDIT, ((LPCREATESTRUCT) lParam) -> hInstance, NULL) ;编辑控件的样式:文字对齐: ES_LEFT、ES_RIGHT和ES_CENTER多行: ES_MULTILINE (缺省为单行)在单行样式中.要文字水平卷动: ES_AUTOHSCROLL在多行样式中.要文字水平卷动(不是出现滚动条)(这阻止了自动换行): ES_AUTOHSCROLL在多行样式中.要文字垂直卷动(不是出现滚动条): ES_AUTOVSCROLL在多行样式中.要显示滚动条用WS_HSCROLL和WS_VSCROLL编辑控件缺省时没有边框.要显示边框用 WS_BORDER选择编辑控件的文字时.文字反白显示.但编辑控件失去焦点后文字将不再加亮显示.要使失去焦点后选择的文字仍然加亮显示用: ES_NOHIDESEL编辑控件给父窗口消息处理程序发送WM_COMMAND消息.消息参数为LOWORD (wParam) 子窗口IDHIWORD (wParam) 通知码lParam 子窗口句柄其中通知码为:EN_SETFOCUS 编辑控件已经获得输入焦点EN_KILLFOCUS 编辑控件已经失去输入焦点EN_CHANGE 编辑控件的内容将改变EN_UPDATE 编辑控件的内容已经改变EN_ERRSPACE 编辑控件执行已经超出中间EN_MAXTEXT 编辑控件在插入时执行超出空间EN_HSCROLL 编辑控件的水平滚动条已经被按下EN_VSCROLL 编辑控件的垂直滚动条已经被按下要处理tab键切换输入焦点或处理Enter键. 可以给它重新指定一个窗口函数并在其中拦截.要在编辑区插入文字用:GetWindowTextLengthGetWindowTextSetWindowText给编辑控件发送消息:SendMessage (hwndEdit, WM_CUT, 0, 0) ; //剪贴 SendMessage (hwndEdit, WM_COPY, 0, 0) ; //复制 SendMessage (hwndEdit, WM_CLEAR, 0, 0) ; //删除选择文字SendMessage (hwndEdit, WM_PASTE, 0, 0) ; //粘贴SendMessage (hwndEdit, EM_GETSEL, (WPARAM) &iStart,(LPARAM) &iEnd) ;//取得目前选择的起始位置和末尾位置.(末尾位置是选择的最后一个文字的位置加1)SendMessage (hwndEdit, EM_SETSEL, iStart, iEnd) ; //选择文字SendMessage (hwndEdit, EM_REPLACESEL, 0, (LPARAM) szString) ;//用其他文字替换目前选择的文字iCount = SendMessage (hwndEdit, EM_GETLINECOUNT, 0, 0) ;//取得多行编辑控件的行数iOffset = SendMessage (hwndEdit, EM_LINEINDEX, iLine, 0) ;//对任何特定的行(从0行开始),您可以取得距离编辑缓冲区文字开头的偏移量.//iLine为-1时返回光标所在行的偏移量.iLength = SendMessage (hwndEdit, EM_LINELENGTH, iLine, 0) ;//取得第iLine行的长度iLength = SendMessage (hwndEdit, EM_GETLINE, iLine, (LPARAM) szBuffer) ;//将第iLine行复制到缓冲区szBuffer中========================== 列表控件 创建列表子窗口时,用"listbox"作为窗口类.列表样式:LBS_NOTIFY 允许父窗口接收列表的WM_COMMAND消息(缺省时不向父窗口发送WM_COMMAND消息)LBS_SORT 对列表中的项目排序LBS_MULTIPLESEL 列表是多选的(缺省时为单选)LBS_NOREDRAW 防止在向列表增加项目时自动重画列表样式LBS_STANDARD包含了最常用的样式:(LBS_NOTIFY | LBS_SORT | WS_VSCROLL | WS_BORDER)将字符串放入列表:SendMessage (hwndList, LB_INSERTSTRING, i, (LPARAM) szString) ;第一个参数是列表控件的窗口句柄.第二个参数LB_INSERTSTRING表示要插入一个项目.第三个参数表示要插入的位置.(位置值从0开始.0表示最上边第一个位置.-1表示插入最后)第四个参数是要插入的字符串(字符串以0结尾).函数返回0表示正常完成.如果列表控件包含LBS_SORT样式.则插入一个字符串时可以将第2个参数设置为LB_ADDSTRING.这样字符串就会被自动插入到一个位置.如:SendMessage (hwndList, LB_ADDSTRING, 0, (LPARAM) szString) ;要在列表控件中删除一个字符串.只要指定第2个参数为LB_DELETESTRING. 例如:SendMessage (hwndList, LB_DELETESTRING, iIndex, 0) ;要删除所有列表中的内容.则指定第2个参数为LB_RESETCONTENT. 如:SendMessage (hwndList, LB_RESETCONTENT, 0, 0) ;在向列表增加或删除字符串时.列表控件会自动被它的窗口函数重画.如果你有许多字符串需要增加.你可能希望在所有字符串增加完成前暂时阻止列表的自动重画.这要:SendMessage (hwndList, WM_SETREDRAW, FALSE, 0) ;在增加完成后再恢复列表控件的自动重画就可以了:SendMessage (hwndList, WM_SETREDRAW, TRUE, 0) ;要取得现在列表控件中的项目数用:iCount = SendMessage (hwndList, LB_GETCOUNT, 0, 0) ;在单选列表控件中.要选择一个项目(它会被加亮显示)用:SendMessage (hwndList, LB_SETCURSEL, iIndex, 0) ;//iIndex参数为要选择第几个项目.指定为-1表示取消所有选择.在单选列表控件中.也可以根据一个字符串的第一个字母来选择一个项目.如:iIndex = SendMessage (hwndList, LB_SELECTSTRING, iIndex, (LPARAM) szSearchString) ;表示从第iIndex位置开始搜索.如果哪个项目的开始字母与 szSearchString相同.则选择该项目.并返回该项目的位置. 没有匹配的项目时返回-1.在单选列表控件中.要取得当前选择项目的索引.用:iIndex = SendMessage (hwndList, LB_GETCURSEL, 0, 0) ;如果没有被选项目.函数返回LB_ERR(值为-1).在单选列表控件中.将某个项目复制到一个字符缓冲区.用:iLength = SendMessage (hwndList, LB_GETTEXT, iIndex, (LPARAM) szBuffer) ;则将列表控件中索引为iIndex的项目复制到szBuffer. 并返回字符串的长度iLength.(为了使szBuffer缓冲区足够大.你可以用LB_GETTEXTLEN做参数先取得该项目的字符串长度.)对于多选列表控件:可以使用LB_SETSEL来设定某特定项目的选择状态(不会影响其他项目的选择状态):SendMessage (hwndList, LB_SETSEL, wParam, iIndex) ;参数wParam为0时.取消选择. 为-1时.选择/取消所有项目. 为其它值时. 选择第iIndex个项目.查看某项目的选择状态:iSelect = SendMessage (hwndList, LB_GETSEL, iIndex, 0) ;//项目被选择时返回非0. 否则返回0.要使清单控件获得输入焦点用:SetFocus (hwndList) ;清单控件发送给父窗口的消息是WM_COMMAND. 参数如下:LOWORD (wParam) 子窗口IDHIWORD (wParam) 通知码lParam 子窗口句柄 其中通知码的值如下:LBN_ERRSPACE -2 表示已经超出执行空间LBN_SELCHANGE 1 表示目前选择已经被改变LBN_DBLCLK 2 说明某项目已经被鼠标双击LBN_SELCANCEL 3LBN_SETFOCUS 4LBN_KILLFOCUS 5 只有清单窗口样式包括LBS_NOTIFY时,清单控件才会向父窗口发送LBN_SELCHANGE和LBN_DBLCLK。文件列表要将文件目录列表填入清单列表.用:SendMessage (hwndList, LB_DIR, iAttr, (LPARAM) szFileSpec) ;//iAttr参数是文件属性代码,其低字节是文件属性代码:iAttr 值 属性DDL_READWRITE 0x0000 普通文件DDL_READONLY 0x0001 只读文件 DDL_HIDDEN 0x0002 隐藏文件 DDL_SYSTEM 0x0004 系统文件 DDL_DIRECTORY 0x0010 子目录 DDL_ARCHIVE 0x0020 归档位设立的档案//高字节提供了一些对所要求项目的附加控制:iAttr 值 属性 DDL_DRIVES 0x4000 包括磁盘驱动器句柄 DDL_EXCLUSIVE 0x8000 互斥搜索
ComboBox
如下图所示,显示了三种不同风格的Combo Box样式。当然,现在这样看不出第一种与第三种之间的区别,但是第二种与其他两种的区别是明显的,第二种的列表框始终是出于现实状态的。
Combo Box:
一个下拉组合框控件拥有文本框及列表框的功能。它允许用户通过输入文本到下拉组合框中或者从下拉列表中选择相应的条目。Combo Box拥有三种风格Drop-down combo box 可输入文本或在下拉列表中选择,当鼠标点击右方的下拉箭头时显示下拉列表框 (CBS_DROPDOWN)Simple combo box 可输入文本或在列表中选择,列表框一直显示 (CBS_SIMPLE)Drop-down list box 只能在下拉列表中选择,当鼠标点击右方的下拉箭头时显示下拉列表框 (CBS_DROPDOWNLIST)
有了直观的感受,现在来就来看如何实现它。其实它的实现并不是很困难,通过一个CreateWindow函数即可实现,如果需要对它的功能进行添加或者是美化外观之类的,可能就需要对它的窗口过程进行重写了(只是推测,并未实现,有兴趣的话可以动手试一试),在CreateWindow之后调用SetWindowLong设置其窗口过程处理函数。说了这么多,先来看看代码吧!
- #include <WINDOWS.H>
- #include <COMMCTRL.H>
- #pragma comment(lib, "comctl32.lib")
- #define DROP_DOWN_COMBO 1
- #define SIMPLE_COMBO 2
- #define DROP_DOWN_LIST 3
- LRESULT CALLBACK WndProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam);
- int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nShowCmd)
- {
- WNDCLASS wc;
- HWND hwnd;
- HWND drop_down_hwnd, simple_combo, drop_down_list;
- MSG msg;
- int i;
- wc.cbClsExtra = 0;
- wc.cbWndExtra = 0;
- wc.hbrBackground = (HBRUSH)GetStockObject(WHITE_BRUSH);
- wc.hCursor = LoadCursor(hInstance, IDC_ARROW);
- wc.hIcon = LoadIcon(hInstance, IDI_APPLICATION);
- wc.hInstance = hInstance;
- wc.lpfnWndProc = WndProc;
- wc.lpszClassName = TEXT("TEST");
- wc.lpszMenuName = NULL;
- wc.style = CS_HREDRAW | CS_VREDRAW;
- if (!RegisterClass(&wc))
- {
- MessageBox(NULL, TEXT("Register class error!"), TEXT("ERROR"), MB_ICONERROR | MB_OK);
- return 0;
- }
- hwnd = CreateWindow(TEXT("TEST"), TEXT("TEST"), WS_OVERLAPPEDWINDOW, 10, 10, 300, 400, NULL,
- NULL, hInstance, NULL);
- if (hwnd == NULL)
- {
- MessageBox(NULL, TEXT("Create window error!"), TEXT("ERROR"), MB_ICONERROR | MB_OK);
- return 0;
- }
- /* Create drop-down combo box */
- drop_down_hwnd = CreateWindow("combobox", NULL, WS_CHILD | WS_VISIBLE | WS_VSCROLL | CBS_DROPDOWN | CBS_HASSTRINGS,
- 10, 20, 80, 100, hwnd, (HMENU)DROP_DOWN_COMBO, hInstance, NULL);
- /* Add some text into the drop-down combo box */
- for (i=0;i<=20;i++)
- {
- char temp[10];
- SendMessage(drop_down_hwnd,CB_ADDSTRING,0,(LPARAM)itoa(i,temp,10));
- }
- /* Create simple combo box */
- simple_combo = CreateWindow("combobox", NULL, WS_CHILD | WS_VISIBLE | WS_VSCROLL | CBS_SIMPLE | CBS_HASSTRINGS,
- 100, 20, 80, 100, hwnd, (HMENU)SIMPLE_COMBO, hInstance, NULL);
- /* Add some text into the simple combo box */
- for (i=0;i<=20;i++)
- {
- char temp[10];
- SendMessage(simple_combo,CB_ADDSTRING,0,(LPARAM)itoa(i,temp,10));
- }
- /* Create drop-down list combo box */
- drop_down_list = CreateWindow("combobox", NULL, WS_CHILD | WS_VISIBLE | WS_VSCROLL | CBS_DROPDOWNLIST | CBS_HASSTRINGS,
- 190, 20, 80, 100, hwnd, (HMENU)DROP_DOWN_LIST, hInstance, NULL);
- /* Add some text into the drop-down list combo box */
- for (i=0;i<=20;i++)
- {
- char temp[10];
- SendMessage(drop_down_list,CB_ADDSTRING,0,(LPARAM)itoa(i,temp,10));
- }
- ShowWindow(hwnd, nShowCmd);
- UpdateWindow(hwnd);
- while (GetMessage(&msg, NULL, 0, 0))
- {
- TranslateMessage(&msg);
- DispatchMessage(&msg);
- }
- return 0;
- }
- LRESULT CALLBACK WndProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam)
- {
- switch (msg)
- {
- case WM_CREATE:
- return 0;
- case WM_DESTROY:
- PostQuitMessage(0);
- return 0;
- }
- return DefWindowProc(hwnd, msg, wParam, lParam);
- }
可以看到在程序的一开始我引入了COMMCTRL.H这个头文件,需要说明一下,在这个头文件中包含了windows的一些通用控件,比如说上面创建的combo box就是其中之一。在创建combo box窗口时,分别用到了CBS_DROPDOWN、CBS_SIMPLE和CBS_DROPDOWNLIST三种风格,接下来就是给控件发送CB_ADDSTRING消息添加条目,对于更多的消息可以在MSDN上面去查看。此外,还有一个当时没注意的问题,把我给困扰了很久,那就是窗口的高度,之前写程序的时候,没有注意这个问题,写出来的控件下拉框始终弹不出来,有种说法是要重写窗口处理函数,在其中处理WM_LBUTTONDOWN消息,并发送CB_SHOWDROPDOWN消息来显示,还需处理CBN_SELCHANGE消息,其实并没有必要。默认的就可以到达那样的效果,如果是下拉列表未弹出来,检查你的窗口高度是否合理。当然,在combo box插入的条目可能会很多,固定的窗口高度并不可行,因为这时只能通过键盘的上下箭头来移动选取,使用起来不是很方便。这时就需要加入垂直滚动条了,在创建时加入WS_VSCROLL即可,这下使用起来就方便多了。