MFC控件--CListCtrl

本文详细介绍了CListCtrl的基本使用方法,包括数据写入、报告样式配置及图标集成,并提供了对CListCtrl中数据进行排序的实现案例。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

优快云 - 文档中心 - Visual C++  
  
标题  CListCtrl使用详解     选择自 fairness 的 Blog
关键字  CListCtrl使用详解
出处 

 

 

1。先来介绍REPORT类型的CListCtrl:
首先使用下面的语句设置CListCtrl的style:
 DWORD SetExtendedStyle( DWORD dwNewStyle );
其中
 LVS_EX_CHECKBOXES 表示添加CheckBox
 LVS_EX_FULLROWSELECT 表示选择整行
 LVS_EX_GRIDLINES 表示添加表格线

如果设置了LVS_EX_CHECKBOXES属性,则可以用
 BOOL GetCheck( int nItem ) const;
来得到某一行是否Checked。

可以先用下面的语句来删除以前的东西:
 for(int k=2;k>=0;k--) //注意要从后往前删,否则出错
  m_ListCtrl.DeleteColumn(k);
 m_ListCtrl.DeleteAllItems();

用下面的语句新建列:
 m_ListCtrl.InsertColumn(0,_T("文件名"),LVCFMT_IMAGE|LVCFMT_LEFT);
 m_ListCtrl.InsertColumn(1,_T("仪器类别"));
 m_ListCtrl.InsertColumn(2,_T("项目类别"));
 
其中LVCFMT_IMAGE表示可以在第一列加入图标。如果不要图标可以删去。

然后设置列宽:
 for(j=0;j<3;j++)
  m_ListCtrl.SetColumnWidth(j ,100);
 
以下为列表加入图标,如果不需要图标,可以跳过这一步。注意只在第一次加入,如果多次加入会出错!
先在头文件中加入声明:
 CImageList m_ImageList;
这是必要的,如果在cpp的某个函数中加入由于生命期结束,CImageList自动释放,则效果是列表中看不到图标,只看到一个白方块。
下面生成CImageList,并将其绑定到CListCtrl中,这是CImageList中还没有图标,只是一个容器:
 static int flag=2;
 if(flag==2){//只调用一次SetImageList,否则出错
  m_ImageList.Create(128, 128, ILC_COLORDDB|ILC_MASK, 20, 1); 
  m_ListCtrl.SetImageList(&m_ImageList,LVSIL_SMALL);
 }
 flag=(flag+1)%2;
如果CListCtrl已经用过,曾经加过图标进去,这时就要删除上次放进m_ImageList中的Image
 for(int kk=0;kk<m_ImageList.GetImageCount();kk++)
  m_ImageList.Remove(k);
 
下面介绍如何向CListCtrl里面加入行,并同时为每一行动态加入图标:
假设m_listRowCount为要加入的行数。
 CBitmap* bitmap;
 bitmap=new CBitmap[m_list1rowCount];
 HBITMAP hbitmap; 
 
 for(int i = 0; i < m_listRowCount; i++)
 {
  //为每一行插入相应的缩略图
  CFile f;
  CFileException e;  
  if( !f.Open(m_FileName, CFile::modeRead, &e )){ //m_FileName为bmp文件名,由你来定
   hbitmap = (HBITMAP)LoadImage(NULL,path+"blank.bmp",IMAGE_BITMAP,0,0,
    LR_CREATEDIBSECTION|LR_DEFAULTSIZE|LR_LOADFROMFILE);
  }else{
   f.Close();
   hbitmap = (HBITMAP)LoadImage(NULL,bmpFile,IMAGE_BITMAP,0,0,
    LR_CREATEDIBSECTION|LR_DEFAULTSIZE|LR_LOADFROMFILE);
  }
  bitmap[i].Attach(hbitmap);
  m_ImageList.Add(&bitmap[i], RGB(0, 128, 128));
  
  //插入行
  m_ListCtrl.InsertItem(i,m_FileName,i);
  m_ListCtrl.SetItemText(i,1,type);
  m_ListCtrl.SetItemText(i,2,m_Path);
 }
  
 //记得删除已经没用的临时文件
 if(m_list1rowCount!=0)
  delete[] bitmap;

2。如果是ICON类型的CListCtrl,则要做一点点改动:
把绑定图标集的代码由
 SetImageList(&m_ImageList,LVSIL_SMALL);
改为
 SetImageList(&m_ImageList,LVSIL_NORMAL);

插入行时只用
 InsertItem(i,mainSet.m_FileName,i);
不用
 SetItemText(i,1,type);
之类的代码。

 -----------------------------------------------

排序实现

CListCtrl

 

随着开发工具不断地更新换代,Microsoft的 .Net 框架到目前已经更新到了2.0。不得不承认,.Net 框架给程序员带来了很多编程上的便利。开发者通过.Net框架,使他开发出来的产品自然的从框架那里获得了稳定、安全的高起点。这是.Net框架的优秀表现之一。关于.Net框架的评价,不是我在这里讨论的内容,我想提的是似乎被遗忘的Microsoft的另外一个框架MFC(Microsoft Foundation Class Library)。

我写这篇文章的目的,是为了向大家说说我使用CListCtrl的心得,主要包括两个方面:

  1. 将数据写入到CListCtrl;
  2. 对CListCtrl中的数据进行排序;

将数据写入到CListCtrl

向CListCtrl中写入数据,一般使用3个成员方法:
CListCtrl::InsertColumn;
CListCtrl::InsertItem;
CListCtrl::SetItemText;


InsertColumn被用于向显示列表中插入显示的列,例如:
    list.InsertColumn(0,"主项",LVCFMT_LEFT, 80);
    list.InsertColumn(1,"子项", LVCFMT_LEFT, 50);
以上代码的目的是向列表中插入两个显示列,第一列标题为"主项",对齐方式为左对齐,宽度为80;第二列标题为"子项",对齐方式为左对齐,宽度为50。

注意: CListCtrl在显示为Report样式时,主项为第一项,后面紧跟着的为子项。主项和子项的数据写入不同,这将在后面提到。

InsertItem被用于向显示列表中插入主项数据,SetItemText被用于向列表中的子项写入数据,例如:
 list.InsertItem(0,"主数据");
插入数据"主数据"到第一行的主项
 list.SetItemText(0,1,"子数据");
将第一行的子项数据设置为"子数据"

其完整的代码为:
/*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;


}

SortListCallBack必须在每一条数据写入完毕后调用,因为排序的算法是假定在写入数据之前,列表中的数据排列是有序的。

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

小节

CListCtrl对于初次使用MFC框架的程序员来说,其数据写入的方法有点让人难以理解,我在这里也只是根据我自己在工作中遇到的问题,而将CListCtrl的一种使用方法讲解了一下,大家如果有兴趣去看一下MSDN上关于CListCtrl的其他用途。关于上面提到的数据排序,我使用了通过函数地址来调用函数的方法。这种方法的一个好处就是可以在不同的情况下定义不同的排序方式,而对已有的程序结构没有影响。
以下为具体的代码,供大家参考,欢迎大家批评指教!

.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;

 
}

------------------------------------------

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值