大家也许熟悉WM_NOTIFY,控件通过WM_NOTIFY向父窗口发送消息。在WM_NOTIFY消息体中,部分控件会发送NM_CUSTOMDRAW告诉父窗口自己需要绘图。
也可以反射NM_CUSTOMDRAW消息,如:
ON_NOTIFY_REFLECT(NM_CUSTOMDRAW, OnCustomDraw) //需要自己加进去
afx_msg void OnCustomDraw(NMHDR *pNMHDR, LRESULT *pResult);
参数:
pNMHDR 说到底只是一个指针,大多数情况下它指向一个NMHDR结构对象,NMHDR结构如下:
typedef structtagNMHDR
{
HWND hwndFrom;
UINT idFrom;
UINT code;
} NMHDR;
其中:
hwndFrom 发送方控件的窗口句柄
idFrom 发送方控件的ID
code 通知代码
对于某些控件来说,pNMHDR则会解释成其它内容更丰富的结构对象的指针,如:对于列表控件来说,pNMHDR常常指向一个NMCUSTOMDRAW对象,NMCUSTOMDRAW结构如下:
typedef structtagNMCUSTOMDRAWINFO
{
NMHDR hdr;
DWORD dwDrawStage;
HDC hdc;
RECT rc;
DWORD dwItemSpec;
UINT uItemState;
LPARAM lItemlParam;
} NMCUSTOMDRAW, FAR* LPNMCUSTOMDRAW;
其中:
hdr NMHDR对象
dwDrawStage 当前绘制状态,其取值如下:
CDDS_POSTERASE
擦除循环结束
CDDS_POSTPAINT
绘制循环结束
CDDS_PREERASE
准备开始擦除循环
CDDS_PREPAINT
准备开始绘制循环
CDDS_ITEM
指定dwItemSpec, uItemState, lItemlParam参数有效
CDDS_ITEMPOSTERASE
列表项擦除结束
CDDS_ITEMPOSTPAINT
列表项绘制结束
CDDS_ITEMPREERASE
准备开始列表项擦除
CDDS_ITEMPREPAINT
准备开始列表项绘制
CDDS_SUBITEM
指定列表子项
hdc指定了绘制操作所使用的设备环境。
rc指定了将被绘制的矩形区域。
dwItemSpec列表项的索引
uItemState 当前列表项的状态,其取值如下:
CDIS_CHECKED
标记状态
CDIS_DEFAULT
默认状态
CDIS_DISABLED
禁止状态
CDIS_FOCUS
焦点状态
CDIS_GRAYED
灰化状态
CDIS_SELECTED
选中状态
CDIS_HOTLIGHT
热点状态
CDIS_INDETERMINATE
不定状态
CDIS_MARKED标注状态
lItemlParam 当前列表项的绑定数据
pResult指向状态值的指针,指定系统后续操作,依赖于dwDrawStage:
当dwDrawStage为CDDS_PREPAINT,pResult含义如下:
CDRF_DODEFAULT
默认操作,即系统在列表项绘制循环过程不再发送NM_CUSTOMDRAW
CDRF_NOTIFYITEMDRAW
指定列表项绘制前后发送消息
CDRF_NOTIFYPOSTERASE
列表项擦除结束时发送消息
CDRF_NOTIFYPOSTPAINT
列表项绘制结束时发送消息
当dwDrawStage为CDDS_ITEMPREPAINT,pResult含义如下:
CDRF_NEWFONT
指定后续操作采用应用中指定的新字体
CDRF_NOTIFYSUBITEMDRAW
列表子项绘制时发送消息
CDRF_SKIPDEFAULT
系统不必再绘制该子项
以下是一个利用NM_CUSTOMDRAW消息绘制出的多色列表框的例子:
对应代码如下:
void CCoolList::OnCustomDraw //从CListCtrl派生(NMHDR *pNMHDR, LRESULT *pResult)
{
//类型安全转换
NMLVCUSTOMDRAW* pLVCD = reinterpret_cast(pNMHDR);
*pResult = 0;
//指定列表项绘制前后发送消息
if(CDDS_PREPAINT == pLVCD->nmcd.dwDrawStage)
{
*pResult =CDRF_NOTIFYITEMDRAW;
}
else if(CDDS_ITEMPREPAINT == pLVCD->nmcd.dwDrawStage)
{
//奇数行
if(pLVCD->nmcd.dwItemSpec % 2)
pLVCD->clrTextBk = RGB(255, 255, 128);
//偶数行
elsepLVCD->clrTextBk = RGB(128, 255, 255);
//继续
*pResult =CDRF_DODEFAULT;
}
}
注意到上例采取了3.1所推荐的第2种实现方法,派生了一个新类CCoolList。
总体步骤:
派生CCoolList类
BEGIN_MESSAGE_MAP(CMyListCtrl, CListCtrl)
//{{AFX_MSG_MAP(CMyListCtrl)
// NOTE - the ClassWizard will add and remove mapping macros here.
ON_NOTIFY_REFLECT(NM_CUSTOMDRAW, OnCustomDraw) //自己添加 //}}AFX_MSG_MAP
END_MESSAGE_MAP()
定义CListCtrl Control m_list变量,再将CListCtrl 改为CCoolList