排序 CMFCListCtrl 时的奇怪 (?) 行为
Strange (?) behaviour while sorting CMFCListCtrl
我有一个支持排序的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 不再正确,因为它们代表行号。
然后我处理了 header 点击通知,它发生在 之后 排序:
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) 点之前和之后拿起文本来检查我是否更新了正确的
行,一切似乎都很好:我先得到旧参数,然后得到新参数。
现在怎么办:我在 header 行中再次单击以降序排序,我得到了什么?一切都乱七八糟。
我决定检查一些东西并添加了以下代码:
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 消息。
问题已解决:在
// bla bla about item data
我更新成员数据的地方:
SetItemData(int, DWORD_PTR);
根据https://msdn.microsoft.com/it-it/library/936147y4.aspx ItemData实际上是LVITEM结构中指定的LPARAM值。
我有一个支持排序的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 不再正确,因为它们代表行号。 然后我处理了 header 点击通知,它发生在 之后 排序:
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) 点之前和之后拿起文本来检查我是否更新了正确的 行,一切似乎都很好:我先得到旧参数,然后得到新参数。
现在怎么办:我在 header 行中再次单击以降序排序,我得到了什么?一切都乱七八糟。 我决定检查一些东西并添加了以下代码:
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 消息。
问题已解决:在
// bla bla about item data
我更新成员数据的地方:
SetItemData(int, DWORD_PTR);
根据https://msdn.microsoft.com/it-it/library/936147y4.aspx ItemData实际上是LVITEM结构中指定的LPARAM值。