CListCtrl控件列表控件使用大全

创建图形列表并和CListCtrl关联:
m_image_list.Create(IDB_CALLER2, 16, 10, RGB(192,192, 192));
m_image_list.SetBkColor( GetSysColor( COLOR_WINDOW ) );
m_caller_list.SetImageList( &m_image_list, LVSIL_SMALL);
为报表添加4列:
   char *szColumn[]={"昵称","IP地址","登陆时间","状态"};
   int widths[]={100,98,70,55};
   LV_COLUMN lvc;
   lvc.mask=LVCF_FMT|LVCF_WIDTH|LVCF_TEXT|LVCF_SUBITEM;
   lvc.fmt=LVCFMT_LEFT;
   for(int i=0;i<4;i++) {//插入各列
    lvc.pszText=szColumn[i];
    lvc.cx=widths[i];
    lvc.iSubItem=i;
    m_caller_list.InsertColumn(i,&lvc);
   }
为报表添加两项,以附加方式添加:
char* data[4];
data[0]="所有人";
data[1]="0.0.0.0";
data[3]="在线";
data[2]=new char;
CTime now=CTime::GetCurrentTime();
       CString temp = now.Format("%H:%M:%S");
data[2]=temp.GetBuffer(1);
LV_ITEM lvi;
lvi.mask=LVIF_TEXT|LVIF_IMAGE|LVIF_PARAM;
lvi.iSubItem=0;
lvi.pszText=(char *)data[0];
lvi.iImage = 0;
lvi.iItem=0;
m_caller_list.InsertItem(&lvi);
for (int j=0;j<4;j++) m_caller_list.SetItemText(count,j,data[j]);
count++;
lvi.iImage = 1;
lvi.iItem=count;
m_caller_list.InsertItem(&lvi);
data[0]="cherami";
data[1]="127.0.0.1"; 
for (int n=0;n<4;n++) m_caller_list.SetItemText(count,n,data[n]);
count++;

设置报表的样式
选中一整行:
m_list_ctrl.SetExtendedStyle(m_list_ctrl.GetExtendedStyle()|LVS_EX_FULLROWSELECT); 
绘制表格:
m_list_ctrl.SetExtendedStyle(m_list_ctrl.GetExtendedStyle()|LVS_EX_GRIDLINES); 
带复选框:
m_list_ctrl.SetExtendedStyle(m_list_ctrl.GetExtendedStyle()|LVS_EX_CHECKBOXES); 
自动切换:
m_list_ctrl.SetExtendedStyle(m_list_ctrl.GetExtendedStyle()|LVS_EX_TRACKSELECT);

选定一行:
设置CListCtrl的Show selection always选项
SetItemState (iIndex, LVIS_SELECTED|LVIS_FOCUSED, LVIS_SELECTED|LVIS_FOCUSED)

选中一个或多个项目时,会发送LVN_ITEMCHANGED消息,可以使用
GetSelectedCount()方法得到被选定的项的数目。

点击列头的消息响应:
ON_NOTIFY(HDN_ITEMCLICKW, 0, ResponseFunc)
消息,需要自己添加 
或者:
ON_NOTIFY(LVN_COLUMNCLICK, ID_yourCtrl, ResponseFunc)//向导添加
前者后响应,后者先响应

响应函数:
ResponseFunc(NMHDR *pNMHDR, LRESULT *pResult)

双击CListCtrl中的ITEM的消息是及消息函数:
ON_NOTIFY(NM_DBLCLK, ID_yourCtrl, ResponseFunc)

单击ITEM的消息响应:
ON_NOTIFY(NM_CLICK, ID_yourCtrl, ResponseFunc)
ResponseFunc(NMHDR *pNMHDR, LRESULT *pResult)


HDN_ITEMCLICK    就是Header control Notify message for mouse left click on the Header control!
而HDN_ITEMCLICK是当List View中存在一个Header Contrl时,Header Ctrl通知父窗口List View的!

CListCtrl中的Item被选中触发LBN_SELCHANGE(通过WM_COMMAND)消息!

删除CListCtrl中选定的项:
POSITION pos;
int nIndex;

for(; pos= GetFirstSelectedItemPosition();)
{
nIndex = GetNextSelectedItem(pos);
DeleteItem(nIndex);
}

在ListCtrl中进行排序
列表控件(CListCtrl)的顶部有一排按钮,用户可以通过选择不同的列来对记录进行排序。但是 CListCtrl并没有自动排序的功能,我们需要自己添加一个用于排序的回调函数来比较两个数据的大小,此外还需要响应排序按钮被点击的消息。下面讲述一下具体的做法。

CListCtrl提供了用于排序的函数,函数原型为:BOOL CListCtrl::SortItems( PFNLVCOMPARE pfnCompare, DWORD dwData )。其中第一个参数为全局排序函数的地址,第二个参数为用户数据,你可以根据你的需要传递一个数据或是指针。该函数返回-1代表第一项排应在第二项前面,返回1代表第一项排应在第二项后面,返回0代表两项相等。

