我的程序中有一个列表,希望像资源管理器那样,能够一按列表头,就能对这一列进行排序,我的做法是:在消息响应函数中调用SortItems((PFNLVCOMPARE)CompareFunc,(DWORD)m_nSortedCol);它在调用回调函数CompareFunc,能对数据进行排序,但是在
列表框里没有显示排序后的数据,为什么?是不是要自己编代码实现,怎么实现,哪位大虾给点提示吧?多谢了!
列表框里没有显示排序后的数据,为什么?是不是要自己编代码实现,怎么实现,哪位大虾给点提示吧?多谢了!
解决方案 »
- [求助]Socket::Accept(CSocket &sckAccept)函数都做了些什么,需要对sckAccept进行Close()么?
- BMP图片颜色过滤的问题?
- 如何将CString转化为int数组
- settimer OnTimer 在非MFC环境 比如在WIN32 application 下怎么使用
- 请教-如何编程刻录DVD
- MFC源代码的问题
- 请问:对话框如何获取自己的窗口句柄?
- 将数据库的内容读取到编辑框中??
- char tchar的关系
- 我刚接触VC不久,对一些简单问题还不明白
- 让PlaySound或者sndPlaySound输出声音到指定设备
- [初学网络编程]知道一个IP,和PORT,怎样发一个UDP包?
在一个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