CListCtrl排序设计

CListCtrl排序设计

随着开发工具不断地更新换代,Microsoft的 .Net 框架到目前已经更新到了2.0。不得不承认,.Net 框架给程序员带来了编程上的很多便利,从框架的角度,使开发者开发出来的产品就有了一个高的起点,这是我对.Net框架众多肯定中的一条。关于.Net框架的评价,不是我在这里讨论的内容,我想提的是似乎被遗忘的Microsoft的另外一个框架MFC(Microsoft Foundation Class Library)。


在开发过程中,我用惯了.Net框架,里面提供的各种控件都很强大,特别是它的类型转换Convert类,现在看来简直是一个完美的设计。但是回过头来,看看MFC提供的控件,几乎让人一筹莫展。
是的,我们平时写的程序离程序本来的面目确实是越来越远了,而MFC框架只是将这个距离稍微拉近了一些。
于是,我就开始在MFC框架上写程序,问题很快就发生了。


我有一部分数据想通过列表的形式体现出来,我第一个想到的控件就是MFC提供的CListCtrl类。在使用过程中,我才发现,用CListCtrl类加载数据的方式远比我想象的要复杂得多,但是凭着我多年的开发经验和MSDN上的有限帮助文档,我究竟还是硬把数据塞进了CListCtrl控件,让它把数据显示出来。但是当我在欣赏我的劳动成果时,我发现那些数据的排列是无序的。


按照平常的经验,CListCtrl控件的属性页里面肯定有sort之类的属性设置,所以我就打开了属性页,一点不假,还真的有。正当我高兴的时候,我又觉得这事情有点不对,我的这个列表里面有3个字段,这个sort属性怎么“可能”知道我要对哪个列排序呢?还是跑到MSDN去看看,CListCtrl是提供了一个排序的方法SortItems,仔细研究了后,发现它也只是针对InsertItem写入的数据排序,而对于SetItemText设置的subItem更本就不具备排序功能,如果要对其它列排序,就必须另外想办法了。

先解决往CListCtrl中写入数据的问题:
/*list为CTestDlg中的成员变量; 已经在别处声明并成功初始化,并在属性页将该控件的View属性调整为Report*/
BOOL CTestDlg::OnInitDialog(){
    CDialog::OnInitDialog();
    .........
    list.InsertColumn(0,"主项",LVCFMT_LEFT, 80);
    list.InsertColumn(1,"子项", LVCFMT_LEFT, 50);
    list.InsertItem(0,"主数据");
    list.SetItemText(0,1,"子数据");

    .........
}
如图:

 














 然后解决排序问题。在我的另一个程序中,需要通过CListCtrl控件显示日期数据,希望更具日期的升序显示,如图:


要将排序功能加入到CListCtrl控件,首先必须从CListCtrl继承,然后将排序方法加入到继承的类中。这里我想到了两种写法:

  1. 创建一个虚拟放方法
    在".h"文件中:
    class CSortableListCtrl:CListCtrl{
          virtual void SortAfterInsert(void);
    }
  2. 创建一个方法,该方法的参数为进行排序的函数地址
    在".h"文件中:
    typedef void(CALLBACK *SORTLISTPROC)(LPARAM dwCtrl);//声明函数指针的类型

    在类中,声明成员方法:
    class CSortableListCtrl:CListCtrl{
          void SortAfterInsert(SORTLISTPROC lpSort, LPARAM dwCtrl);
    }

比较这两种方法后,得出一下结论:
采用第一种方法,要对列表中的数据进行排序,就必须从CSortableListCtrl类继承,然后再使用它;使用第二种方法,要对列表中的数据进行排序,就可以直接使用CSortableListCtrl来声明,然后定义SORTLISTPROC的执行函数来进行排序。
很显然,对列表中的数据进行排序,实际上就是对排序方法的从载,所以使用第二种方法更直接。

