Click here to Skip to main content
Click here to Skip to main content

Implementing an autocompleting Combobox

By , 18 Oct 2000
 

ComboCompletion.gif

Introduction

I had a need for a combobox that would auto-complete, very much like the URL edit box in the toolbar of Netscape Navigator. It was actually surprisingly simple since the base CComboBox is so rich in functionality.

The basic idea is that every time the text in the edit box changes, check to see if there is any text in the drop down list that is prefixed by this edit box text. Handle the CBN_EDITUPDATE message to get the text change notifications, and use GetWindowText() to get the text. CComboBox::SelectString will look for a string in the list which is prefixed by the given string, and select it into the edit box. I then select the portion of text that was added to the users typed text so that they can continue typing and have the additions ignored if they wish. That takes care of 90% of the work.

The only trick is in handling backspaces and deletes. When a user hits delete, the text is changed, and the auto-completion routine will try to restore that text back again. Just check in PreTranslateMessage for a KEY_DOWN message with a virtual key of VK_DELETE or VK_BACK, and temporarily disable the auto-complete mechanism for those key strokes.

Source code

#if !defined(AFX_ComboCompletion_H__115F422E_5CD5_11D1_ABBA_00A02__INCLUDED_)
#define AFX_ComboCompletion_H__115F422E_5CD5_11D1_ABBA_00A02__INCLUDED_

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

// ComboCompletion.h : header file
//
// Copyright (c) Chris Maunder 1997.
// Please feel free to use and distribute.


/////////////////////////////////////////////////////////////////////////////
// CComboCompletion window

class CComboCompletion : public CComboBox
{
// Construction
public:
    CComboCompletion();

// Attributes
public:

// Operations
public:

// Overrides
    // ClassWizard generated virtual function overrides
    //{{AFX_VIRTUAL(CComboCompletion)
    public:
    virtual BOOL PreTranslateMessage(MSG* pMsg);
    //}}AFX_VIRTUAL

// Implementation
public:
    virtual ~CComboCompletion();

    BOOL m_bAutoComplete;

    // Generated message map functions
protected:
    //{{AFX_MSG(CComboCompletion)
    afx_msg void OnEditUpdate();
    //}}AFX_MSG

    DECLARE_MESSAGE_MAP()
};

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

//{{AFX_INSERT_LOCATION}}
// Microsoft Developer Studio will insert additional declarations immediately 
// before the previous line.

#endif 



// ComboCompletion.cpp : implementation file
//
// Copyright (c) Chris Maunder 1997.
// Please feel free to use and distribute.

#include "stdafx.h"
#include "ComboCompletion.h"

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

/////////////////////////////////////////////////////////////////////////////
// CComboCompletion

CComboCompletion::CComboCompletion()
{
    m_bAutoComplete = TRUE;
}

CComboCompletion::~CComboCompletion()
{
}


BEGIN_MESSAGE_MAP(CComboCompletion, CComboBox)
    //{{AFX_MSG_MAP(CComboCompletion)
    ON_CONTROL_REFLECT(CBN_EDITUPDATE, OnEditUpdate)
    //}}AFX_MSG_MAP
END_MESSAGE_MAP()

/////////////////////////////////////////////////////////////////////////////
// CComboCompletion message handlers

BOOL CComboCompletion::PreTranslateMessage(MSG* pMsg)
{
    // Need to check for backspace/delete. These will modify the text in
    // the edit box, causing the auto complete to just add back the text
    // the user has just tried to delete. 

    if (pMsg->message == WM_KEYDOWN)
    {
        m_bAutoComplete = TRUE;

        int nVirtKey = (int) pMsg->wParam;
        if (nVirtKey == VK_DELETE || nVirtKey == VK_BACK)
            m_bAutoComplete = FALSE;
    }

    return CComboBox::PreTranslateMessage(pMsg);
}

void CComboCompletion::OnEditUpdate() 
{
  // if we are not to auto update the text, get outta here
  if (!m_bAutoComplete) 
      return;

  // Get the text in the edit box
  CString str;
  GetWindowText(str);
  int nLength = str.GetLength();
  
  // Currently selected range
  DWORD dwCurSel = GetEditSel();
  WORD dStart = LOWORD(dwCurSel);
  WORD dEnd   = HIWORD(dwCurSel);

  // Search for, and select in, and string in the combo box that is prefixed
  // by the text in the edit box
  if (SelectString(-1, str) == CB_ERR)
  {
      SetWindowText(str);            // No text selected, so restore what 
                                     // was there before
      if (dwCurSel != CB_ERR)
        SetEditSel(dStart, dEnd);    //restore cursor postion
  }

  // Set the text selection as the additional text that we have added
  if (dEnd < nLength && dwCurSel != CB_ERR)
      SetEditSel(dStart, dEnd);
  else
      SetEditSel(nLength, -1);
}

