Listcontrol用法

ListControl 控件可在窗体中管理和显示列 表项。可控制列表内容的显示方式,能够以图标和表格的形式显示数据。打开ListControl控件的属性窗口,在Styles选项卡中的View属性中 可以设置显示风格,包括:Icon表示图标视图;Small Icon表示小图标视图;List表示列表视图;Report表示报表视图。

0283 为ListControl控件设置列标题

在使用 ListControl控件Report显示风格时,需要设置列标题信息,否则不能向控件中添加数据信息,编辑列标题需要使用InsertColumn方 法。该方法的语法如下:

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

参数说明如下。

l     nCol:标识新列的索引。

l     lpszColumnHeading:标识列标题。

l     nFormat:标识列的对齐方式。

l     nWidth:标识列宽度。

l     nSubItem:标识关联当前列的子视图项索引。

程序代码如下:

 

m_Grid.InsertColumn(0,"姓名",LVCFMT_LEFT,150,0);                 //添加列
    m_Grid.InsertColumn(1,"联系电话",LVCFMT_LEFT,150,1);         //添加列

0284 为ListControl控件添加行

在ListControl 控件中添加信息时不能直接向控件中添加列信息,需要先为控件添加行,使用InsertItem方法,该方法用于向ListControl控件中添加行。语 法如下:

int InsertItem( int nItem, LPCTSTR lpszItem );

参数说明如下。

l     nItem:表示被插入的行索引。

l     lpszItem:表示行文本。

程序代码如下:

m_Grid.InsertItem(0,"");
0285 为ListControl控件添加列

通过 SetItemText方法可以为任意行的任意列添加数据,该方法用于设置ListControl控件中的文本。语法如下:

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

参数说明如下。

l     nItem:标识行索引。

l     nSubItem:标识列索引。

l     lpszText:标识设置的显示文本。

程序代码如下:

    m_Grid.SetItemText(0,0,"周X");

    m_Grid.SetItemText(0,1,"12345XXXXXX");

0286 设置ListControl控件的扩展风格

在使用 ListControl控件的Report显示风格时,需要设置一些常用的扩展风格。如图5.33所示。

图5.33 设置ListControl控件的扩展风格

程序代码如下:

SetExtendedStyle(

        LVS_EX_FLATSB               //扁平风格滚动条

        |LVS_EX_FULLROWSELECT     //允许整行选中

        |LVS_EX_HEADERDRAGDROP    //允许标题拖曳

        |LVS_EX_ONECLICKACTIVATE //高亮显示

        |LVS_EX_GRIDLINES          //画出网格线

        );

    m_Grid.InsertColumn(0,"姓名",LVCFMT_LEFT,150,0);

    m_Grid.InsertColumn(1,"联系电话",LVCFMT_LEFT,150,1);

    m_Grid.InsertItem(0,"");

    m_Grid.SetItemText(0,0,"周X");

    m_Grid.SetItemText(0,1,"12345XXXXXX");

    m_Grid.InsertItem(1,"");

    m_Grid.SetItemText(1,0,"诸葛X");

    m_Grid.SetItemText(1,1,"67890XXXXXX");

0287 设置ListControl控件数据的排列顺序

通过 ListControl控件的属性窗口可以设置控件中数据的排列顺序,在Styles选项卡中的Sort属性中可以对控件中的数据进行排序,包 括:None表示不进行排序;Ascending表示升序排列;Decending表示降序排列。

0288 单击ListControl控件列标题进行排序

在使用 ListControl控件的Report显示风格时,要实现单击列标题进行排序需要在控件的LVN_COLUMNCLICK消息的处理函数中添加 SortItems函数,SortItems函数实现了对列表项排序的功能,语法如下:

BOOL SortItems( PFNLVCOMPARE pfnCompare, DWORD dwData );

参数说明如下。

l     pfnCompare:排序时所使用的回调函数,如果不进行排序就让回调函数返回-1。

l     dwData:指定列表控件对象指针。

程序代码如下:

void CCompositorDlg::OnColumnclickList1(NMHDR* pNMHDR, LRESULT* pResult)
{
    NM_LISTVIEW* pNMListView = (NM_LISTVIEW*)pNMHDR;
    m_Grid.SortItems(Compositor,(DWORD)reinterpret_cast<DWORD>(this));
    *pResult = 0;
}
int CALLBACK CCompositorDlg::Compositor(LPARAM lParam1, LPARAM lParam2, LPARAM lParamSort)
{
    CListCtrl* pListCtrl=reinterpret_cast<CListCtrl*>(lParamSort);   
    return 1;
}

0289 具有热点效果的视图控件

在使用列表视图控件时,如果列表视图以表格形式显示数据,并且包含有 LVS_EX_TWOCLICKACTIVATE扩展风格,那么,当鼠标移动到某一行时,整行数据会出现热点效果。如何将某一个单元格设置为热点效果呢? 本例实现了该功能,效果如图5.34所示。

