自动过滤选择的组合框

发表于:2007-07-01来源:作者:点击数: 标签:
经过两天的编码调试,完成了一个自动过滤选择的组合框,具有以下优点: 1。丝毫不妨碍组合框原有的功能 2。根据用户在编辑框里已经输入的单词,从组合框的列表选项过滤出匹配的项,并使用下拉列表显示出来以供用户选择。 3。不需要特别的设置。从View\ClassW

经过两天的编码调试,完成了一个自动过滤选择的组合框,具有以下优点:
1。丝毫不妨碍组合框原有的功能
2。根据用户在编辑框里已经输入的单词,从组合框的列表选项过滤出匹配的项,并使用下拉列表显示出来以供用户选择。
3。不需要特别的设置。从View\ClassWizard\Member Variables里生成一个CComboBox,然后在源文件里把CComboBox替换成CACComboBox即可;或者让ClassWizard认识CACComboBox(删除MyApp.clw,选View\ClassWizard,按提示重新生成MyApp.clw,绝大多数时候能成功),从View\ClassWizard\Member Variables里直接生成一个CACComboBox。
4。你可以经过简单修改使之成为具有其他类似功能(自动填写??)的组合框。

由于时间仓促,没有写详细的注释。感兴趣的朋友请等几天后向我索要。联系方法:

代码如下:


// File: ACComboBox.h
//
// Desc:
//   根据用户在编辑框里已经输入的单词,从组合框的列表选项过滤出匹配的项,并使用下拉列表显示出来以供用户选择
//
// Copyright (c) 2001 EagleFly Studio.
//
// Original Author: Zhengpeng.Lan
// Author:
//
// Create Time:  2001/10/10
// Modify Time:  2001/10/11
//

#if !defined(AFX_ACCOMBOBOX_H__81CBBD04_3955_4076_A688_74D3EA9730D9__INCLUDED_)
#define AFX_ACCOMBOBOX_H__81CBBD04_3955_4076_A688_74D3EA9730D9__INCLUDED_

#if _MSC_VER > 1000
#pragma once
#endif // _MSC_VER > 1000

#include "afxtempl.h"

// ACComboBox.h : header file
//
class CACComboBox;
/////////////////////////////////////////////////////////////////////////////
// CWndForACComboBox window
typedef struct
{
 int nid;
 CString strDepict;
}CInfoForACComboBox;

typedef CArray<CInfoForACComboBox,CInfoForACComboBox &>  CInfoForACComboBoxArray;

class CWndForACComboBox : public CWnd
{
 friend CACComboBox;
private:
// Construction
 LPCTSTR lpWndCls;
// Construction
public:
 CWndForACComboBox();

// Attributes
public:
 CComboBox * m_pComboBox;
 int   m_nMaxCount;
 CInfoForACComboBoxArray   m_aInfo;
 int   m_nMousePos;
 int   m_nBeginShow;

 inline void RemoveAll()
 {
  m_aInfo.RemoveAll();
  CWnd::SetScrollRange(SB_VERT,0,0);
 }
 inline void Add(CInfoForACComboBox &info)
 {
  m_aInfo.Add(info);
 }
 inline int GetSize()
 {
  return m_aInfo.GetSize();
 }
 inline void ShowList()
 {
  if(IsWindow(GetSafeHwnd()))
  {
   ShowWindow(SW_SHOWNOACTIVATE);
  }
 }
// Operations
public:

// Overrides
 // ClassWizard generated virtual function overrides
 AFX_VIRTUAL(CWndForACComboBox)
 public:
 virtual BOOL Create(CComboBox * pWnd);
 AFX_VIRTUAL

// Implementation
public:
 virtual ~CWndForACComboBox();

 // Generated message map functions
protected:
 AFX_MSG(CWndForACComboBox)
 afx_msg void OnDestroy();
 afx_msg void OnPaint();
 afx_msg void OnKeyDown(UINT nChar, UINT nRepCnt, UINT nFlags);
 afx_msg void OnLButtonUp(UINT nFlags, CPoint point);
 afx_msg BOOL OnMouseWheel(UINT nFlags, short zDelta, CPoint pt);
 afx_msg void OnKillFocus(CWnd* pNewWnd);
 afx_msg void OnMouseMove(UINT nFlags, CPoint point);
 afx_msg void OnVScroll(UINT nSBCode, UINT nPos, CScrollBar* pScrollBar);
 AFX_MSG
 DECLARE_MESSAGE_MAP()
};


