MFC控件——ListCtrl控件

来源:互联网 发布:青岛特来电怎么样知乎 编辑:程序博客网 时间:2024/05/05 00:16
 

声明:

1、本文为翻译文章,水平有限,错误之处,烦请指正(chinajiezhang@gmail.com)。

2、使用VS2005,所以在某些细节上和原作者不太一样

3、工程中使用图片和作者略有不同

4、文章如有争议,以原作者文章为准

5、转载请标明出处

原文链接:http://www.functionx.com/visualc/controls/listcontrol.htm

一、ListControl概述

概述:

ListCtrl控件由四种列表的显示方式构成,它最典型的用法是使用icon来显示。ListCtrl通常显示下面四种项目(item):

Icons: 控件显示项的列表,使用32*32(像素)大小的icons,推荐使用这么大小的icon,如果你想用图标来概述你的观点。

Small Icons:和剩下的两项相似的想法,它使用16*16(像素)大小的图标来显示一个单一的列表项。再次,没有提供关于这个列表的详细说明。这种列表是用来组织和其他主题在不同的列,如果列表有序,序列编排为按字母顺序从左到右。

List:这种列表,使用small icons,也是组织列;这种情况下,在第二列填充之前必须填充第一列。如果列表有序,序列编排为自顶向下的顺序。

Report:安排项的目的是提供列表开发者的信息。

二、实践学习:List Control介绍

1. 用Visual C++或者Visual Studio 创建一个名为DeptStore2的MFC应用程序

2. 基于Dialog创建它

3. 删除“TODO: 在此放置对话框控件。”行和确定按钮。

4. 将“取消”按钮的标题改为“关闭”

(1)      创建List Control

ListCtrl控件在MFC类库用CListCtrl类来实现。在设计的时候,创建一个ListCtrl,在toolbox中点击按钮 后单击对话框中想要放置的区域。通常来说,需要拉伸默认ListCtrl的默认大小,因为它通常需要一个更大的区域。

为了编程创建一个ListCtrl,声明CListCtrl变量或者CListCtrl指针。初始化控件调用它的Create方法,示例如下:

 

