• 积累的VC编程小技巧之列表框


    1.列表框中标题栏(Column)的添加

    创建一个List Control,其ID为IDC_LIST,在其Styles属性项下的View项里选择Report、Align项里选择Top、Sort项里选择None.

    然后在该List所在对话框的类(头文件)里创建ClistCtrl的一个对象m_list然后在.cpp文件的OnInitDialog()之类的函数里实现如下代码:

    CString strname[3];

    strname[0]="Screen Name";

    strname[1]="Form ID";

    strname[2]="Category Path";

    for(int i=0;i<3;i++)

    {

    m_List.InsertColumn(i,strname[i],LVCFMT_LEFT,130);

    }

    在这之前也要将List Control的ID与ClistCtrl的对象m_list在DoDataExchange(CDataExchange* pDX)函数里绑定,如下:

    DDX_Control(pDX, IDC_LIST, m_List);

    2.如何防止在列表框中添加很多数据出现不停的刷新?

      [问题提出]
        在listbox添加很多数据的时候,由于控件不停的刷新,导致出现闪烁,如何解决?
      [解决方法]
        再添加数据以前,禁止控件刷新,数据添加完毕以后,再刷新一次。
      [程序实现](其中:m_ListBox是CListBox的控件类型的变量)
        m_ListBox.LockWindowUpdate();//禁止本listbox刷新。
        for(int i=0;i<9999;i++)
        {
             m_ListBox.AddString("test");
        }//添加数据。 
        this->RedrawWindow(NULL,NULL,RDW_INVALIDATE | RDW_UPDATENOW | RDW_ERASE);

    3.列表框中选择变化时如何获得通知?

    我在Report View中使用了一个CListCtrl(自绘制类型),我想知道什么时候选择项发生了改变.

    在选择项变化时,可以使用按钮有效或失效,按如下操作:

      加入LVN_ITEMCHANGED消息处理.
    void CYourClassNameHere::OnItemchangedEventList(NMHDR* pNMHDR, LRESULT* pResult)
    {
      NM_LISTVIEW* pNMListView = (NM_LISTVIEW*)pNMHDR;
      *pResult = 0;

    if (pNMListView->uChanged == LVIF_STATE)

    {
        if (pNMListView->uNewState)

    {
            GetDlgItem(IDC_DELETE)->EnableWindow(TRUE);

    }
         else

    {
            GetDlgItem(IDC_DELETE)->EnableWindow(FALSE);
        }
      }

    }

    4.列表框控件中整栏选择?

    我在处理List控件时碰到了麻烦,我想创建一个ListView,来依据Tree控件的选择同时在ListView和ReportView中显示列表的信息.以下是相关的代码:

    // Set full line select
    ListView_SetExtendedListViewStyle(m_plstCustomers->GetSafeHwnd(),
    LVS_EX_FULLROWSELECT);

    按如下方法处理:

    // -------------------- begin of snippet --------------------------------
    bool CCommCtrlUtil32::ListCtrl_ModifyExtendedStyle(CListCtrl& p_rListCtrl,
                                       const DWORD p_dwStyleEx,
                                       const bool p_bAdd)
    {
        HWND t_hWnd = p_rListCtrl.GetSafeHwnd();
        DWORD t_dwStyleEx = ListView_GetExtendedListViewStyle(t_hWnd);

        if(p_bAdd)
        {
            if(0 == (p_dwStyleEx & t_dwStyleEx))
            {
                // add style
                t_dwStyleEx |= p_dwStyleEx;
            }
        }
        else
        {
            if(0 != (p_dwStyleEx & t_dwStyleEx))
            {
                // remove style
                t_dwStyleEx &= ~p_dwStyleEx;
            }
        }

        ListView_SetExtendedListViewStyle(t_hWnd, t_dwStyleEx);

    return true;
    }

    5.如何双击列表框项启动一个与文件关联的程序?

    有人问我如何双击列表框项启动一个程序?其实这个问题很简单,Windows中有一个API函数可以打开任何类型的文件:

    ShellExecute(NULL,"open",lpFileName,NULL,NULL,SW_SHOWNORMAL);

    参数 lpFileName 是文件的全路径名。用这个变量你可以传递象“C:\MyExcelFile.xls”或者“http://www.vckbase.com”启动Excel程序或者浏览器程序。如果你只是想获取与文件关联的程序名,而不是要运行程序,那么调用::FindExecutable就可以了。

    6.如何得到列表框中所选择项的String?

      [问题提出]
      如何得到CListBox所选择项的String
      [解决方法]
      用到:CListBox::GetText()
      [程序实现]
      CString scInfo; 
      pList->GetText( GetCurSel(),scInfo);

    7.锁定ListView的栏目头宽度

    编译:NorthTibet

       世界之大,真是无其不有。Windows 应用程序的GUI标准明确规定了 ListView 栏目头(Column Header)的宽度必须是可调整的,这本来是专门为用户考虑而设计的控制特性,可是偏偏就有用户拒绝这样的特性。作为技术人员,用户的需求是很难拒绝的。尽管这明显是一种“非典型性需求”。本文将通过一个实例来示范如何实现 ListView Column Header 宽度的锁定。
        ListView 及其 Column Header 实际上都是 Windows 通用控件(Comctl32.dll) 的一部分。所以查一查 MSDN 中与“Header Control”相关的控件资料不难发现,栏目头的锁定与否与几个 Windows 的通知消息密切相关,这几个消息分别是 HDN_TRACK、HDN_BEGINTRACK 和 HDN_ENDTRACKA。其中 HDN_BEGINTRACK 是本文要特别关照的一个。当用户在栏目头上拖拽鼠标时,如果位置正好在改变宽度的分割条上,则栏目头控件会向其父窗口发送一个 HDN_BEGINTRACK 通知消息。为了实现栏目头宽度的锁定,就必须搞掂这个通知消息。不能将它传递到父窗口,但是,这个消息与 Windows 中形形色色的其它通知消息一样,有两个版本:一个版本是 HDN_BEGINTRACKW,专门用于宽字符和 Unicode 字符集;另一个版本是 HDN_BEGINTRACKA,专门用于 ANSI 字符集。这两个版本的使用方法可以从公共控件的头文件 commctrl.h 中获取:

    // From commctrl.h
    #ifdef UNICODE
    #define HDN_BEGINTRACK HDN_BEGINTRACKW
    #else
    #define HDN_BEGINTRACK HDN_BEGINTRACKA
    #endif     

        所以在实现对消息的 HDN_BEGINTRACK 处理时,实际上是根据 UNICODE 的取值实现对 HDN_BEGINTRACKA 或 HDN_BEGINTRACKW 的处理。那么 Header Control 到底是发送的哪一个消息呢?在这里必须明白:Header Control 是 Windows 通用控件的一部分,它的实现都在 comctl32.dll 动态链接库中。由于这个 DLL 已经被编译成可执行代码,因此在工程中修改 UNICODE 的设置将无济于事。如何知道栏目头控件发送哪一个版本的通知消息呢?是 A 版本还是 W 版本?
        为了找到答案,我们必须求助一个经常被遗忘的消息 WM_NOTIFYFORMAT。一般控件第一次被创建时,都要向父窗口一个消息询问父窗口需要哪个版本的通知消息。然后父窗口返回 NFR_ANSI 或 NFR_UNICODE。如果父窗口不处理 WM_NOTIFYFORMAT,那么这个消息将根据父窗口或对话框本身的首选项被传递到 Windows 的 DefWindowProc 消息处理例程进行默认处理。默认为 UNICODE。因此,要知道通知消息的版本,必须处理 ListCtrl 的 WM_NOTIFYFORMAT。为了确认父窗口的返回值,你可以做一个试验便明白了。
        如果你不想处理 WM_NOTIFYFORMAT 消息,那么完全可以通过双双实现 HDN_BEGINTRACKA 和 HDN_BEGINTRACKW 通知消息的处理来简化问题的解决方案,同时这种方法也更可靠和通用。此时代码将同时支持 ANSI 和 Unicode。本文附带的例子程序示范了这种方法的实现。如图一所示:

    图一 锁定栏目头宽度

        实现代码很简单,Header 控件发送 HDN_XXX 到父窗口(ListCtrl),在 MFC 中可以利用消息反射来处理 Header 控件的通知消息。因为“可锁定栏目头”特性本身更趋向于 Header 控件的属性,而不是 ListCtrl 的属性。如果你不用 MFC ,那么就得处理 ListCtrl 中的通知消息。例子程序使用了消息反射机制,在 Header 控件的消息映射使用 ON_NOTIFY_REFLECT,也就是该写虚拟成员函数 OnChildNotify:

    BOOL CLockableHeader::OnChildNotify(UINT msg, WPARAM wp, LPARAM lp, LRESULT* pRes)
    {
         NMHDR& nmh = *(NMHDR*)lp;
         if (nmh.code==HDN_BEGINTRACKW || nmg.code==HDN_BEGINTRACKA)
             return *pRes=TRUE;
         ......
    }

        因为 OnChildNotify 是虚函数,所以没有必要具备消息映射入口。只要实现此函数即可。在任何应用中,Header 发送的消息非此即彼,不会两者都发送。不管怎样,所发送的通知消息在到达父窗口之前都会被吃掉。也就是说,消息处理总是返回 TRUE,是否锁定栏目头的宽度通过一个标志来控制:应用程序通过 Lock 来修改标志的值。
    如果锁定了头宽度,那么同时也必须禁用改变宽度的光标,这样用户界面才会有一致性,要实现这一点也很简单:

    BOOL CLockableHeader::OnSetCursor( CWnd* pWnd, UINT nHit, UINT msg)
    {
         return m_bLocked ? TRUE : CHeaderCtrl::OnSetCursor(pWnd, nHit, msg);
    }      

        如果栏目头被锁定,则 OnSetCursor 返回 TRUE,此时光标不会被重新设置,否则由 Header 控件的进行默认处理。锁定宽度后,当鼠标移到栏目头上时,Windows 显示标准的箭头光标,而不是带左右箭头光标。
        从 CHeaderCtrl 派生类出来的类的使用方法与处理对话框控制一样,通过在父窗口的 OnCreate 的处理例程中进行子类化。实现细节请参考例子源代码:

         
    // CMyView is derived from CListView
    int CMyView::OnCreate(LPCREATESTRUCT lpcs)
    {
      VERIFY(CListView::OnCreate(lpcs)==0);
      return m_header.SubclassDlgItem(0,this) ? 0 : -1;
    }

        由于 Header 控制的资源 ID = 0,所以上面的代码是行得通的。为了有一个友好的用户界面,例子程序创建了一个命令菜单和界面更新处理例程。如图一所示。

  • 相关阅读:
    Git上手:四种常见的Git协同工作方式
    Git上手:Git扫盲区
    理解web缓存
    浅谈对技术债的理解
    保护个人隐私,从我做起
    cookie注意事项
    了解JavaScript核心精髓(二)
    简单实现nodejs爬虫工具
    浅谈我所见的CSS组织风格
    利用正则表达式清除多余的空行
  • 原文地址:https://www.cnblogs.com/lidabo/p/3703492.html
Copyright © 2020-2023  润新知