本例是通过绘制列表 视图控 件实现的。首先从CListCtrl派生一个子类,然后添加 ON_NOTIFY_REFLECT_EX(NM_CUSTOMDRAW,OnCustomDraw)消息映射实现CListCtrl的绘制。最后处理鼠 标移动时的事件,判断鼠标在移动过程中,单元格是否发生了改变,如果发生了改变,重绘窗口。程序主要代码如下:

BOOL CCustomView::OnCustomDraw(NMHDR *pNotifyStruct, LRESULT *lResult)
{
    NMLVCUSTOMDRAW * nmDraw = (NMLVCUSTOMDRAW *)pNotifyStruct;
    NMCUSTOMDRAW & pDraw = nmDraw->nmcd;
        UINT flag = pDraw.uItemState;
    switch (pDraw.dwDrawStage)
   {
    case CDDS_PREPAINT :
        {
            *lResult =CDRF_NOTIFYITEMDRAW;
            break;
        }
    case CDDS_ITEMPREPAINT :
        {
            *lResult =CDRF_NOTIFYSUBITEMDRAW ;
            break;   
        }
    case (CDDS_ITEMPREPAINT | CDDS_SUBITEM) :
        {
            int row = pDraw.dwItemSpec;
            int col = nmDraw->iSubItem;
            CPoint point;
            GetCursorPos(&point);
            ScreenToClient(&point);
            CRect rect,secRC;
            this->GetSubItemRect(row,col,LVIR_BOUNDS ,rect);
           
            int cWidth = this->GetColumnWidth(0); //获取第一列的宽度
            nmDraw->clrText = m_FontColor;
            if (col !=0)
            {
                if (rect.PtInRect(point))
                     nmDraw->clrText = m_HotFontColor;   
            }
            else
            {
                secRC.CopyRect(CRect(0,rect.top,cWidth,rect.bottom));   
                if (secRC.PtInRect(point))
                    nmDraw->clrText = m_HotFontColor;               
            }
           
            break;       
        }
    default:
        {
            *lResult = CDRF_DODEFAULT;
            break;
        }
    }
    return TRUE;
}
void CCustomView::OnMouseMove(UINT nFlags, CPoint point)
{   
    CListCtrl::OnMouseMove(nFlags, point);
    LVHITTESTINFO pos;
    pos.pt = point;
    pos.flags = LVHT_ABOVE;
   
    int curcol = -1;
    int currow = -1;
    if (this->SubItemHitTest(&pos)>=0)
    {       
        curcol = pos.iSubItem;
        currow = pos.iItem;
        if (m_PreCol ==-1)
            m_PreCol = curcol;
        if (m_PreRow== -1)
            m_PreRow = currow;
        if (currow==0 && m_Valid == FALSE )
        {
            m_Valid = TRUE;
            CRect rect;
             GetSubItemRect(pos.iItem,pos.iSubItem,LVIR_BOUNDS ,rect);
            int cWidth = rect.right;
            if (curcol==0)
            {
                cWidth = this->GetColumnWidth(0); //获取第一列的宽度
            }
            RedrawWindow(CRect(rect.left,rect.top,cWidth,rect.bottom));   
            GetSubItemRect(m_PreRow,m_PreCol,LVIR_BOUNDS ,rect);
            cWidth = rect.right;
            if (m_PreCol==0)
            {
                cWidth = this->GetColumnWidth(0); //获取第一列的宽度
            }
            RedrawWindow(CRect(rect.left,rect.top,cWidth,rect.bottom));           
        }
        else if (m_PreCol != curcol ||(m_PreRow != currow)) //用户移动鼠标时列发生了改变//或行发生了改变
        {
            CRect rect;
            GetSubItemRect(pos.iItem,pos.iSubItem,LVIR_BOUNDS ,rect);
            int cWidth = rect.right;
            if (curcol==0)
            {
                cWidth = this->GetColumnWidth(0); //获取第一列的宽度
            }
            RedrawWindow(CRect(rect.left,rect.top,cWidth,rect.bottom));   
            GetSubItemRect(m_PreRow,m_PreCol,LVIR_BOUNDS ,rect);
            cWidth = rect.right;
            if (m_PreCol==0)
            {
                cWidth = this->GetColumnWidth(0); //获取第一列的宽度
            }
            RedrawWindow(CRect(rect.left,rect.top,cWidth,rect.bottom));   
        }
        m_PreCol = curcol;
        m_PreRow = currow;
    }

}

0290 具有背景的列表视图控件

在使用列表视图控件 时,默认情况下,列表视图控件具有白色的背景,如果能够在列表视图中以图片为背景,界面效果会好很多。本例实现了一个具有背景的列表视图控件,如图 5.35所示。

图5.35 具有背景的列表视图控件

有些用户可能认为只 要从 CListCtrl派生一个子类,然后在WM_PAINT消息处理函数中绘制一幅图片就可以了。但是,这样会导致列表视图中的数据被背景图片覆盖。其实, 实现具有背景的列表视图控件并不复杂,首先在程序初始化时调用AfxOleInit()函数初始化Com;然后调用CListCtrl的 SetBkImage方法设置背景位图;最后调用SetTextBkColor方法将文本背景透明。程序主要代码如下:

m_List.SetBkImage("c:\\background2.bmp");
    m_List.SetTextBkColor(CLR_NONE);