CListView类中包含了一个CListCtrl类的引用,并由这个CListCtrl类实现了List View窗口,窗口中包含了由程序添加的List Item。List Item则以LVITEM结构形式存在。
在一个ListView窗口中,当所有的Item以Report的形式出现时,应该允许用户单击列标题,然后以该列的顺序进行升序排序或反序排序List Item。
当用户单击列标题时,我们可在该动作的响应函数中调用CListCtrl类的SortItems函数来实现排序。该函数的原型是:BOOL SortItems( PFNLVCOMPARE pfnCompare, DWORD dwData ),其中pfnCompare是一个用户自定义的用来进行比较的函数,dwData是程序准备向pfnCompare传递的附加的信息。
PFNLVCOMPARE的原型定义是这样的:
int CALLBACK CompareFunc(LPARAM lParam1, LPARAM lParam2,
LPARAM lParamSort);
lParam1与lParam2参数分别是两个Item所包含的LVITEM结构中的lParam成员变量,lParamSort参数则是上面SorItems函数传递过来的dwData数据,一般用它表示是按增序还是降序来进行排序。CompareFunc函数一般返回-1,0,1,表示此次比较的结果。
SortItems函数通过循环调用CompareFunc函数来实现对List Item的排序,并根据CompareFunc函数的返回值决定每两个Item的相对位置。
一般来说,每条List Item只能指定一个lParam数据,当我们要实现对不同的列进行排序时,则需要在用户单击列标题后,根据不同的列对每个List Item的lParam数据进行更新,使其反应每条List item相应Sub Item的内容,并提供给比较函数,使CListCtrl在排序时根据该列的内容进行排序。
为此,我们可以在添加List Item时为List Item的lParam变量指定一个字符串指针保存当前排序列的文本信息,并在需要对lParam的数据进行更新时,指定不同的字符器指针来返回排序列的信息即可。
下面举例说明上述操作。
// 在类说明中必须将下面这个ListViewCompareFunc定义成static型
int CALLBACK CMyListView::ListViewCompareFunc(LPARAM lParam1, LPARAM lParam2, LPARAM lParamSort)
{
// 得到排序方式
int * pisortorder = (int *)lParamSort;
// 得到两个列的排序信息
TCHAR * sz1 = (TCHAR *)lParam1;
TCHAR * sz2 = (TCHAR *)lParam2;
// 比较列的信息并返回比较结果。
// 若为减序,则将比较结果乘上-1。
if (* pisortorder == LVS_SORTASCENDING)
return lstrcmp(sz1, sz2);
else
return lstrcmp(sz1, sz2) * (-1);
}
void CMyListView::OnColumnclick(NMHDR* pNMHDR, LRESULT* pResult)
{
static int ncurSortCol = -1; // 保存当前的排序列。
// 一开始表示为-1,表示尚未按任何列排序。
NM_LISTVIEW* pNMListView = (NM_LISTVIEW*)pNMHDR;
CListCtrl* lc = &GetListCtrl();
LONG ws = GetWindowLong(lc->m_hWnd, GWL_STYLE);
int nSortOrder; // 排序的方式
// 若点击列与当前排序列不同的列,则改变排序序,并将排序方式改为增序。
// 若当前排序列与点击列相同,则更改增、减序的排序方式
if (ncurSortCol == pNMListView->iSubItem)
{
if (ws & LVS_SORTASCENDING)
{
ws ^= LVS_SORTASCENDING;
nSortOrder = LVS_SORTDESCENDING;
}
else
{
ws ^= LVS_ SORTDESCENDING;
nSortOrder = LVS_SORTASCENDING;
}
}
else
{
if (ws & LVS_SORTASCENDING)
ncurSortCol = pNMListView->iSubItem;
if (ws & LVS_SORTASCENDING)
nSortOrder = LVS_SORTASCENDING;
else
nSortOrder = LVS_ SORTDESCENDING;
}
// 将当前的排序信息保存在窗口Style中,供以后使用
ws |= nSortOrder;
SetWindowLong(lc->m_hWnd, GWL_STYLE, ws);
// 将各ITEM的LPARAM用新排序列的内容替换
LVITEM li;
li.mask = LVIF_PARAM|LVIF_TEXT;
TCHAR szItemText[1024];
for (int i = 0; i < lc.GetItemCount(); i++)
{
li.iItem = i;
li.iSubItem = ncurSortCol;
li.cchTextMax = 1024;
li.pszText = szItemText;
lc->GetItem(&li);
TCHAR * szlparam = (TCHAR *)li.lParam;
if (szlparam != NULL)
// 删除以前的信息,释放空间
// 添加List Item时应注意将lParam初始化NULL
delete szlparam;
// 复制当前列的szItemText到Item的lParam中
szlparam = new TCHAR[lstrlen(szItemText) + 1];
lstrcpy(szlparam, szItemText);
lc.SetItemData(i, szlparam);
}
// 开始排序
GetListCtrl().SortItems(ListViewCompareFunc,(LPARAM) (&nSortOrder));
*pResult = 0;
}
阿文