License

This article, along with any associated source code and files, is licensed under The Code Project Open License (CPOL)

About the Author

Chris Maunder
Founder CodeProject
Canada Canada
Member
Chris is the Co-founder, Administrator, Architect, Chief Editor and Shameless Hack who wrote and runs The Code Project. He's been programming since 1988 while pretending to be, in various guises, an astrophysicist, mathematician, physicist, hydrologist, geomorphologist, defence intelligence researcher and then, when all that got a bit rough on the nerves, a web developer. He is a Microsoft Visual C++ MVP both globally and for Canada locally.
 
His programming experience includes C/C++, C#, SQL, MFC, ASP, ASP.NET, and far, far too much FORTRAN. He has worked on PocketPCs, AIX mainframes, Sun workstations, and a CRAY YMP C90 behemoth but finds notebooks take up less desk space.
 
He dodges, he weaves, and he never gets enough sleep. He is kind to small animals.
 
Chris was born and bred in Australia but splits his time between Toronto and Melbourne, depending on the weather. For relaxation he is into road cycling, snowboarding, rock climbing, and storm chasing.

Sign Up to vote   Poor Excellent
Add a reason or comment to your vote: x
Votes of 3 or less require a comment

Comments and Discussions

 
Hint: For improved responsiveness ensure Javascript is enabled and choose 'Normal' from the Layout dropdown and hit 'Update'.
You must Sign In to use this message board.
Search this forum  
    Spacing  Noise  Layout  Per page   
GeneralCComboCOmpletion in two dialogsmemberAnu_Bala27 Jul '08 - 19:25 
GeneralLittle Bugmemberchris17522 Aug '06 - 23:56 
GeneralRe: Little Bugmemberellolazo28 Aug '06 - 9:27 
GeneralRe: Little Bugmemberellolazo31 Aug '06 - 5:34 
GeneralEnter in combobox and default buttonmemberellolazo12 Jun '06 - 6:57 
GeneralHelp..Enter keymemberG.Inbakumar20 Feb '06 - 5:29 
QuestionGreat Works,Can you complete it with C#?memberkv40008 Jun '05 - 14:59 
AnswerRe: Great Works,Can you complete it with C#?memberBob-ish27 Aug '05 - 11:51 
GeneralSuggestionsussstarschen20 Apr '05 - 17:48 
GeneralBackspace and Enter keymemberrlaley28 Oct '04 - 1:54 
GeneralRe: Backspace and Enter keymemberrlaley31 Oct '04 - 16:21 
GeneralRe: Backspace and Enter keysussReDFoX20 Feb '05 - 4:23 
GeneralRe: Backspace and Enter keymemberrlaley20 Feb '05 - 17:20 
GeneralCapturing a right-click on the combo boxmemberFlic13 Feb '03 - 17:07 
GeneralRe: Capturing a right-click on the combo boxmemberabdulsoni18 Dec '03 - 20:29 
GeneralView optionsmemberAnonymous9 May '02 - 4:16 
GeneralSimple-type ComboBox HighlightmemberMaxwell B.6 Jan '02 - 13:49 
GeneralGetCurSel() for the CComboBox is returns 2 different values on consecutive callsmemberEd Mitchell3 Oct '01 - 3:23 
GeneralCBN_SELENDOK and CBN_SELCHANGEmemberMichael Martin13 Sep '01 - 3:25 
GeneralRe: CBN_SELENDOK and CBN_SELCHANGEmemberCarlos Antollini13 Sep '01 - 3:53 
GeneralGetCurSel() doesn't return the selected index.memberKenneth Chen2 Aug '01 - 20:27 
GeneralRe: GetCurSel() doesn't return the selected index.memberKyle Poole17 May '03 - 7:03 
GeneralRe: GetCurSel() doesn't return the selected index.memberM. Wohlers3 Jan '05 - 1:43 
GeneralUsing your Class in a non Dialog based app for example in a SDI/MDI embedded in toolbar & Rebar similar to IE 5memberSean25 Jan '01 - 3:03 
GeneralRe: Using your Class in a non Dialog based app for example in a SDI/MDI embedded in toolbar & Rebar similar to IE 5memberChris Sims18 Jul '01 - 9:34 

General General    News News    Suggestion Suggestion    Question Question    Bug Bug    Answer Answer    Joke Joke    Rant Rant    Admin Admin   

Permalink | Advertise | Privacy | Mobile
Web02 | 2.6.130516.1 | Last Updated 18 Oct 2000
Article Copyright 2000 by Chris Maunder
Everything else Copyright © CodeProject, 1999-2013
Terms of Use
Layout: fixed | fluid