/////////////////////////////////////////////////////////////////////////////
// CACComboBox window

class CACComboBox : public CComboBox
{
private:
 CEdit*      m_pEdit;

// Construction
public:
 CACComboBox();

// Attributes
public:
 CWndForACComboBox  m_wndList;
 inline void WindowMove(void)
 {
  if(IsWindow(m_wndList.GetSafeHwnd()))
   m_wndList.Invalidate(FALSE);
 }
// Operations
public:
  virtual void HandleCompletion();

// Overrides
 // ClassWizard generated virtual function overrides
 AFX_VIRTUAL(CACComboBox)
 protected:
 virtual BOOL OnCommand(WPARAM wParam, LPARAM lParam);
 virtual void PreSubclassWindow();
 AFX_VIRTUAL

// Implementation
public:
 virtual ~CACComboBox();

 // Generated message map functions
protected:
 AFX_MSG(CACComboBox)
 afx_msg void OnKillfocus();
 afx_msg void OnSelchange();
 AFX_MSG

 DECLARE_MESSAGE_MAP()
};

/////////////////////////////////////////////////////////////////////////////

/////////////////////////////////////////////////////////////////////////////
AFX_INSERT_LOCATION}}
// Microsoft Visual C++ will insert additional declarations immediately before the previous line.

#endif // !defined(AFX_ACCOMBOBOX_H__81CBBD04_3955_4076_A688_74D3EA9730D9__INCLUDED_)



// File: ACComboBox.cpp
//
// Desc:
//   根据用户在编辑框里已经输入的单词,从组合框的列表选项过滤出匹配的项,并使用下拉列表显示出来以供用户选择
//
// Copyright (c) 2001 EagleFly Studio.
//
// Original Author: Zhengpeng.Lan
// Author:
//
// Create Time:  2001/10/10
// Modify Time:  2001/10/11
//

#include "stdafx.h"
#include "ACComboBox.h"

#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#endif

/////////////////////////////////////////////////////////////////////////////
// CACComboBox

CACComboBox::CACComboBox()
:CComboBox()
{
 m_pEdit = NULL;
}

CACComboBox::~CACComboBox()
{
 if(m_pEdit)
  delete m_pEdit;
}


BEGIN_MESSAGE_MAP(CACComboBox, CComboBox)
 AFX_MSG_MAP(CACComboBox)
 ON_CONTROL_REFLECT(CBN_KILLFOCUS, OnKillfocus)
 ON_CONTROL_REFLECT(CBN_SELCHANGE, OnSelchange)
 AFX_MSG_MAP
END_MESSAGE_MAP()

/////////////////////////////////////////////////////////////////////////////
// CACComboBox message handlers
/////////////////////////////////////////////////////////////////////////////
// CWndForACComboBox

CWndForACComboBox::CWndForACComboBox()
:CWnd()
{
 lpWndCls = AfxRegisterWndClass(0);
 m_pComboBox = NULL;
 m_nMaxCount = 8;
 m_aInfo.SetSize(0,8);
 m_nMousePos = -1;
 m_nBeginShow = 0;
}

CWndForACComboBox::~CWndForACComboBox()
{

}


BEGIN_MESSAGE_MAP(CWndForACComboBox, CWnd)
 AFX_MSG_MAP(CWndForACComboBox)
 ON_WM_DESTROY()
 ON_WM_PAINT()
 ON_WM_KEYDOWN()
 ON_WM_LBUTTONUP()
 ON_WM_MOUSEWHEEL()
 ON_WM_KILLFOCUS()
 ON_WM_MOUSEMOVE()
 ON_WM_VSCROLL()
 AFX_MSG_MAP
END_MESSAGE_MAP()


/////////////////////////////////////////////////////////////////////////////
// CWndForACComboBox message handlers

