CListCtrl::SetExtendedStyle
CListCtrl的成员函数声明
DWORD SetExtendedStyle(
DWORD dwNewStyle );
函数功能设置CListCtrl的扩展样式
dwNewStyle指定的扩展样式
LVS_EX_GRIDLINES //绘制表格,网格线。
LVS_EX_SUBITEMIMAGES//子项目图标列表
LVS_EX_CHECKBOXES //带复选框
LVS_EX_TRACKSELECT //自动换行
LVS_EX_HEADERDRAGDROP//报表头可以拖拽
LVS_EX_FULLROWSELECT //选择整行,允许选择整行。
LVS_EX_ONECLICKACTIVATE//单击激活单击选中项目。
LVS_EX_TWOCLICKACTIVATE//双击激活
LVS_EX_FLATSB//扁平滚动条
LVS_EX_REGIONAL
LVS_EX_INFOTIP
LVS_EX_UNDERLINEHOT
LVS_EX_UNDERLINECOLD
LVS_EX_MULTIWORKAREAS//多工作区
----------------------------------
例子:m_TextList.SetExtendedStyle(...);
This entry was posted in VC and tagged 函数, 详细介绍. Bookmark the permalink.
← VC:IP段到IP段,写入文件
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); 之类的代码。 |
设置报表的样式 选中一整行: 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; Step 2: Initialize them in the constructor. Initialize nSortedCol to -1 to indicate that no column has been sorted on. If the list is initially sorted, then this variable should reflect that. nSortedCol = -1; bSortAscending = TRUE; Step 3: Add entry in message map to handle HDN_ITEMCLICK Actually you need to add two entries. For HDN_ITEMCLICKA and HDN_ITEMCLICKW. Do not use the class wizard to add the entry. For one, you need to add two entries whereas the class wizard will allow you only one. Secondly, the class wizard uses the wrong macro in the entry. It uses ON_NOTIFY_REFLECT() instead of ON_NOTIFY(). Since the HDN_ITEMCLICK is a notification from the header control to the list view control, it is a direct notification and not a reflected one. ON_NOTIFY(HDN_ITEMCLICKA, 0, OnHeaderClicked) ON_NOTIFY(HDN_ITEMCLICKW, 0, OnHeaderClicked) Note that we specify the same function for both the notification. Actually the program will receive one or the other and not both. What notification it receives will depend on the OS. The list view control on Windows 95 will send the ANSI version and the control on NT will send the UNICODE version. Also, note that the second argument is zero. This value filters for the id of the control and we know that header control id is zero. Step 4: Write the OnHeaderClicked() function Here’s where you decide what to do when the user clicks on a column header. The expected behaviour is to sort the list based on the values of the items in that column. In this function we have used the SortTextItems() function developed in a previous section. If any of the columns displays numeric or date values, then you would have to provide custom sorting for them. void CMyListCtrl::OnHeaderClicked(NMHDR* pNMHDR, LRESULT* pResult) { HD_NOTIFY *phdn = (HD_NOTIFY *) pNMHDR; if( phdn->iButton == 0 ) { // User clicked on header using left mouse button if( phdn->iItem == nSortedCol ) bSortAscending = !bSortAscending; else bSortAscending = TRUE; nSortedCol = phdn->iItem; SortTextItems( nSortedCol, bSortAscending ); } *pResult = 0; } 让CListCtrl的SubItem也具有编辑功能: 要重载一个文本框,然后在LVN_BEGINLABELEDIT时改变文本框位置。 CInEdit m_InEdit; if( ( GetStyle() & LVS_TYPEMASK ) == LVS_REPORT && ( m_nEditSubItem != 0 ) ) { HWND hwndEdit; CRect rtBound; CString strText; hwndEdit = (HWND)SendMessage( LVM_GETEDITCONTROL ); GetSubItemRect( pDispInfo->item.iItem, m_nEditSubItem, LVIR_LABEL, rtBound ); m_InEdit.SubclassWindow( hwndEdit ); m_InEdit.m_left = rtBound.left; strText = GetItemText( pDispInfo->item.iItem, m_nEditSubItem ); m_InEdit.SetWindowText( strText ); } void CInEdit::OnWindowPosChanging(WINDOWPOS FAR* lpwndpos) { CRect rtClient; lpwndpos->x = m_left; // m_left在LVN_BEGINLABELEDIT中设置 CEdit::OnWindowPosChanging(lpwndpos); // TODO: Add your message handler code here }
|
LVS_ALIGNLEFT 用来确定表项的大小图标以左对齐方式显示;
LVS_ALIGNTOP 用来确定表项的大小图标以顶对齐方式显示;
LVS_AUTOARRANGE 用来确定表项的大小图标以自动排列方式显示;
LVS_EDITLABELS 设置表项文本可以编辑,父窗口必须设有LVN_ENDLABELEDIT风格;
LVS_ICON 用来确定大图标的显示方式;
LVS_LIST 用来确定列表方式显示;
LVS_NOCOLUMNHEADER 用来确定在详细资料方式时不显示列表头;
LVS_NOLABELWRAP 用来确定以单行方式显示图标的文本项;
LVS_NOSCROLL 用来屏蔽滚动条;
LVS_NOSORTHEADER 用来确定列表头不能用作按钮功能;
LVS_OWNERDRAWFIXED 在详细列表方式时允许自绘窗口;
LVS_REPORT 用来确定以详细资料即报告方式显示;
LVS_SHAREIMAGELISTS用来确定共享图像列表方式;
LVS_SHOWSELALWAYS 用来确定一直显示被选中表项方式;
LVS_SINGLESEL 用来确定在某一时刻只能有一项被选中;
LVS_SMALLICON 用来确定小图标显示方式;
LVS_SORTASCENDING 用来确定表项排序时是基于表项文本的升序方式;
LVS_SORTDESCENDING 用来确定表项排序时是基于表项文本的降序方式;
typedef struct _LV_ITEM {
UINT mask; //结构成员屏蔽位
int iItem; //表项索引号
int iSubItem; //子表项索引号
UINT state; //表项状态
UINT stateMask; //状态有效性屏蔽位
LPTSTR pszText; //表项名文本
int cchTextMax; //表项名最大长度
int iImage; // 表项图标的索引号
LPARAM lParam; // 与表项相关的32位数
} LV_ITEM;
typedef struct _LV_COLUMN {
UINT mask; //结构成员有效性屏蔽位
int fmt; //表列对齐方式
int cx; //表列的象素宽度
LPTSTR pszText; //表列的表头名
int cchTextMax; //表列名的文本长度
int iSubItem; //与表列关联的子表项索引号
} LV_COLUMN;
其中fmt可以取如下值:
LVCFMT_CENTER 表列居中对齐
LVCFMT_LEFT 表列左对齐
CListCtrl 使用完全指南
构造函数
ClistCtrl构造一个CListCtrl对象。
Create创建列表控件并将其附加给CListCtrl对象。
属性
GetBkColor获取列表视图控件的背景色。
SetBkColor设置列表视图控件的背景色。
GetImageList获取用于绘制列表视图项的图象列表的句柄。
SetImageList指定一个图象列表到列表视图控件。
GetItemCount获取列表视图控件中的项的数量。
GetItem获取列表视图项的属性。
GetCallbackMask获取列表视图控件的回调掩码。
SetCallbackMask设置列表视图控件的回调掩码。
GetNextItem查找指定特性和指定指定项关系的列表视图项。
GetFirstSeletedItemPosition在列表视图控件中获取第一个选择的列表视图项的位置。
GetNextSeletedItem为重复而获取下一个选择的列表视图。
GetItemRect获取项的有界矩形。
SetItemPosition在列表视图控件中移动一项到指定位置。
GetItemPosition获取列表视图项的位置。
GetStringWidth指定需要显示所有指定字符串的最小列宽。
GetEditControl获取用于编辑一个项文本的编辑控件的句柄。
GetColumn获取控件的列的属性。
SetColumn设置列表视图列的属性。
GetColumnWidth获取报表视图或列表视图中的列的宽度。
SetColumnWidth改变报表视图或列表视图中的列的宽度。
GetCheck获取与某项相关的状态图象的当前显示状态。
SetCheck设置与某项相关的状态图象的当前显示状态。
GetViewRect获取列表视图控件中所有项的有界矩形。
GetTextColor获取列表视图控件的文本颜色。
SetTextColor设置列表视图控件的文本颜色。
GetTextBkColor获取列表视图控件的文本背景色。
SetTextBkColor设置列表视图控件的文本背景色。
GetTopIndex获取最高级项的索引。
GetCountPerPage计算可正好垂直放入列表视图控件中的项的数目。
GetOrigin获取列表视图控件的最初的当前视图。
SetItemState改变列表视图控件的项的状态。
GetItemState获取列表视图控件的项的状态。
GetItemText获取列表视图项或子项的文本。
SetItemText设置列表视图项或子项的文本。
SetItemCount准备一个列表视图控件以添加大量的项。
GetItemData获取与某项相关的应用所指定的值。
SetItemData设置项的应用指定的值。
GetSelectedCount获取列表视图控件中选择项的数量。
SetColumnOrderArray设置列表视图控件的列序(左或右)。
GetColumnOrderArray获取列表视图控件的列序(左或右)。
SetIconSpacing设置列表视图控件中的图标的距离。
GetHeaderCtrl获取列表视图控件的标题控件。
GetHotCursor获取在热调试对列表视图控件有效时使用的游标。
SetHotCursor设置在热调试对列表视图控件有效时使用的游标。
GetSubItemRect获取列表视图控件中某项的有界矩形。
GetHotItem获取当前在游标下的列表视图项。
SetHotItem设置列表视图控件的当前热项。
GetSelectionMark获取列表视图控件的选择屏蔽。
SetSelectionMark设置列表视图控件的选择屏蔽。
GetExtendedStyle获取列表视图控件的当前扩展风格。
SetExtendedStyle设置列表视图控件的当前扩展风格。
SubItemHitTest指定哪个列表视图项在指定位置。
GetWorkAreas获取列表视图控件的当前工作区。
GetNumberOfWorkAreas获取列表视图控件的当前工作区数量。
SetItemCountEx设置虚列表视图控件的项的数量。
SetWorkAreas设置列表视图控件中图标可以显示的区域。
ApproximateViewRect指定显示列表视图控件项所需的宽度和高度。
GetBkImage获取列表视图控件的当前背景图象。
SetBkImage设置列表视图控件的当前背景图象。
GetHoverTime获取列表视图控件的当前逗留时间。
SetHoverTime设置列表视图控件的当前逗留时间。
操作
InsertItem在列表视图控件中插入一个新项。
DeleteItem从控件中删除一项。
DeleteAllItems从控件中删除所有项。
FindItem查找具有指定的字符的列表视图项。
SortItems使用应用定义的比较函数排序列表视图项。
HitTest指定哪个列表视图在指定的位置上。
EnsureVisible保证项是可见的。
Scroll滚动列表视图控件的内容。
ReDrawItems强迫列表视图控件刷新一些项。
Update强迫控件刷新一个指定的项。
Arrange调整一栏里的项。
EditLabel开始项文本该处编辑。
InsertColumn插入列表视图控件中的新列。
DeleteColumn从列表视图控件中删除一列。
CreateDragImage为指定的项构造一个拖动图象列表。
可覆盖的函数
DrawItem当自绘制控件的可视部分改变时被调用。
创建图形列表并和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 selectionalways选项
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 controlNotify 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);
}
POSITION GetFirstSelectedItemPosition() const
如果函数执行成功,则返回条目的POSITION值,如果返回值为KULL,则表示当前列表视图控件中没有条目选中。
Int GetNextSelectedItem(POSITION& pos) const
如果函数执行成功,则返回列表视图控件中下一个被选中的条目索引。其中参数pos为将接收条目POSITION值的变量。
注意:使用这两个函数的时候 ListCtrl最好是有LVS_FULLROWSELECT的扩展风格。
在ListCtrl中进行排序
列表控件(CListCtrl)的顶部有一排按钮,用户可以通过选择不同的列来对记录进行排序。但是 CListCtrl并没有自动排序的功能,我们需要自己添加一个用于排序的回调函数来比较两个数据的大小,此外还需要响应排序按钮被点击的消息。下面讲述一下具体的做法。
CListCtrl提供了用于排序的函数,函数原型为:BOOLCListCtrl::SortItems( PFNLVCOMPARE pfnCompare, DWORD dwData )。其中第一个参数为全局排序函数的地址,第二个参数为用户数据,你可以根据你的需要传递一个数据或是指针。该函数返回-1代表第一项排应在第二项前面,返回1代表第一项排应在第二项后面,返回0代表两项相等。
用于排序的函数原形为:int CALLBACK ListCompare(LPARAM lParam1, LPARAM lParam2, LPARAMlParamSort),其中第三个参数为调用者传递的数据(即调用SortItems时的第二个参数dwData)。第一和第二个参数为用于比较的两项的ItemData,你可以通过DWORD CListCtrl::GetItemData( int nItem )/BOOLCListCtrl::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 acontrol
}
//处理消息
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 byclicking on the header, you can use the style LVS_NOSORTHEADER. However, if youdo 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_ITEMCLICKnotification from the header control and process it appropriately. In the codebelow, we have used the sorting function SortTextItems() developed in a previoussection. 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 whichcolumn has been sorted on, if any. The second variable to track if the sort is ascendingor descending.
int nSortedCol;
BOOL bSortAscending;
Step 2: Initialize them in the constructor.
Initialize nSortedCol to -1 to indicate that no column has been sorted on. Ifthe list is initially sorted, then this variable should reflect that.
nSortedCol = -1;
bSortAscending = TRUE;
Step 3: Add entry in message map to handle HDN_ITEMCLICK
Actually you need to add two entries. For HDN_ITEMCLICKA and HDN_ITEMCLICKW. Donot use the class wizard to add the entry. For one, you need to add two entrieswhereas the class wizard will allow you only one. Secondly, the class wizarduses the wrong macro in the entry. It uses ON_NOTIFY_REFLECT() instead ofON_NOTIFY(). Since the HDN_ITEMCLICK is a notification from the header controlto the list view control, it is a direct notification and not a reflected one.
ON_NOTIFY(HDN_ITEMCLICKA, 0, OnHeaderClicked)
ON_NOTIFY(HDN_ITEMCLICKW, 0, OnHeaderClicked)
Note that we specify the same function for both the notification.Actually the program will receive one or the other and not both. Whatnotification it receives will depend on the OS. The list view control onWindows 95 will send the ANSI version and the control on NT will send theUNICODE version.
Also, note that the second argument is zero. This value filters for the id ofthe control and we know that header control id is zero.
Step 4: Write the OnHeaderClicked() function
Here’s where you decide what to do when the userclicks on a column header. The expected behaviour is to sort the list based onthe values of the items in that column. In this function we have used theSortTextItems() function developed in a previous section. If any of the columnsdisplays numeric or date values, then you would have to provide custom sortingfor them.
void CMyListCtrl::OnHeaderClicked(NMHDR* pNMHDR, LRESULT* pResult)
{
HD_NOTIFY *phdn = (HD_NOTIFY *)pNMHDR;
if( phdn->iButton ==0 )
{
// User clicked on header using left mouse button
if( phdn->iItem == nSortedCol )
bSortAscending = !bSortAscending;
else
bSortAscending = TRUE;
nSortedCol = phdn->iItem;
SortTextItems( nSortedCol, bSortAscending );
}
*pResult = 0;
}
让CListCtrl的SubItem也具有编辑功能:
要重载一个文本框,然后在LVN_BEGINLABELEDIT时改变文本框位置。
CInEdit m_InEdit;
if( ( GetStyle() & LVS_TYPEMASK ) ==LVS_REPORT && ( m_nEditSubItem != 0 ) )
{
HWND hwndEdit;
CRect rtBound;
CString strText;
hwndEdit =(HWND)SendMessage( LVM_GETEDITCONTROL );
GetSubItemRect(pDispInfo->item.iItem, m_nEditSubItem, LVIR_LABEL, rtBound );
m_InEdit.SubclassWindow( hwndEdit );
m_InEdit.m_left = rtBound.left;
strText = GetItemText(pDispInfo->item.iItem, m_nEditSubItem );
m_InEdit.SetWindowText( strText );
}
void CInEdit::OnWindowPosChanging(WINDOWPOS FAR* lpwndpos)
{
CRect rtClient;
lpwndpos->x = m_left; // m_left在LVN_BEGINLABELEDIT中设置
CEdit::OnWindowPosChanging(lpwndpos);
// TODO: Add your message handler code here
}