MFC CListCtrl吃鼠标事件?

时间:2011-10-12 21:05:55

标签: c++ user-interface mfc mouseevent clistctrl

除非双击,否则CListCtrl似乎不会发送鼠标事件。

我尝试从鼠标向下处理程序发送丢失的消息以进行补偿,但这会导致其他不良行为。然后我想我可以通过检查状态在鼠标移动处理程序中发送消息更准确一些。然而,这些都是可怕的黑客,除了丑陋之外,它们可能无法适用于衍生控件的每个可能的实现。

如果有人知道为什么没有收到鼠标事件,我会很好奇。更重要的是我如何使用CListCtrl样式获取LVS_OWNERDATA以便像其他控件一样发送鼠标消息?

编辑:我知道LVN_BEGINDRAGLVN_BEGINRDRAG等,但为了使用这些,我需要阻止WM_LBUTTONDOWNWM_RBUTTONDOWN和{{1} }从转到父窗口或WM_MOUSEMOVE挂钩到DragDropManager / CWinAppEx所以我可以为这个控件做一个特殊的一次性案例来处理现有系统。

这是因为我有一个中央拖放管理器,它可以通知各种类型的控件何时开始拖动操作,何时结束,取消,更改动画,在自定义消息中传递源和目标的显示对象等。它需要足够灵活,根据控件,输入,所选项目或目标项目的类型,不同的控制类型(包括3D),甚至不同的应用程序等,具有不同的启动方式和不同的操作。

4 个答案:

答案 0 :(得分:2)

这里的参考是我的工作,但这是一个可耻的黑客。如果没有人能想出比这更好的东西,那真的很难过。

部首:

#pragma once

// CListCtrlEx
class CListCtrlEx : public CListCtrl
{
    DECLARE_DYNAMIC(CListCtrlEx)

public:
    CListCtrlEx();
    virtual ~CListCtrlEx();

    bool IsSelected(int index);
    BOOL SelectDropTarget(int item);

protected:
    DECLARE_MESSAGE_MAP()

    afx_msg void OnStateChanged(NMHDR* pNMHDR, LRESULT* pResult);

    afx_msg void OnLButtonDown(UINT nFlags, CPoint point);
    afx_msg void OnRButtonDown(UINT nFlags, CPoint point);
    afx_msg void OnLButtonUp(UINT nFlags, CPoint point);
    afx_msg void OnRButtonUp(UINT nFlags, CPoint point);
    afx_msg void OnMouseMove(UINT nFlags, CPoint point);

private:
    bool m_lbDown;
    bool m_rbDown;
};

实现:

#include "stdafx.h"
#include "ListCtrlEx.h"

// CListCtrlEx
IMPLEMENT_DYNAMIC(CListCtrlEx, CListCtrl)

CListCtrlEx::CListCtrlEx() : m_lbDown(false), m_rbDown(false)
{
}

CListCtrlEx::~CListCtrlEx()
{
}

BEGIN_MESSAGE_MAP(CListCtrlEx, CListCtrl)
    ON_WM_LBUTTONDOWN()
    ON_WM_RBUTTONDOWN()
    ON_WM_LBUTTONUP()
    ON_WM_RBUTTONUP()
    ON_WM_MOUSEMOVE()

    ON_NOTIFY_REFLECT(LVN_ODSTATECHANGED, &CListCtrlEx::OnStateChanged)
END_MESSAGE_MAP()

// CListCtrlEx message handlers
void CListCtrlEx::OnLButtonDown(UINT nFlags, CPoint point)
{
    m_lbDown = true;
    CListCtrl::OnLButtonDown(nFlags, point);
}

void CListCtrlEx::OnRButtonDown(UINT nFlags, CPoint point)
{
    m_rbDown = true;
    CListCtrl::OnRButtonDown(nFlags, point);
}

void CListCtrlEx::OnLButtonUp(UINT nFlags, CPoint point)
{
    m_lbDown = false;
    CListCtrl::OnLButtonUp(nFlags, point);
}

void CListCtrlEx::OnRButtonUp(UINT nFlags, CPoint point)
{
    m_rbDown = false;
    CListCtrl::OnRButtonUp(nFlags, point);
}

void CListCtrlEx::OnMouseMove(UINT nFlags, CPoint point)
{
    if (m_lbDown && ((nFlags & MK_LBUTTON) == 0))
    {
        PostMessage(WM_LBUTTONUP,
            MAKEWPARAM(LOWORD(nFlags), HIWORD(nFlags)),
            MAKELPARAM(point.x, point.y));
    }

    if (m_rbDown && ((nFlags & MK_RBUTTON) == 0))
    {
        PostMessage(WM_RBUTTONUP,
            MAKEWPARAM(LOWORD(nFlags), HIWORD(nFlags)),
            MAKELPARAM(point.x, point.y));
    }

    CListCtrl::OnMouseMove(nFlags, point);
}

bool CListCtrlEx::IsSelected(int index)
{
    return (GetItemState(index, LVIS_SELECTED) & LVIS_SELECTED) != 0;
}

// highlight drop targets sort of like CTreeCtrl
BOOL CListCtrlEx::SelectDropTarget(int item)
{
    static int prevHighlight(-1);
    if (item >= 0 && item < GetItemCount())
    {
        if (item != prevHighlight)
        {
            if (prevHighlight >= 0)
            {
                SetItemState(prevHighlight, 0, LVIS_DROPHILITED); // remove highlight from previous target
                RedrawItems(prevHighlight, prevHighlight);
            }

            prevHighlight = item;
            SetItemState(item, LVIS_DROPHILITED, LVIS_DROPHILITED); // highlight target
            RedrawItems(item, item);

            UpdateWindow();
            return TRUE;
        }
    }
    else
    {
        for (int i(0); i < GetItemCount(); ++i)
            SetItemState(i, 0, LVIS_DROPHILITED); // un-highlight all
        prevHighlight = -1;
    }

    return FALSE;
}

void CListCtrlEx::OnStateChanged(NMHDR* pNMHDR, LRESULT* pResult)
{
// MSDN:
// If a list-view control has the LVS_OWNERDATA style,
// and the user selects a range of items by holding down the SHIFT key and clicking the mouse,
// LVN_ITEMCHANGED notification codes are not sent for each selected or deselected item.
// Instead, you will receive a single LVN_ODSTATECHANGED notification code,
// indicating that a range of items has changed state.

    NMLVODSTATECHANGE* pStateChanged = (NMLVODSTATECHANGE*)pNMHDR;

    // redraw newly selected items
    if (pStateChanged->uNewState == LVIS_SELECTED)
        RedrawItems(pStateChanged->iFrom, pStateChanged->iTo);
}

答案 1 :(得分:0)

您不需要这些活动。控件为您提供所需的一切。

从你的代码中我收集到你想在列表控件中实现项目的拖放。 而不是将无法正常工作的东西和badmouth MS混在一起,而是以正确的方式做到:

处理LVN_BEGINDRAG通知以开始拖动操作。

答案 2 :(得分:0)

您能否通过以下链接告诉我们是否有效。

http://support.microsoft.com/kb/147842

你可以从那里得到一个想法。

答案 3 :(得分:-1)

斯特凡的回答可能是最好的服务....

但是,你可以挂钩CListCtrl的winproc(地狱,你可以在MFC中继承CListCtrl,提供你自己的虚拟WindowProc()并转发/拦截你想要的任何鼠标消息。

只需在运行时使用标准MFC子类控件机制,即可在您希望的任何对话框或窗口上“替换”您的类的标准CListCtrl。

我假设你知道怎么做?