BOOL CWndForACComboBox::Create(CComboBox * pWnd)
{
 ASSERT(pWnd);

 CRect rect;
 CWnd * pParentWnd = pWnd->GetParent();

 pWnd->GetWindowRect(&rect);
// pParentWnd->ScreenToClient(&rect);

 BOOL bRet = CWnd::CreateEx( WS_EX_TOPMOST | WS_EX_TOOLWINDOW,\
  lpWndCls, NULL, WS_POPUP | WS_VSCROLL, rect.left, rect.top, rect.Width(), rect.Height(),
  pParentWnd->GetSafeHwnd(), NULL, NULL);

 if(bRet)
  SetOwner(pParentWnd);

 m_pComboBox = pWnd;
 return bRet;
}

void CWndForACComboBox::OnDestroy()
{
 CWnd::OnDestroy();
 
 m_aInfo.RemoveAll(); 
}

void CWndForACComboBox::OnPaint()
{
 CRect rect;
 CFont * pFont;
 LOGFONT logFont;

 if(m_pComboBox)
 {
  ASSERT(::IsWindow(m_pComboBox->GetSafeHwnd()));
 
  CWnd * pParentWnd = m_pComboBox->GetParent();
  m_pComboBox->GetWindowRect(&rect);
  pFont = pParentWnd->GetFont();
  ASSERT(pFont);
  pFont->GetLogFont(&logFont);

  if(logFont.lfHeight < 0) logFont.lfHeight = -logFont.lfHeight;
  if(m_aInfo.GetSize() > m_nMaxCount)
  {
   logFont.lfWidth = m_nMaxCount * logFont.lfHeight;
  }
  else
  {
   logFont.lfWidth = m_aInfo.GetSize() * logFont.lfHeight;
  }
  if(logFont.lfWidth + rect.bottom > ::GetSystemMetrics(SM_CYSCREEN))
  {
   rect.bottom = rect.top;
   rect.top = rect.bottom - logFont.lfWidth - 2;
   if(rect.top < 0)
    rect.top = 0;
  }
  else
  {
   rect.top = rect.bottom;
   rect.bottom = rect.top + logFont.lfWidth + 2;
  }
  rect.right = rect.left + m_pComboBox->GetDroppedWidth();
//  pParentWnd->ScreenToClient(&rect);
  MoveWindow(&rect);
  if(rect.Height() < m_aInfo.GetSize() * logFont.lfHeight)
  {
   TRACE("%d   %d\r\n",rect.Height(),m_aInfo.GetSize() * logFont.lfHeight);
   SCROLLINFO info;

   info.cbSize = sizeof(info);
   info.fMask = SIF_PAGE;
   info.nPage = rect.Height() / logFont.lfHeight;
   CWnd::SetScrollRange(SB_VERT,0,m_aInfo.GetSize() - 1,FALSE);
   CWnd::ShowScrollBar(SB_VERT,TRUE);
   CWnd::SetScrollInfo(SB_VERT,&info);
  }
  else
  {
   CWnd::ShowScrollBar(SB_VERT,FALSE);
  }
 }
 CPaintDC dc(this); // device context for painting

 (0,0,rect.Width(),rect.Height(),RGB(255,255,255));
 dc.Rectangle(0,0,rect.Width(),rect.Height());
 logFont.lfWidth = 1;
 dc.SelectObject(pFont);
 int m_nBeginShow = CWnd::GetScrollPos(SB_VERT);
 for(int i=m_nBeginShow;i<m_aInfo.GetSize() && logFont.lfWidth < rect.Height() - 2;i++)
 {
  if(i == m_nMousePos)
  {
   dc.SetBkColor(RGB(0,0,128));
   dc.FillSolidRect(1,logFont.lfWidth,rect.Width() - 2,logFont.lfHeight,RGB(0,0,128));
   dc.SetTextColor(RGB(255,255,255));
  }
  else
  {
   dc.SetBkColor(RGB(255,255,255));
   dc.SetTextColor(RGB(0,0,0));
  }
  dc.TextOut(3,logFont.lfWidth,m_aInfo[i].strDepict);
  logFont.lfWidth += logFont.lfHeight;
 }
}