现在已经定了用哪构成方法来处理排序,那么下一步便是如何排序。由于我们采用的是第二种方法,那么直接写排序的执行函数。
void CALLBACK SortListCallBack(LPARAM dwCtrl)
{

 CSortableListCtrl* list = (CSortableListCtrl*) dwCtrl;

 ITEM *lItem;
 lItem = new ITEM;
 COleDateTime tmFirst, tmItem;

 //获取第一项数据
 strcpy(lItem->lpszItem, (LPCTSTR)list->GetItemText(0,0));//主键
 strcpy(lItem->lpszSItem1 , (LPCTSTR)list->GetItemText(0,1));//修改时间
 strcpy(lItem->lpszSItem2, (LPCTSTR)list->GetItemText(0,2));//读取时间

 //第一个数据的修改时间
 tmFirst.ParseDateTime(lItem->lpszSItem1);
     
 int items;//指针位置,当前的数据
 items = list->GetItemCount();
 //比较数据


 for (int i = 1; i< items; i++)
 {
  

  //以时间排序
  tmItem.ParseDateTime((LPCTSTR)list->GetItemText(i, 1));
 
  if (tmFirst > tmItem)
  {
   //插入时间比读取的时间大
   //将第一项的数据插入到i - 1这个位置
   list->InsertItem( i , lItem->lpszItem);
   list->SetItemText(i ,1,lItem->lpszSItem1);
   list->SetItemText(i ,2,lItem->lpszSItem2);
   //将第一项删除
   list->DeleteItem(0);
   delete lItem;
   return;
  }

 }
  //当程序执行到这个位置时,说明插入的值为最小值
  list->InsertItem(items, lItem->lpszItem);
  list->SetItemText(items, 1, lItem->lpszSItem1);
  list->SetItemText(items, 2, lItem->lpszSItem2);

  list->DeleteItem(0);
             
  delete lItem;


}

在这个排序中,其中将两个值进行比较的方法还有待改进,希望朋友们多提意见。



.h文件

#pragma once


// CSortableListCtrl

typedef void(CALLBACK* SORTLISTPROC)(LPARAM dwCtl);

class CSortableListCtrl : public CListCtrl
{
 DECLARE_DYNAMIC(CSortableListCtrl)

public:
 CSortableListCtrl();
 virtual ~CSortableListCtrl();

protected:
 DECLARE_MESSAGE_MAP()

public:
 //在插入数据到列表后立即排序,如果用此方法,在插入数据时必须将该数据插入到第一行
 void SortAfterInsert(SORTLISTPROC lpSort, LPARAM dwCtrl);

};

 

.Cpp文件

// SortableListCtrl.cpp : 实现文件
//

#include "stdafx.h"
#include "KnAssistant.h"
#include "SortableListCtrl.h"
#include "./sortablelistctrl.h"


// CSortableListCtrl

IMPLEMENT_DYNAMIC(CSortableListCtrl, CListCtrl)
CSortableListCtrl::CSortableListCtrl()
{
}

CSortableListCtrl::~CSortableListCtrl()
{
}


BEGIN_MESSAGE_MAP(CSortableListCtrl, CWnd)
END_MESSAGE_MAP()

//在插入数据到列表后立即排序
void CSortableListCtrl::SortAfterInsert(SORTLISTPROC lpSort, LPARAM dwCtrl)
{
 lpSort(dwCtrl);
}

 

执行函数


//对ListCtrl进行排序的回调函数
//始终将第一个数据进行排序,在
每个执行函数针对不同的排序方式,在这里,只针对第二列(日期)进行降序排列
void CALLBACK SortListCallBack(LPARAM dwCtrl)
{
 CSortableListCtrl* list = (CSortableListCtrl*) dwCtrl;

 ITEM *lItem;
 lItem = new ITEM;
 COleDateTime tmFirst, tmItem;

 //获取第一项数据
 strcpy(lItem->lpszItem, (LPCTSTR)list->GetItemText(0,0));//主键
 strcpy(lItem->lpszSItem1 , (LPCTSTR)list->GetItemText(0,1));//修改时间
 strcpy(lItem->lpszSItem2, (LPCTSTR)list->GetItemText(0,2));//读取时间

 //修改时间
 tmFirst.ParseDateTime(lItem->lpszSItem1);
     
 int items;//指针位置,当前的数据
 items = list->GetItemCount();
 //比较数据


 for (int i = 1; i< items; i++)
 {
  

  //以时间排序
  tmItem.ParseDateTime((LPCTSTR)list->GetItemText(i, 1));
 
  if (tmFirst > tmItem)
  {
   //插入时间比读取的时间大
   //将第一项的数据插入到i - 1这个位置
   list->InsertItem( i , lItem->lpszItem);
   list->SetItemText(i ,1,lItem->lpszSItem1);
   list->SetItemText(i ,2,lItem->lpszSItem2);
   //将第一项删除
   list->DeleteItem(0);
   delete lItem;
   return;
  }

 }
  //当程序执行到这个位置时,说明插入的值为最小值
  list->InsertItem(items, lItem->lpszItem);
  list->SetItemText(items, 1, lItem->lpszSItem1);
  list->SetItemText(items, 2, lItem->lpszSItem2);

  list->DeleteItem(0);
             
  delete lItem;


}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值