转自:http://blog.csdn.net/xiaodan007/article/details/6197478
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字符串szText
Window 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_PUSHBUTTON
BS_DEFPUSHBUTTON
BS_CHECKBOX (带有复选框.复选框的状态要手动发送BM_SETCHECK消息设置)
BS_AUTOCHECKBOX (带有复选框.复选框的状态自动设置)
BS_RADIOBUTTON (带单选按钮.其状态要手动发送BM_SETCHECK消息设置)
BS_3STATE
BS_AUTO3STATE
BS_GROUPBOX (分组方块. 它只是一个标题框. 不处理输入.)
BS_AUTORADIOBUTTON
BS_OWNERDRAW
按钮会向父窗口发送WM_COMMAND消息.参数:
LOWORD(wParam) 为子窗口ID.
HIWORD(wParam) 为通知码.
lParam 为子窗口句柄.
通知码HIWORD(wParam)有:
BN_CLICKED
BN_PAINT
BN_HILITE or BN_PUSHED
BN_UNHILITE or BN_UNPUSHED
BN_DISABLE
BN_DOUBLECLICKED or BN_DBLCLK
BN_SETFOCUS
BN_KILLFOCUS
父窗口也可以向按钮发送消息.
BM_GETCHECK (复选框的选定标记)
BM_SETCHECK
BM_GETSTATE ("正常状态" 还是"按下状态" )
BM_SETSTATE
BM_SETSTYLE (改变按钮样式)
BM_CLICK
BM_GETIMAGE
BM_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) 子窗口ID
HIWORD (wParam) 通知码
lParam 子窗口句柄
其中通知码为:
EN_SETFOCUS 编辑控件已经获得输入焦点
EN_KILLFOCUS 编辑控件已经失去输入焦点
EN_CHANGE 编辑控件的内容将改变
EN_UPDATE 编辑控件的内容已经改变
EN_ERRSPACE 编辑控件执行已经超出中间
EN_MAXTEXT 编辑控件在插入时执行超出空间
EN_HSCROLL 编辑控件的水平滚动条已经被按下
EN_VSCROLL 编辑控件的垂直滚动条已经被按下
要处理tab键切换输入焦点或处理Enter键. 可以给它重新指定一个窗口函数并在其中拦截.
要在编辑区插入文字用:
GetWindowTextLength
GetWindowText
SetWindowText
给编辑控件发送消息:
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) 子窗口ID
HIWORD (wParam) 通知码
lParam 子窗口句柄
其中通知码的值如下:
LBN_ERRSPACE -2 表示已经超出执行空间
LBN_SELCHANGE 1 表示目前选择已经被改变
LBN_DBLCLK 2 说明某项目已经被鼠标双击
LBN_SELCANCEL 3
LBN_SETFOCUS 4
LBN_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 互斥搜索