void CWndForACComboBox::OnKeyDown(UINT nChar, UINT nRepCnt, UINT nFlags)
{
 // TODO: Add your message handler code here and/or call default
 
 CWnd::OnKeyDown(nChar, nRepCnt, nFlags);
}

void CWndForACComboBox::OnLButtonUp(UINT nFlags, CPoint point)
{
 CFont * pFont;
 LOGFONT logFont;

 if(m_pComboBox)
 {
  pFont = m_pComboBox->GetParent()->GetFont();
  pFont->GetLogFont(&logFont);
  if(logFont.lfHeight < 0) logFont.lfHeight = -logFont.lfHeight;
  m_nMousePos = (point.y - 1) / logFont.lfHeight + CWnd::GetScrollPos(SB_VERT);
  if(m_nMousePos >= 0 && m_nMousePos < m_aInfo.GetSize())
   m_pComboBox->SetCurSel(m_aInfo[m_nMousePos].nid);
  ShowWindow(SW_HIDE);
 }
 
 CWnd::OnLButtonUp(nFlags, point);
}

BOOL CWndForACComboBox::OnMouseWheel(UINT nFlags, short zDelta, CPoint pt)
{
 // TODO: Add your message handler code here and/or call default
 
 return CWnd::OnMouseWheel(nFlags, zDelta, pt);
}

void CWndForACComboBox::OnKillFocus(CWnd* pNewWnd)
{
 CWnd::OnKillFocus(pNewWnd);
 
 if(pNewWnd->GetSafeHwnd() != m_pComboBox->GetSafeHwnd() && pNewWnd->GetSafeHwnd() != GetSafeHwnd() && IsWindow(GetSafeHwnd()))
  ShowWindow(SW_HIDE);
}

void CWndForACComboBox::OnMouseMove(UINT nFlags, CPoint point)
{
 CFont * pFont;
 LOGFONT logFont;
 ::ShowCursor(TRUE);
 if(m_pComboBox)
 {
  pFont = m_pComboBox->GetParent()->GetFont();
  pFont->GetLogFont(&logFont);
  if(logFont.lfHeight < 0) logFont.lfHeight = -logFont.lfHeight;
  logFont.lfWidth = point.y / logFont.lfHeight + CWnd::GetScrollPos(SB_VERT);
  if(logFont.lfWidth != m_nMousePos)
  {
   m_nMousePos = logFont.lfWidth;
   Invalidate();
  }
 }
 
 CWnd::OnMouseMove(nFlags, point);
}