用于排序的函数原形为:int CALLBACK ListCompare(LPARAM lParam1, LPARAM lParam2, LPARAM lParamSort),其中第三个参数为调用者传递的数据(即调用SortItems时的第二个参数dwData)。第一和第二个参数为用于比较的两项的ItemData,你可以通过DWORD CListCtrl::GetItemData( int nItem )/BOOL CListCtrl::SetItemData( int nItem, DWORD dwData )来对每一项的ItemData进行存取。在添加项时选用特定的CListCtrl::InsertItem也可以设置该值。由于你在排序时只能通过该值来确定项的位置所以你应该比较明确的确定该值的含义。

最后一点,我们需要知道什么时候需要排序,实现这点可以在父窗口中对LVN_COLUMNCLICK消息进行处理来实现。

下面我们看一个例子,这个例子是一个派生类,并支持顺序/倒序两种方式排序。为了简单我对全局数据进行排序,而在实际应用中会有多组需要排序的数据,所以需要通过传递参数的方式来告诉派序函数需要对什么数据进行排序。


//全局数据
struct DEMO_DATA
{
char szName[20];
int iAge;
}strAllData[5]={ {"王某",30},{"张某",40},{"武某",32},{"陈某",20},{"李某",36}};

//CListCtrl派生类定义
class CSortList : public CListCtrl
{
// Construction
public:
CSortList();
BOOL m_fAsc;//是否顺序排序
int m_nSortedCol;//当前排序的列
protected:
//{ {AFX_MSG(CSortList)
//}}AFX_MSG
...
};

//父窗口中包含该CListCtrl派生类对象
class CSort_in_list_ctrlDlg : public CDialog
{
// Construction
public:
CSort_in_list_ctrlDlg(CWnd* pParent = NULL); // standard constructor

// Dialog Data
//{ {AFX_DATA(CSort_in_list_ctrlDlg)
enum { IDD = IDD_SORT_IN_LIST_CTRL_DIALOG };
CSortList m_listTest;
//}}AFX_DATA
}

//在父窗口中定义LVN_COLUMNCLICK消息映射
BEGIN_MESSAGE_MAP(CSort_in_list_ctrlDlg, CDialog)
//{ {AFX_MSG_MAP(CSort_in_list_ctrlDlg)
ON_NOTIFY(LVN_COLUMNCLICK, IDC_LIST1, OnColumnclickList1)
//}}AFX_MSG_MAP
END_MESSAGE_MAP()

//初始化数据
BOOL CSort_in_list_ctrlDlg::OnInitDialog()
{
CDialog::OnInitDialog();

//初始化ListCtrl中数据列表
m_listTest.InsertColumn(0,"姓名");
m_listTest.InsertColumn(1,"年龄");
m_listTest.SetColumnWidth(0,80);
m_listTest.SetColumnWidth(1,80);
for(int i=0;i<5;i++)
{
   m_listTest.InsertItem(i,strAllData[i].szName);
   char szAge[10];
   sprintf(szAge,"%d",strAllData[i].iAge);
   m_listTest.SetItemText(i,1,szAge);
   //设置每项的ItemData为数组中数据的索引
   //在排序函数中通过该ItemData来确定数据
   m_listTest.SetItemData(i,i);
}
return TRUE; // return TRUE unless you set the focus to a control
}

//处理消息
void CSort_in_list_ctrlDlg::OnColumnclickList1(NMHDR* pNMHDR, LRESULT* pResult) 
{
NM_LISTVIEW* pNMListView = (NM_LISTVIEW*)pNMHDR;
//设置排序方式
if( pNMListView->iSubItem == m_listTest.m_nSortedCol )
   m_listTest.m_fAsc = !m_listTest.m_fAsc;
else
{
   m_listTest.m_fAsc = TRUE;
   m_listTest.m_nSortedCol = pNMListView->iSubItem;
}
//调用排序函数
m_listTest.SortItems( ListCompare, (DWORD)&m_listTest );        
*pResult = 0;
}

//排序函数实现
int CALLBACK ListCompare(LPARAM lParam1, LPARAM lParam2, LPARAM lParamSort)
{
//通过传递的参数来得到CSortList对象指针,从而得到排序方式
CSortList* pV=(CSortList*)lParamSort;

//通过ItemData来确定数据
DEMO_DATA* pInfo1=strAllData+lParam1;
DEMO_DATA* pInfo2=strAllData+lParam2;
CString szComp1,szComp2;
int iCompRes;
switch(pV->m_nSortedCol)
{
case(0):
   //以第一列为根据排序
   szComp1=pInfo1->szName;
   szComp2=pInfo2->szName;
   iCompRes=szComp1.Compare(szComp2);
   break;
case(1):
   //以第二列为根据排序
   if(pInfo1->iAge == pInfo2->iAge)
    iCompRes = 0;
   else
    iCompRes=(pInfo1->iAge < pInfo2->iAge)?-1:1;
   break;
default:
   ASSERT(0);
   break;
}
//根据当前的排序方式进行调整
if(pV->m_fAsc)
   return iCompRes;
else
   return iCompRes*-1;
}

