自绘CListCtrl时发现ON_WM_MEASUREITEM_REFLECT和DrawItem(LPDRAWITEMSTRUCT lpMeasureItemStruct) 无响应

在MFC中使用自绘CListCtrl时,发现在对话框中绑定自定义类后,ON_WM_MEASUREITEM_REFLECT和DrawItem函数未被调用。原因是MFC的机制导致的。正确解决方案是不在对话框中直接添加列表,而是手动创建并设置风格,例如在OnInitDialog中创建。遵循此方法后,自定义的CMFCCtrlList子类能够正常进行重绘操作。

自绘CListCtrl时发现ON_WM_MEASUREITEM_REFLECT和DrawItem(LPDRAWITEMSTRUCT lpMeasureItemStruct) 无响应

一般情况下,我们都是在 dialog 里面 放一个 list 控件,然后 用自己的类绑定自己的自绘对象,如同下面:

1.  在 dialog 里面 加一个 list 列表

2. 然后 绑定这个列表

哈哈,那么恭喜你,你的自绘 CListCtrl 控件 重写的ON_WM_MEASUREITEM_REFLECT和DrawItem(LPDRAWITEMSTRUCT lpMeasureItemStruct) 无响应的。

为什么呢? 自己想吧,其他不多说,只是 MFC 的机制很日怪!

给出正确的做法吧,在 dialog 里面 去掉 list, 在去掉 DDX_Control(pDX, IDC_LIST_Glass, m_glassListCtrl);

然后在 适当的地方加上,比如下面:

BOOL CPickingPanel::OnInitDialog()
{
    CBackgroundDlg::OnInitDialog();

    DWORD dwListStyle = WS_VISIBLE | WS_CHILD | LVS_REPORT | LVS_OWNERDRAWFIXED;
    if (m_list.Create(dwListStyle, CRect(0, 0, 0, 0), this, 0) == -1)
        return FALSE;

....

现在 你从 CMFCCtrlList 的子类 就可以 重绘了。

成都本征科技, 三维测量系统 定制!!

### 解决方案 `ON_WM_MEASUREITEM_REFLECT` 无响应的问题通常与控件的制模式以及消息映射机制有关。以下是可能的原因及其解决方案: #### 1. **Owner Draw 属性未启用** 如果 `CListCtrl` 或其他控件有设置为 Owner Draw 模式,则 `WM_MEASUREITEM` 其反射版本 `WM_MEASUREITEM_REFLECT` 不会被触发。 - 需要通过资源编辑器或代码显式设置控件的 Owner Draw 属性。 - 对于 `CListCtrl`,可以通过调用 `SetStyle` 方法添加 `LVS_OWNERDRAWFIXED` 或 `LVS_OWNERDRAWVARIABLE` 样式[^4]。 ```cpp m_ListCtrl.SetExtendedStyle(m_ListCtrl.GetExtendedStyle() | LVS_OWNERDRAWFIXED); ``` #### 2. **消息映射未正确配置** 确保在类的消息映射中已添加 `ON_WM_MEASUREITEM_REFLECT()` 宏。如果有此宏,框架无法将 `WM_MEASUREITEM_REFLECT` 消息路由到对应的处理函数。 ```cpp BEGIN_MESSAGE_MAP(CMyListCtrl, CListCtrl) ON_WM_MEASUREITEM_REFLECT() END_MESSAGE_MAP() ``` 同,在类声明中实现相应的处理函数: ```cpp afx_msg void MeasureItem(LPMEASUREITEMSTRUCT lpMeasureItemStruct); ``` #### 3. **PreSubclassWindow 的初始化问题** 某些情况下,默认的子类化过程可能导致消息反射失败。可以在 `PreSubclassWindow` 中手动发送 `WM_WINDOWPOSCHANGED` 消息以强制刷新布局。 ```cpp void CMyListCtrl::PreSubclassWindow() { CRect rcWin; GetWindowRect(&rcWin); WINDOWPOS wp; wp.hwnd = m_hWnd; wp.cx = rcWin.Width(); wp.cy = rcWin.Height(); wp.flags = SWP_NOACTIVATE | SWP_NOMOVE | SWP_NOOWNERZORDER | SWP_NOZORDER; SendMessage(WM_WINDOWPOSCHANGED, 0, (LPARAM)&wp); CListCtrl::PreSubclassWindow(); } ``` #### 4. **MeasureItem 处理逻辑错误** 即使消息成功到达,但如果处理函数中的逻辑有误也可能导致行为异常。以下是一个典型的 `MeasureItem` 实现示例: ```cpp void CMyListCtrl::MeasureItem(LPMEASUREITEMSTRUCT lpMeasureItemStruct) { // 设置每一项的高度 lpMeasureItemStruct->itemHeight = 30; } ``` 注意:对于具有 `LBS_OWNERDRAWFIXED` 样式的控件,系统只会发送一次 `WM_MEASUREITEM` 消息[^5]。因此,一旦设置了固定的项高,后续更改不会再次触发该消息。 --- ### 示例代码 以下是一个完整的例子,展示如何正确配置处理 `ON_WM_MEASUREITEM_REFLECT`: ```cpp // 类声明部分 class CMyListCtrl : public CListCtrl { public: DECLARE_DYNAMIC(CMyListCtrl) protected: afx_msg void PreSubclassWindow(); afx_msg void MeasureItem(LPMEASUREITEMSTRUCT lpMeasureItemStruct); DECLARE_MESSAGE_MAP() }; // 消息映射 IMPLEMENT_DYNCREATE(CMyListCtrl, CListCtrl) BEGIN_MESSAGE_MAP(CMyListCtrl, CListCtrl) ON_WM_MEASUREITEM_REFLECT() END_MESSAGE_MAP() // 初始化窗口 void CMyListCtrl::PreSubclassWindow() { CRect rcWin; GetWindowRect(&rcWin); WINDOWPOS wp; wp.hwnd = m_hWnd; wp.cx = rcWin.Width(); wp.cy = rcWin.Height(); wp.flags = SWP_NOACTIVATE | SWP_NOMOVE | SWP_NOOWNERZORDER | SWP_NOZORDER; SendMessage(WM_WINDOWPOSCHANGED, 0, (LPARAM)&wp); CListCtrl::PreSubclassWindow(); } // 测量子项大小 void CMyListCtrl::MeasureItem(LPMEASUREITEMSTRUCT lpMeasureItemStruct) { lpMeasureItemStruct->itemHeight = 30; // 设置每项高度 } ``` --- ### 总结 以上方法涵盖了常见的 `ON_WM_MEASUREITEM_REFLECT` 无响应原因及解决策略。具体实施应根据实际需求调整控件样式测量逻辑。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值