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; }
阿文 http://tulipstudio.yeah.net

|