c++ - 排序 CMFCListCtrl 时的奇怪(?)行为

标签 c++ visual-studio-2010 mfc mfc-feature-pack

我有一个支持排序的CMFCListCtrl的子类,阅读我必须实现的文档 虚方法:

virtual int OnCompareItems(LPARAM lParam1, LPARAM lParam2, int nColumn);

其中 lParam1 和 lParam2 是某种类型的参数,在创建列表控件时指定。

因为我需要按字母顺序排序,所以我以这种方式插入了我的行:

item.mask = LVIF_COLFMT | LVIF_IMAGE | LVIF_PARAM | LVIF_TEXT;
for (nI = 0; nI ....)
{
    // UMASK, ITEM IDX, TEXT, STATE 1, STATE 2, IMAGE IDX, LPARAM
    m_lstMateriali.InsertItem(item.mask, nI, _T("   "), 0, 0, SelezionaIcona(itIM->m_bSelez, itIM->m_eStatus), nI);
    strQnt.Format(_T("%d"), itIM->m_nIdBolla);
    m_lstMateriali.SetItemText(nI, 1, strQnt);
    m_lstMateriali.SetItemText(nI, 2, itIM->m_strCodMater);
    m_lstMateriali.SetItemText(nI, 3, itIM->m_strDescrMater);
    m_lstMateriali.SetItemText(nI, 4, itIM->m_strColore);
    strQnt.Format(_T("%d"), itIM->m_nDaTag);
    m_lstMateriali.SetItemText(nI, 5, strQnt);
    m_lstMateriali.SetItemData(nI, nI);
}

此时,我的Subclass排序方法:

virtual int OnCompareItems(LPARAM lParam1, LPARAM lParam2, int nColumn)
{
    int     nCol_1;             ///< 
    int     nCol_2;             ///< 
    CString strCol_1;           ///< 
    CString strCol_2;           ///< 

    strCol_1 = GetItemText(lParam1, nColumn);
    strCol_2 = GetItemText(lParam2, nColumn);
    switch (nColumn)
    {
    case 2:
        return strCol_1.Compare(strCol_2);
    }
}

效果很好。但是现在,由于排序,我的 LPARAM 不再正确,因为它们代表行号。 然后我处理了标题点击通知,它发生在排序之后:

void CEliCUTK2SceltaMaterialiDlg::OnHdnItemclickLstSmListapezzi(NMHDR * pNMHDR, LRESULT * pResult)
{
    int         nNumRig;
    CString     strCodMat;
    LPNMHEADER  phdr = reinterpret_cast<LPNMHEADER>(pNMHDR);
    LVITEM      item;

    nNumRig = m_lstMateriali.GetItemCount();
    memset(&item, 0, sizeof(LVITEM));
    item.mask = LVIF_PARAM;
    for (int nI = 0; nI < nNumRig; nI++)
    {
        item.iItem = nI;
        item.lParam = nI;
        m_lstMateriali.SetItem(&item);  // (1)
        strCodMat = m_lstMateriali.GetItemText(nI, 2);
        // bla bla about item data
    }
    *pResult = 0;
}

这可能是最糟糕的做法,但目前效果还不错:我根据 新的行号。我还在第 (1) 点之前和之后拿起文本来检查我是否更新了正确的 行,一切似乎都很好:我先得到旧参数,然后得到新参数。

现在怎么办:我在行标题中再次单击以降序排序,我得到了什么?一切都乱七八糟。 我决定检查一些东西并添加了以下代码:

void CEliCUTK2SceltaMaterialiDlg::OnLvnItemchangedLstSmListapezzi(NMHDR * pNMHDR, LRESULT * pResult)
{
    int             nSot, nTip;
    CString         strID;
    DWORD_PTR       dwID;
    LPNMLISTVIEW    pNMLV = reinterpret_cast<LPNMLISTVIEW>(pNMHDR);
    LVITEM          item;

    if (pNMLV->uNewState == 3)
    {
        dwID = m_lstMateriali.GetItemData(pNMLV->iItem);
        memset(&item, 0, sizeof(LVITEM));
        item.mask = LVIF_PARAM;
        item.iItem = pNMLV->iItem;
        m_lstMateriali.GetItem(&item);
        strID.Format(_T("PARAM: %ld."), item.lParam);
        AfxMessageBox(strID);
    }
    *pResult = 0;
}

重新启动程序,再次对列表进行排序并单击第一行(再次检查新的 LPARAM 是否正确后) 我收到消息:

PARAM: 10.

虽然我应该得到:

PARAM: 0.

10 是旧位置。我在某处遗漏了什么吗?

最佳答案

一般来说,您想要在 lParam 中存储的是一些可以让您返回到用于填充列表的数据的东西,或者是指向包含要比较的数据的结构的指针。如果您有并行 CList 或 std::list 集合(或 CArray 或 std::vector),并且在 CListCtrl 的生命周期内不会更改其顺序,则放入索引将起作用。

排序后或插入期间,列表中的第 10 项在您的示例中的 lParam 可能为 2。您希望使用“2”索引原始数据源(CList、CArray、记录集或其他...),然后获取数据并在原始数据上进行比较。因为,当你排序时,lParam 中的索引将不会与 CListCtrl(或 CMFCListCtrl)中的索引相同。

您可以为比较所需的所有数据分配一个结构,并将其存储在 lParam 成员中。 (在某些 MFC 类中,您使用 SetItemDataPtr)。要清理数据,请处理 LVN_DELETEITEM 消息。

关于c++ - 排序 CMFCListCtrl 时的奇怪(?)行为,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41065407/

相关文章:

c++ - 奇怪的 GCC 编译错误(包括简单示例)

C++ 排序和跟踪索引

c++ - 在 C++ 中使用和不使用调试信息设计函数的正确方法

QT、wxwidget、MFC

c++ - CString 对象总是返回指针值

c++ - Ribbon UI动态按钮菜单编辑

c++ - 我从 g++ 编译器收到此错误 - 从 ‘int*’ 到 ‘int’ 的无效转换 [-fpermissive]

c++ - 重新分配 C 样式字符串会导致内存泄漏吗?

visual-studio-2010 - 如何在 UML 关系的 MemberEnds 上应用构造型?

visual-studio-2010 - Visual Studio 应用程序在 Windows 7 上停止工作