排序最快:
CListCtrl::SortItems
Example

// Sort the item in reverse alphabetical order.
static int CALLBACK 
MyCompareProc(LPARAM lParam1, LPARAM lParam2, LPARAM lParamSort)
{
// lParamSort contains a pointer to the list view control.
// The lParam of an item is just its index.
CListCtrl* pListCtrl = (CListCtrl*) lParamSort;
CString    strItem1 = pListCtrl->GetItemText(lParam1, 0);
CString    strItem2 = pListCtrl->GetItemText(lParam2, 0);

return strcmp(strItem2, strItem1);
}

void snip_CListCtrl_SortItems()
{
// The pointer to my list view control.
extern CListCtrl* pmyListCtrl;

// Sort the list view items using my callback procedure.
pmyListCtrl->SortItems(MyCompareProc, (LPARAM) pmyListCtrl);
}


If you don’t want to allow the users to sort the list by clicking on the header, you can use the style LVS_NOSORTHEADER. However, if you do want to allow sorting, you do not specify the LVS_NOSORTHEADER. The control, though, does not sort the items. You have to handle the HDN_ITEMCLICK notification from the header control and process it appropriately. In the code below, we have used the sorting function SortTextItems() developed in a previous section. You may choose to sort the items in a different manner. 
Step 1: Add two member variables
Add two member variables to the CListCtrl. The first variable to track which column has been sorted on, if any. The second variable to track if the sort is ascending or descending. 
        int nSortedCol; 
        BOOL bSortAscending;

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
CListCtrl 控件的表头默认只能显示一行,如果需要实现多行显示,可以通过继承 CHeaderCtrl 并重写其 OnPaint 函数来实现。 具体实现步骤如下: 1. 定义一个新的,继承自 CHeaderCtrl 。 2. 在该的头文件中添加一个变量,用于保存每个表头项的高度。 3. 在该的构造函数中,设置表头的高度并获取每个表头项的高度。 4. 重写 OnPaint 函数,绘制多行表头。 下面是一个示例代码: ```cpp class CMultiLineHeaderCtrl : public CHeaderCtrl { public: CMultiLineHeaderCtrl(); virtual ~CMultiLineHeaderCtrl(); protected: int m_nHeaderHeight; // 表头高度 CArray<int, int> m_arrHeaderItemHeight; // 表头项高度 afx_msg void OnPaint(); DECLARE_MESSAGE_MAP() }; CMultiLineHeaderCtrl::CMultiLineHeaderCtrl() { m_nHeaderHeight = 50; // 设置表头高度 for (int i = 0; i < GetItemCount(); i++) { HDITEM hdi; hdi.mask = HDI_HEIGHT; GetItem(i, &hdi); m_arrHeaderItemHeight.Add(hdi.cy); // 获取每个表头项的高度 } } CMultiLineHeaderCtrl::~CMultiLineHeaderCtrl() { } BEGIN_MESSAGE_MAP(CMultiLineHeaderCtrl, CHeaderCtrl) ON_WM_PAINT() END_MESSAGE_MAP() void CMultiLineHeaderCtrl::OnPaint() { CPaintDC dc(this); CRect rc; GetClientRect(&rc); // 绘制表头背景 CBrush brush(GetSysColor(COLOR_3DFACE)); dc.FillRect(rc, &brush); // 绘制表头项 int nOffset = 0; for (int i = 0; i < GetItemCount(); i++) { HDITEM hdi; TCHAR szText[255]; hdi.mask = HDI_TEXT | HDI_FORMAT; hdi.pszText = szText; hdi.cchTextMax = sizeof(szText) / sizeof(TCHAR); GetItem(i, &hdi); // 计算表头项的矩形区域 CRect rcItem(nOffset, 0, nOffset + GetColumnWidth(i), m_arrHeaderItemHeight[i]); // 绘制表头项的背景 CBrush brushItem(GetSysColor(COLOR_BTNFACE)); dc.FillRect(rcItem, &brushItem); // 绘制表头项的文本 dc.SetBkMode(TRANSPARENT); dc.DrawText(szText, -1, rcItem, DT_CENTER | DT_SINGLELINE | DT_VCENTER); // 更新表头项的位置偏移量 nOffset += GetColumnWidth(i); } // 绘制表头分隔线 nOffset = 0; for (int i = 0; i < GetItemCount() - 1; i++) { CRect rcDivider(nOffset + GetColumnWidth(i) - 1, 0, nOffset + GetColumnWidth(i), m_nHeaderHeight); dc.FillRect(rcDivider, &brush); nOffset += GetColumnWidth(i); } } ``` 在使用的时候,只需要将 CListCtrl 控件的表头替换为 CMultiLineHeaderCtrl 控件即可实现多行表头的显示: ```cpp CMultiLineHeaderCtrl* pHeaderCtrl = new CMultiLineHeaderCtrl(); pHeaderCtrl->SubclassDlgItem(IDC_LIST_HEADER, this); ``` 其中,IDC_LIST_HEADER 是 CListCtrl 控件的表头控件的 ID。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值