void CWndForACComboBox::OnVScroll(UINT nSBCode, UINT nPos, CScrollBar* pScrollBar)
{
 if(m_aInfo.GetSize() <= 0)
 {
  CWnd::OnVScroll(nSBCode, nPos, pScrollBar);
  return ;
 }

 switch(nSBCode)
 {
// case SB_BOTTOM://Scroll to bottom.
// case SB_ENDSCROLL://End scroll.
 case SB_LINEDOWN://Scroll one line down.
  {
   RECT rc;
   LOGFONT logFont;
   CFont * pFont;

   GetClientRect(&rc);
   pFont = m_pComboBox->GetParent()->GetFont();
   pFont->GetLogFont(&logFont);
   ASSERT(logFont.lfHeight != 0);
   if(logFont.lfHeight < 0) logFont.lfHeight = -logFont.lfHeight;
   rc.left = (rc.bottom - rc.top) / logFont.lfHeight;

   int nPos = GetScrollPos(SB_VERT) + 1;
   if(nPos + rc.left <= m_aInfo.GetSize())
   {
    SetScrollPos(SB_VERT,nPos);
    Invalidate();
   }
  }
  break;
 case SB_LINEUP://Scroll one line up.
  {
   int nPos = GetScrollPos(SB_VERT) - 1;
   if(nPos >= 0)
   {
    SetScrollPos(SB_VERT,nPos);
    Invalidate();
   }
  }
  break;
 case SB_PAGEDOWN://Scroll one page down.
  {
   RECT rc;
   LOGFONT logFont;
   CFont * pFont;

   GetClientRect(&rc);
   pFont = m_pComboBox->GetParent()->GetFont();
   pFont->GetLogFont(&logFont);
   ASSERT(logFont.lfHeight != 0);
   if(logFont.lfHeight < 0) logFont.lfHeight = -logFont.lfHeight;

   rc.top = rc.bottom / logFont.lfHeight;
   int nPos = GetScrollPos(SB_VERT);
   if(nPos < m_aInfo.GetSize() - rc.top)
   {
    nPos += rc.top;
    if(nPos > m_aInfo.GetSize() - rc.top) nPos = m_aInfo.GetSize() - rc.top;
    SetScrollPos(SB_VERT,nPos);
    Invalidate();
   }
  }
  break;
 case SB_PAGEUP://Scroll one page up.
  {
   RECT rc;
   LOGFONT logFont;
   CFont * pFont;

   GetClientRect(&rc);
   pFont = m_pComboBox->GetParent()->GetFont();
   pFont->GetLogFont(&logFont);
   ASSERT(logFont.lfHeight != 0);
   if(logFont.lfHeight < 0) logFont.lfHeight = -logFont.lfHeight;

   rc.top = rc.bottom / logFont.lfHeight;
   int nPos = GetScrollPos(SB_VERT);
   if(nPos > 0)
   {
    nPos -= rc.top;
    if(nPos < 0) nPos = 0;
    SetScrollPos(SB_VERT,nPos);
    Invalidate();
   }
  }
  break;
// case SB_THUMBPOSITION://Scroll to the absolute position. The current position is provided in nPos.
 case SB_THUMBTRACK://Drag scroll box to specified position. The current position is provided in nPos.
  {
   SetScrollPos(SB_VERT,nPos);
   Invalidate();
  }
  break;
// case SB_TOP://Scroll to top  
 }
 
 CWnd::OnVScroll(nSBCode, nPos, pScrollBar);
}

void CACComboBox::HandleCompletion()
{
 // Make sure we can ´talk´ to the edit control
 if( m_pEdit == NULL ) 
 {
  m_pEdit = new CEdit();
  m_pEdit->SubclassWindow(GetDlgItem(1001)->GetSafeHwnd());
 }

 // Save the state of the edit control
 CString text,bestText;
 CInfoForACComboBox info;
 int n = GetCount();

 m_wndList.RemoveAll();
 m_pEdit->GetWindowText(text);
 CPoint ptCaret = m_pEdit->GetCaretPos();

// int start,end;
// m_pEdit->GetSel(start,end);

 // Perform actual completion
 int bestindex = -1;
 int bestfrom  = INT_MAX;
 int from;
 for ( int x = 0; x < n; x++ )
 {
  GetLBText(x,info.strDepict);
  info.nid = x;

  from = info.strDepict.Find(text);

  if(from != -1)
  {
   m_wndList.Add(info);
   if( from != -1 && from < bestfrom )
   {
    bestindex = x;
    bestText = info.strDepict;
    bestfrom = from;
   }
  }
 }

 if ( bestindex != -1)
 {
  // Restore the edit control
  m_pEdit->SetWindowText(bestText);
  m_pEdit->SetSel(text.GetLength(),bestText.GetLength());
  m_pEdit->SetCaretPos(ptCaret);
 }
 if(m_wndList.GetSize() != 0)
 {
  m_wndList.ShowList();
  m_wndList.Invalidate();
 }
 else
 {
  m_wndList.ShowWindow(SW_HIDE);
 }
}

BOOL CACComboBox::OnCommand(WPARAM wParam, LPARAM lParam)
{
 if(HIWORD(wParam) == EN_CHANGE)
 {
  HandleCompletion();
  return TRUE;
 }
  
 return CComboBox::OnCommand(wParam, lParam);
}

void CACComboBox::PreSubclassWindow()
{
 CComboBox::PreSubclassWindow();

 m_wndList.Create(this);
}

void CACComboBox::OnKillfocus()
{
 m_wndList.OnKillFocus(GetFocus());
}

void CACComboBox::OnSelchange()
{
 m_wndList.ShowWindow(SW_HIDE); 
}


原文转自:http://www.ltesting.net