C语言实例教程(PDF格式)-第50部分
按键盘上方向键 ← 或 → 可快速上下翻页,按键盘上的 Enter 键可回到本书目录页,按键盘上方向键 ↑ 可回到本页顶部!
————未阅读完?加入书签已便下次继续阅读!
内容
Dir 添加文件名列表到组合框的列表框中
FindString 在组合框的列表框中查找包括指定前缀的
第一个字符串
FindStringExact 在组合框的列表框中查找与指定字符串匹
配的字符串
SelectString 在组合框的列表框中查找字符串,如果找
到的话,在列表框中选择该字符串,并将
字符串复制到编辑控件中
DrawItem 当一个自绘制组合框的可视部分改变时由
框架调用
MeasureItem 在创建自绘制组合框时,由框架调用以判
断组合框的尺寸
pareItem 当将一新项插入到排序的自绘制框中时由
框架调用以判断项的相对位置
DeleteItem 当一列表项被从自绘制组合框中删除时由
框架调用
下面的示例程序演示了自绘制组合框的使用。
1。 使用AppWizard创建名为boDemo的基于对话框的工程,按图
6。54添加工程的主对话框 (IDD_BODEMO_DIALOG)中的各个控件。每
个控件的属性如表6。30所示。
2。 在ClassView中用鼠标右击boDemo classes,选择New Class命
令。上面的操作将弹出如图6。55所示的对话框,确认在Class type下
拉列表框'注' 中选择了MFC Class。然后在Name处输入新的类名
CClrboBox,在Base class下拉列表框中选择CboBox。如果需
要修改新类的头文件或实现文件的文件名,可以单击Change按钮,这
里,我们接受默认的文件名ClrboBox。cpp和ClrboBox。h。
…………………………………………………………Page 380……………………………………………………………
图6。 54 工程boDemo的主对话框
表6。 30 对话框IDD_BODEMO_DIALOG的控件属性设置
控件类 ID 属性值
型
组合框 IDC_CLRBO Type:Dropdown
Owner draw:
Fixed
Sort:真
Vertical
scroll:真
Has string:假
下压按 IDC_ADDCLR Caption:添加颜
钮 色(&A)
IDC_CHGCLR Caption:改变颜
色(&C)
静态控 IDC_STATICCLR Caption属性值为
件 空
3。 使用ClassWizard的Message Map选项卡在类CClrboBox中重载
基类的MeasureItem成员函数,其重载版本的代码如下:
void CClrboBox::MeasureItem(LPMEASUREITEMSTRUCT lpMeasureItemStruct)
{
// 由于组合框具有 CBS_OWNERDRAWFIXED 样式,因此以 0 为参数调用成员函数
// GetItemHeight 获得每一项的固定高度
lpMeasureItemStruct…》itemHeight=GetItemHeight(0);
…………………………………………………………Page 381……………………………………………………………
}
图6。 55 从CboBox派生新类CClrboBox
函数MeasureItem在自绘制样式的组合框创建时由框架调用。该函数
将每一项的高度放入MEASUREITEMSTRUCT结构的成员中。如果对话框
以CBS_OWNERDRAWVARIABLE样式创建,框架将为列表框中的每一项调
用一次该成员函数,否则,该成员函数只被调用一次。
接着,在CClrboBox的重载基类的DrawItem成员函数,其代码如
下:
void CClrboBox::DrawItem(LPDRAWITEMSTRUCT lpDrawItemStruct)
{
CDC* pDC=CDC::FromHandle(lpDrawItemStruct…》hDC);
COLORREF cr=(COLORREF)lpDrawItemStruct…》itemData;
// 注意到在出错的情况下,GetCurSel 和 GetItemData 返回 CB_ERR,而常量
// CB_ERR 被定义为 …1,这时不应把它视为一种系统颜色。
if (cr==CB_ERR)
cr=GetSysColor(COLOR_WINDOW);
if (lpDrawItemStruct…》itemAction & ODA_DRAWENTIRE)
{
…………………………………………………………Page 382……………………………………………………………
// 需要重绘整个项
// 以该项所对应的颜色填充整个项
CBrush br(cr);
pDC…》FillRect(&lpDrawItemStruct…》rcItem; &br);
// 反色居中显示该颜色的 RGB 组成
CString str;
str。Format(〃R: %d G: %d B: %d〃; GetRValue(cr); GetGValue(cr); GetBValue(cr));
CSize size;
size=pDC…》GetTextExtent(str);
CRect rect=lpDrawItemStruct…》rcItem;
COLORREF tcr;
tcr=~cr & 0x00FFFFFF; // 获得背景色的反色,不能简单的使用 ~cr
pDC…》SetTextColor(tcr);
pDC…》SetBkColor(cr);
pDC…》TextOut(rect。left+(rect。Width()…size。cx)/2;
rect。top+(rect。Height()…size。cy)/2; str);
}
if ((lpDrawItemStruct…》itemState & ODS_SELECTED) &&
(lpDrawItemStruct…》itemAction & (ODA_SELECT | ODA_DRAWENTIRE)))
{
// 选中状态由未选中变为选中,其边框被加亮显示
COLORREF crHilite=~cr & 0x00FFFFFF;
CBrush br(crHilite);
pDC…》FrameRect(&lpDrawItemStruct…》rcItem; &br);
}
if (!(lpDrawItemStruct…》itemState & ODS_SELECTED) &&
…………………………………………………………Page 383……………………………………………………………
(lpDrawItemStruct…》itemAction & ODA_SELECT))
{
// 选中状态由选中变为非选中,清除其边框的加亮显示
CBrush br(cr);
pDC…》FrameRect(&lpDrawItemStruct…》rcItem; &br);
}
}
对于自绘制组合框来说,成员函数DrawItem是需要重载的一个很重要
的成员函数。该函数在自绘制组合框的可视部分发生改变时由框架调
用。在默认情况下,该成员函数不做任何操作。其参数
lpDrawItemStruct所指向的DRAWITEMSTRUCT结构包括了重绘制所需要
的各种信息,如所需重绘的项、其设备上下文以及所执行的重绘行为
等。在该成员函数终止前,应用程序应该恢复由该DRAWITEMSTRUCT结
构所提供的为该显示上下文所选定图形设备接口。
由表6。30可知在本示例程序中所使用的自绘组合框中的可选项是有序
的,而它们都是一些颜色值,框架如何知道当一个新的颜色值被添加
到组合框的列表框中时,它应该处于哪个颜色值之前,哪个颜色值之
后呢?这时通过调用成员函数pareItem成员函数来实现的。如果
在创建组合框时指定了LBS_SORT样式,则必须重载该成员函数以提供
足够的理由来帮助框架对新添加入组合框的列表框中的颜色项进行排
序。这里,我们首先根据颜色亮度的大小来对颜色进行排序,对于亮
度相同的颜色,我们依次以从蓝色到红色的优先级来判定其相对位
置。这个操作是以下面的代码来实现的:
int CClrboBox::pareItem(LPPAREITEMSTRUCT lppareItemStruct)
{
// TODO: 添加判断指定项的排序顺序的代码
// 当项 1 在项 2 之前时返回 …1
// 当项 1 和项 2 顺序相同时返回 0
// 当项 1 在项 2 之后时返回 1
// 获得项 1 和项 2 的颜色值
COLORREF cr1 = (COLORREF)lppareItemStruct…》itemData1;
…………………………………………………………Page 384……………………………………………………………
COLORREF cr2 = (COLORREF)lppareItemStruct…》itemData2;
if (cr1 == cr2)
{
// 项 1 和项 2 具有相同的颜色
return 0;
}
// 进行亮度比较; 亮度低的排列顺序在前
int intensity1 = GetRValue(cr1) + GetGValue(cr1) + GetBValue(cr1);
int intensity2 = GetRValue(cr2) + GetGValue(cr2) + GetBValue(cr2);
if (intensity1 《 intensity2)
return …1;
else if (intensity1 》 intensity2)
return 1;
// 如果亮度相同; 按颜色进行排序; (蓝色最前; 红色最后)
if (GetBValue(cr1) 》 GetBValue(cr2))
return …1;
else if (GetGValue(cr1) 》 GetGValue(cr2))
return …1;
else if (GetRValue(cr1) 》 GetRValue(cr2))
return …1;
else
return 1;
}
上面的代码同时也说明了pareItem成员函数的不同返回值所代表
的不同含义。这里要注意的是,由于同亮度的不同颜色给人的眼睛的
亮度感觉是不一样的(这好比人耳对声音的高频段和低频段的听觉灵
敏度要比对中频段的听觉灵敏度要小一样),上面的排序结果给人感
…………………………………………………………Page 385……………………………………………………………
觉并不象我们所想象的那样,是通过颜色的亮度来进行的。但我们没
有必要在这个问题上过分的纠缠而浪费时间。另外我们解释一下,为
什么要使用~cr & 0x00FFFFFF来代替~cr。很多人会认为直接将颜色
值按位取反就可以得到其对比色,但事实不是这样的。这是因为如果
32位颜色值的高位字节不为零的话,该颜色值将不被当作一个RGB颜
色值,而使用某个32位值与0x00FFFFFF按位与恰可以使其高位字节为
零,而其它位则不变。
到目前为止我们完成了自绘制组合框对应的类CClrboBox的设计,
下面我们来看如何在程序中将类CClrboBox的对象与对话框模板中
的现存组合框相关联,这是使用函数SubclassDlgItem来实现的。首
先在类CboDemoDlg中添加一个类型为CClrboBox的成员变量
m_clrbo,其访问限制在本工程中是不重要的,可以将它设置为
protected。然后,在类CboDemoDlg的OnInitDialog成员函数中
的// TODO注释之后添加下面的一行代码,这行代码将对象
m_clrbo与ID为IDC_CLRBO相关联,第一个参数为控件的父窗口
的指针。
m_clrbo。SubclassDlgItem(IDC_CLRBO; this);
这样就可以通过CWnd的消息映射机制和消息传递路径在类
CClrboBox中处理IDC_CLRBO中事件了。比如当组合框中的项需
要重绘时,在CClrboBox中定义的DrawItem成员函数将被调用,在
正确的绘制组合框中的内容。
4。 这里,我们在初始时没有为组合框添加任何选择项,用户可以单
击如图6。54的对话框中所示的 “添加颜色”按钮向组合框的列表框中
添加新颜色项。单击该按钮首先将弹出一个颜色选择对话框,用户如
果从颜色选择对话框中选择了一种具体的颜色,该颜色将被添加到组
合框的列表框中以供选择。基于这个要求,我们为按钮IDC_ADDCLR的
BN_CLICKED事件添加如下的命令处理成员函数OnAddClr:
void CboDemoDlg::OnAddClr()
{
CColorDialog dlg(0; 0; this);
int iRes=dlg。DoModal();
if (iRes==IDOK)
{
…………………………………………………………Page 386……………………………………………………………
COLORREF cr=dlg。GetColor();
m_clrbo。AddString( (LPCTSTR)cr );
}
else
{
}
}
虽然使用颜色对话框在本书的前面内容中没有讲述过,但即使是对初
学者而言,上面的代码也是非常之简单的,我们这里就不过多的作讲
解了。
下面来看如何为 “改变颜色”按钮 (IDC_CHGCLR)添加单击命令处理成
员函数。我们希望用户在单击该按钮时,改变静态文本控件
IDC_CLRSTATIC的颜色以反映用户所选择的颜色。如果改变静态文本
控件的颜色呢?我们这里使用了将 自绘制静态文本控件的办法。这并
不是最简