BOOL CDeptStore2Dlg::OnInitDialog(){ CDialog::OnInitDialog(); // ... // TODO: 在此添加额外的初始化代码 CListCtrl *lstCtrl = new CListCtrl; lstCtrl->Create(WS_CHILD | WS_VISIBLE, CRect(10, 10, 320, 280), this, 0x285); return TRUE; // 除非将焦点设置到控件,否则返回 TRUE}

 

 

 

译者提示:这是内存泄露的做法,也许作者意图只是为了说明应在OnInitDialog添加代码,正确的做法在CDeptStore2Dlg中添加变量m_lstCtrl,之后在OnInitDialog中new,最后在析构函数中delete。

在之前提到,ListCtrl可以显示四种情况之一,在设计的时候应该在窗口属性中指定想要的风格,在属性中选择combo box,默认值是Icon,他可能是情况是:

Icon:当编程创建控件的时候,增加LVS_ICON风格;

Small Icon:类似的可以动态添加LVS_SMALLICON风格:

List:当创建的时候可以类似的添加LVS_REPORT风格;

Report:这种情况清楚的显示了列,同样应添加LVS_REPORT风格。

 

除了常规的几种风格外,Win32类库为ListCtrl提供了扩展风格,提供了一个扩展风格,调用CListCtrl::SetExtendStyle()方法,它的语法为:

DWORD SetExtendedStyle(DWORD dwNewStyle);

当调用这个函数的时候,把指定的扩展风格或者他们的组合风格作为参数来传递。其中的一些值是:

LVS_EX_CHECKBOXES:在左侧显示选择框(check box)

LVS_EX_FULLROWSELECT:这种风格允许Report View的整行来被选择而不是仅仅一项;

LVS_EX_GRIDLINES:相对上一种风格来说增加了水平和垂直分割线。

 

LVS_EX_TRACKSELECT:当设置这种风格的时候,用户可以隔项选取。

列表控件仅仅能在控件内显示,如果这儿他们太多或者各项的总宽度比控件可以显示的区域大,应该给它配备水平滚动条和垂直滚动条,或者两个都有。如果你想阻止滚动条的显示,设置无滚动条的属性为true或者创建属性为LVS_NOSCROLL风格。

一旦list控件创建,使用者可以选择一个选项通过点击它,然后选择更多的项,使用者也可以辅助ctrl来随机选择或者辅助shift连续选取,下面是一个随机选取的一个样例:

 

如果你不想用户每次选择多项的话,你可以设置通过增加LVS_SINGLESEL单选属性为true。

任何项被选中的时候都是高亮的,当用户点击另一个控件或者另一个应用程序的时候,你是否可以确定你想那些被选项仍旧被选中呢?典型的情况是在设计的时候通过选择“Always”属性来实现。默认情况下,设置为false,意味着当控件失去焦点或者它的父窗口无效的情况下,被选项将不会显示(高亮)。否则,你可以把这个属性设置为true,这样即使它失去焦点也也可以是选中状态。这个属性也可以通过设置LVS_SHOWSELALWAYS风格来实现。

三、实践学习:创建一个ListCtrl

1.在ToolBox上,单击ListCtrl控件然后点击对话框,将ID改为IDC_STORE_ITEMS

2.右击控件,添加变量

3.设置变量为:m_StoreItems

4.点击完成

ListCtrl的项

通过添加(使用toolboax)或者动态创建一个ListCtrl之后,下一个环节你应该让它来显示项。这个可以通过调用CListCtrl::InsertItem()方法,它的一种声明如下:

int InsertItem(const LVITEM* pItem);

这个版本需要一个LVITEM的指针作为参数,LVITEM定义如下:

 

typedef struct { UINT mask; int iItem; int iSubItem; UINT state; UINT stateMask; LPTSTR pszText; int cchTextMax; int iImage; LPARAM lParam;#if (_WIN32_IE >= 0x0300) int iIndent;#endif #if (_WIN32_WINNT >= 0x0501) int iGroupId; UINT cColumns; UINT puColumns;#endif #if (_WIN32_WINNT >= 0x0600) int piColFmt; int iGroup;#endif } LVITEM, *LPLVITEM;

mask:用来指定你想给当前项设置的类型。

iItem:指定改变项的索引,第一个item应该为0,第二个item为1等等。

iSumItem:当前值的的子项,如果当前项是主导项,iSumItem应该存储从0开始的数组。

pszText:要显示项的字符串,你可以通过cchTextMask指定文本的长度。

初始化LVITEM之后,调用InsertItem()方法来为list添加一个项。以下是范例:

 

BOOL CDeptStore2Dlg::OnInitDialog(){ CDialog::OnInitDialog(); // ... // TODO: 在此添加额外的初始化代码 LVITEM lvItem; lvItem.mask = LVIF_TEXT; lvItem.iItem = 0; lvItem.iSubItem = 0; lvItem.pszText = _T("Sandra C. Anschwitz"); m_StoreItems.InsertItem(&lvItem); lvItem.mask = LVIF_TEXT; lvItem.iItem = 1; lvItem.iSubItem = 0; lvItem.pszText = _T("Roger A. Miller"); m_StoreItems.InsertItem(&lvItem); lvItem.mask = LVIF_TEXT; lvItem.iItem = 2; lvItem.iSubItem = 0; lvItem.pszText = _T("Marie-Julie W. Gross"); m_StoreItems.InsertItem(&lvItem); lvItem.mask = LVIF_TEXT; lvItem.iItem = 3; lvItem.iSubItem = 0; lvItem.pszText = _T("Ella Pius Roger"); m_StoreItems.InsertItem(&lvItem); return TRUE; // 除非将焦点设置到控件,否则返回 TRUE}

LVITEM结构体的值将决定新项的功能。例如,一旦增加了一个项,你可能想要准备删除前面的项、剪切和粘贴操作,在这种情况下,你应该将设置为LVIS_CUT;如果你想实现拖放操作,你可以设置LVIS_DROPHILIGHTED;如果想让项获得焦点,设置LVIS_FOCUSED;包含LVIS_SELECTED属性的情况下可以被选中。

除了上面的CListCtrl::InsertItem()版本,CListCtrl类还提供了如下版本:

int InsertItem(int nItem, LPCTSTR lpszItem);

这是一个早期最简洁的版本。nItem表示新增项的索引。和LVITEM::iItem成员类似,如果参数为0,新增项是首项。lpszItem参数值是当前项内容的字符串。

四、实战学习:构建(Populating)ListCtrl

1.             新建一个dialog box,选择主菜单项->右击工程->增加资源

2.             在增加资源对话框中选择dialog

3.             修改对话框的ID为:IDD_STOREITEMS_DLG

4.             设计对话框如下:

Control

ID

Caption

Static Text

 

Item #:

Edit Control

IDC_ITEMNUMBER

 

5.    右击对话框,选择添加类

6.    设置类的名字为CNewStoreItemDlg基类为CDialog

7.    点击完成

8.    为edit control增加一个CString变量,命名为m_ItemNumber

9.    切换到第一个对话框,在ListCtrl下面,增加一个按钮,设置它的属性如下:

标题:新建项

ID:IDC_NEWITEM

10.  双击新建项按钮,生成它的OnBnClicked事件。

11.  在文件的前面,键入:#include "NewStoreItemDlg.h"

 

#include "stdafx.h"#include "DeptStore2.h"#include "DeptStore2Dlg.h"#include "NewStoreItemDlg.h"#ifdef _DEBUG#define new DEBUG_NEW#endif

12.   如下实现事件内容:

 

void CDeptStore2Dlg::OnBnClickedNewitem(){ // TODO: Add your control notification handler code here CNewStoreItemDlg dlg; srand( (unsigned)time(NULL) ); char strNumber[20]; int number1 = rand() % 999; int number2 = rand() % 999; sprintf(strNumber, "%d-%d", number1, number2); dlg.m_ItemNumber = strNumber; if( dlg.DoModal() ) { LVITEM lvItem; lvItem.mask = LVIF_TEXT; lvItem.iItem = 0; lvItem.iSubItem = 0; lvItem.pszText = strNumber; this->m_StoreItems.InsertItem(&lvItem); }}

 

 

13.  运行应用程序,测试新建项

 

报告方式显示(The Report View)

无论你是用第一还是第二个版本,InsertItem只允许你去创建你要显示的图标,小图标,或者控件的List Views的项。如果你计划去显示在Report View(或者允许用户在几种显示方式中转换)或者你想要提供更多的信息为每一个项,你必须要创建一个关于每一个项信息的报告。

在ListCtrl可能的显示方式中,其中有一种可以显示列。这种使用报告的方式。这种方式不需要一个列表,而是提供一个列表的项的详细信息。如果你打算在你的ListCtrl控件中显示,那么你应该去创建列。(要么,你可以省略创建列而是每个分离的列的首项,可以使用CHeaderCtrl类来实现。否则,ListCtrl提供创建列为它的报告显示)

To create the column(s) of a list control, you can use the CListCtrl::InsertColumn() method. One of its syntaxes is:

创建一个ListCtrl的列,你可以使用CListCtrl::InsertColumn()方法,它的一种声明为:

int InsertColumn(int nCol, const LVCOLUMN* pColumn);

nCol:将要创建列的索引

pColumn:指向LVCOLUMN结构体指针,结构体定义如下:

 

typedef struct _LVCOLUMN { UINT mask; int fmt; int cx; LPTSTR pszText; int cchTextMax; int iSubItem; #if (_WIN32_IE >= 0x0300) int iImage; int iOrder; #endif} LVCOLUMN, FAR *LPLVCOLUMN;

mask: 用来指定你想要的列的属性

Fmt: 格式化列的文本。例如,它可以使列左对齐(默认值LVCFMT_LEFT),居中(LVCFMT_CENTER),或者右对齐(LVCFMT_RIGHT)。如果你想为这个值设置一个值,那么在初始化mask变量的时候增加LVCF_FMT属性。

cx 用来指定列文本的宽度。如果你初始化,它的长度将被初始化为刚好能显示文本的长度。因此,除非你有一个好的理由去省略它,你应该总是指定它的值。如果你打算去初始化这个值,那么在初始化mask变量的时候增加LVCF_WIDTH属性。

pszText 是将要显示的列的字符串。和其他成员一样,此变量不是必须要初始化的,除了每一个列的第一个元素。这个成员可能是最重要的列的特征,因为它告诉用户这一列的作用。它的值是一个"\0"结尾的字符串。和其他MFC应用程序中的字符串一样,该字符串可以是一个字符串表项。它也可以从一个字符串数组中获取。为此变量初始化值,在初始化mask时应增加LVCF_TEXT变量,它的长度可以通过cchTextMax成员来设置。

初始化LVCOLUMN之后,把它作为InsertColumn的第二个参数进行传递。下面是示例:

BOOL CDeptStore2Dlg::OnInitDialog(){ CDialog::OnInitDialog(); // TODO: 在此添加额外的初始化代码 LVCOLUMN lvColumn; int nCol; lvColumn.mask = LVCF_FMT | LVCF_TEXT | LVCF_WIDTH; lvColumn.fmt = LVCFMT_LEFT; lvColumn.cx = 120; lvColumn.pszText = TEXT("Full Name"); nCol = m_StoreItems.InsertColumn(0, &lvColumn); lvColumn.mask = LVCF_FMT | LVCF_TEXT | LVCF_WIDTH; lvColumn.fmt = LVCFMT_LEFT; lvColumn.cx = 100; lvColumn.pszText = TEXT("Profession"); m_StoreItems.InsertColumn(1, &lvColumn); lvColumn.mask = LVCF_FMT | LVCF_TEXT | LVCF_WIDTH; lvColumn.fmt = LVCFMT_LEFT; lvColumn.cx = 80; lvColumn.pszText = TEXT("Fav Sport"); m_StoreItems.InsertColumn(2, &lvColumn); lvColumn.mask = LVCF_FMT | LVCF_TEXT | LVCF_WIDTH; lvColumn.fmt = LVCFMT_LEFT; lvColumn.cx = 75; lvColumn.pszText = TEXT("Hobby"); m_StoreItems.InsertColumn(3, &lvColumn); return TRUE; // 除非将焦点设置到控件,否则返回 TRUE}

(译者注:需要修改ListCtrl为Report型)

iOrder:用来标识列的地址(列的索引)

你也可以用CListCtrl::InsertColumn()方法的另一个版本,你可以用下面的版本,来为控件创建列:

 

int InsertColumn(int nCol, LPCTSTR lpszColumnHeading, int nFormat = LVCFMT_LEFT, int nWidth = -1, int nSubItem = -1);

 

这个版本是上个版本的简化版。

nCol: 列的索引,必须要指定的;

lpszColumnHeading:在列的开始要显示的字符串,和LVCOLUMN::pszText成员相似;

nFormat:用来指定lpszColumnHeading的水平对齐方式。默认为左对齐(LVCFMT_LEFT),居中为LVCFMT_CENTER,右对齐为LVCFMT_RIGHT;

nWidth:以像素为单位来设定列的宽度,如果你不想指定它的大小,可设置为-1;

nSubItem:用来设定当前列子项的索引。

列配置好之后,你必须要提供在特定列下显示的字符串项。为了实现这一功能,你必须先设定你要增加项的信息。InsertColumn()方法将会返回一个整形值来标识你增加列的信息。接着为每一列指定将要显示的项的字符串,调用CListCtrl::SetItemText()方法。它的声明为:

BOOL SetItemText(int nItem, int nSubItem, LPTSTR lpszText);

nItem:你要增加行的索引,它可以通过InsertColumn()的返回值获得。每一项的信息存储在一个以"\0"结尾的数组。

nSubItem:当前项所在的位置。(译者注:通过nItem和nSumItem确定项的位置,在平面中第nItem行,第nSumItem列)

lpszText:当前项的信息

示例:

 

BOOL CDeptStore2Dlg::OnInitDialog(){ CDialog::OnInitDialog(); // TODO: 在此添加额外的初始化代码 LVCOLUMN lvColumn; lvColumn.mask = LVCF_FMT | LVCF_TEXT | LVCF_WIDTH; lvColumn.fmt = LVCFMT_LEFT; lvColumn.cx = 120; lvColumn.pszText = TEXT("Full Name"); m_StoreItems.InsertColumn(0, &lvColumn); lvColumn.mask = LVCF_FMT | LVCF_TEXT | LVCF_WIDTH; lvColumn.fmt = LVCFMT_LEFT; lvColumn.cx = 100; lvColumn.pszText = TEXT("Profession"); m_StoreItems.InsertColumn(1, &lvColumn); lvColumn.mask = LVCF_FMT | LVCF_TEXT | LVCF_WIDTH; lvColumn.fmt = LVCFMT_LEFT; lvColumn.cx = 80; lvColumn.pszText = TEXT("Fav Sport"); m_StoreItems.InsertColumn(2, &lvColumn); lvColumn.mask = LVCF_FMT | LVCF_TEXT | LVCF_WIDTH; lvColumn.fmt = LVCFMT_LEFT; lvColumn.cx = 75; lvColumn.pszText = TEXT("Hobby"); m_StoreItems.InsertColumn(3, &lvColumn); LVITEM lvItem; int nItem; lvItem.mask = LVIF_TEXT; lvItem.iItem = 0; lvItem.iSubItem = 0; lvItem.pszText = TEXT("Sandra C. Anschwitz"); nItem = m_StoreItems.InsertItem(&lvItem); m_StoreItems.SetItemText(nItem, 1, TEXT("Singer")); m_StoreItems.SetItemText(nItem, 2, TEXT("HandBall")); m_StoreItems.SetItemText(nItem, 3, TEXT("Beach")); lvItem.mask = LVIF_TEXT; lvItem.iItem = 1; lvItem.iSubItem = 0; lvItem.pszText = TEXT("Roger A. Miller"); nItem = m_StoreItems.InsertItem(&lvItem); m_StoreItems.SetItemText(nItem, 0, TEXT("Footballer")); m_StoreItems.SetItemText(nItem, 1, TEXT("Tennis")); m_StoreItems.SetItemText(nItem, 2, TEXT("Teaching")); lvItem.mask = LVIF_TEXT; lvItem.iItem = 2; lvItem.iSubItem = 0; lvItem.pszText = TEXT("Marie-Julie W. Gross"); nItem = m_StoreItems.InsertItem(&lvItem); m_StoreItems.SetItemText(nItem, 1, TEXT("Student")); m_StoreItems.SetItemText(nItem, 2, TEXT("Boxing")); m_StoreItems.SetItemText(nItem, 3, TEXT("Programming")); lvItem.mask = LVIF_TEXT; lvItem.iItem = 3; lvItem.iSubItem = 0; lvItem.pszText = TEXT("Ella Pius Roger"); nItem = m_StoreItems.InsertItem(&lvItem); m_StoreItems.SetItemText(nItem, 1, TEXT("Architect")); m_StoreItems.SetItemText(nItem, 2, TEXT("Ping-Pong")); m_StoreItems.SetItemText(nItem, 3, TEXT("Songo")); return TRUE; // 除非将焦点设置到控件,否则返回 TRUE}

 

未完,待续……

原创粉丝点击