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

Using comboboxes in the MFC Grid Control

By , 8 Jan 2013
 

Sample Image - gridctrl_combo.gif

Introduction

Since I posted my MFC grid control I've had a lot of requests asking how to use other methods of editing cells in the grid. Ken Bertelson has taken this to the extreme, with his Grid Tree control. For some this was a little too involved, so I've created a demo project that shows how to replace the default editing with a simple combo box.

Previous methods of achieving this required that a new CGridCtrl derived class be created. With the new structure in the 2.X versions of CGridCtrl it's now a lot simpler.

A new cell type

Changing the way cells are edited is simply a matter of deriving a new grid cell class from CGridCellBase (or derivatives such as CGridCell).

I'm using the In-Place list that I used in previous versions. It's not the greatest - but it does demonstrate how to replace the default editing.

The first step is to derive a new class from CGridCell - I call it CGridCellCombo - and override Edit (which initiates editing) and EndEdit (which stops editing). You can find this class in the GridCellCombo.* files.

In Edit I create a CWnd derived class that will perform the actual editing - in this case my CInPlaceList from previous versions of the grid. There are a number of things to be aware of when creating such CWnd derived Edit classes:

  • The control will receive the row and column to be edited, the dimensions and location of the in-place edit control, the style of control to create, the original text in the cell plus the initial character that caused editing to commence (or VK_LBUTTON if the mouse was clicked on the current cell).
  • Your editing control should stop editing when it loses input focus.
  • Your editing object should handle mouse keys in a way that allows the user to navigate between cells while editing. If the control gets an arrow key for instance, it should cancel editing and return the last key it encountered back to the grid via the GVN_ENDLABELEDIT notification message.
    Note that the grid control accepts Ctrl+<arrows> for navigation, so if your in-place edit control needs to use the arrows, you can always reserve Ctrl+<arrows> to move to other cells.
  • Your edit control should pass back information about the last key pressed before editing ended. This allows correct movement among cells

The EndEdit function of your grid cell class should cease editing and destroy the in-place editing window.

If you look at the code you will see that the grid cell derived class is merely a wrapper that creates an instance of a CInPlaceList window and then sits back with the knowledge of a job well done. The actual guts of the grid cell class are as follows:

// Create a control to do the editing
BOOL CGridCellCombo::Edit(int nRow, int nCol, CRect rect, CPoint /* point */,
                          UINT nID, UINT nChar)
{
    m_bEditing = TRUE;
    
    // CInPlaceList auto-deletes itself
    m_pEditWnd = new CInPlaceList(GetGrid(), rect, GetStyle(), nID, nRow, nCol, 
                                  m_Strings, GetText(), nChar);

    return TRUE;
}

// Cancel the editing.
void CGridCellCombo::EndEdit()
{
    if (m_pEditWnd)
        ((CInPlaceList*)m_pEditWnd)->EndEdit();
}

We rely on CGridCell::OnEndEdit to set m_bEditing to FALSE and m_pEditWnd to NULL when editing has finished. CInPlaceList is also self-deleting, which saves a whole line of code in EndEdit.

The CInPlaceList does the actual work, and keeps track of issues such as the last key pressed and it's current focus state. It also has niceties such as ensuring that the drop down list is correctly sized.

The major issue is that the control must pass the necessary data back to the parent grid once editing has been completed. It uses the following code snippet in it's EndEdit method to do this:

void CInPlaceList::EndEdit()
{
    CString str;
    GetWindowText(str);
 
    // Send Notification to parent
    GV_DISPINFO dispinfo;

    dispinfo.hdr.hwndFrom = GetSafeHwnd();
    dispinfo.hdr.idFrom   = GetDlgCtrlID();
    dispinfo.hdr.code     = GVN_ENDLABELEDIT;
 
    dispinfo.item.mask    = LVIF_TEXT|LVIF_PARAM;
    dispinfo.item.row     = m_nRow;
    dispinfo.item.col     = m_nCol;
    dispinfo.item.strText = str;
    dispinfo.item.lParam  = (LPARAM) m_nLastChar; 
 
    CWnd* pOwner = GetOwner();
    if (IsWindow(pOwner->GetSafeHwnd()))
        pOwner->SendMessage(WM_NOTIFY, GetDlgCtrlID(), (LPARAM)&dispinfo );
 
    // Close this window (PostNcDestroy will delete this)
    PostMessage(WM_CLOSE, 0, 0);
}

I pass the character that initiated editing to the edit control itself so that it can deal with keys appropriately. For instance in the default edit control, if the edit control is passed a normal character or arrow key as the "initiating" key, then it will cease editing when it encounters an arrow key (to allow navigation using the keyboard). If the edit control is passed VK_LBUTTON (meaning editing was initiated by a mouse click) then arrow keys will not cause the editing to cease.

One last thing I do is ensure that the new cell displays a user hint that it is not a normal grid cell. When a cell is selected I override drawing so that a small drop down arrow is shown. The code is quite simple:

BOOL CGridCellCombo::Draw(CDC* pDC, int nRow, int nCol, CRect rect,
                          BOOL bEraseBkgnd /*=TRUE*/)
{
    // Cell selected?
    DWORD dwState = GetState();
    if ( !(dwState & GVIS_FIXED) && (dwState & GVIS_FOCUSED))
    {
        // Get the size of the scroll box
        CSize sizeScroll(GetSystemMetrics(SM_CXVSCROLL), 
                         GetSystemMetrics(SM_CYHSCROLL));

        // enough room to draw?
        if (sizeScroll.cy < rect.Width() && sizeScroll.cy < rect.Height())
        {
            // Draw control at RHS of cell
            CRect ScrollRect = rect;
            ScrollRect.left   = rect.right - sizeScroll.cx;
            ScrollRect.bottom = rect.top + sizeScroll.cy;

            // Do the draw 
            pDC->DrawFrameControl(ScrollRect, DFC_SCROLL, DFCS_SCROLLDOWN);

            // Adjust the remaining space in the cell
            rect.right = ScrollRect.left;
        }
    }

    // drop through and complete the cell drawing using the base class' method
    return CGridCell::Draw(pDC, nRow, nCol, rect,  bEraseBkgnd);
}

Using the new cell type

Using the new cell is simple. You can either use the Grid control's default cell instantiation method (ie. do nothing) and then call CGridCtrl::SetCellType or you can call CGridCtrl::SetDefaultCellType and then do nothing. The first option changes cells from one type to another, and is good if you only want specific cells to be affected. The second option tells the grid the class type that you want to be used at the outset, and is good if you want all cells in the grid to be of your new type.

Thanks

Thanks go to Roelf Werkman for his work in extending the CInPlaceList to allow the CBS_DROPDOWN and CBS_SIMPLE styles to be used, and thanks to Fred Ackers for moticating me to finally write this up.

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   
QuestionBeep when I hit ENTERmemberandrewtruckle5 Feb '07 - 0:51 
AnswerRe: Beep when I hit ENTERmemberandrewtruckle9 Feb '07 - 10:29 
QuestionDisabling the cellmemberandrewtruckle23 Jan '07 - 0:45 
AnswerRe: Disabling the cellmemberandrewtruckle23 Jan '07 - 0:51 
QuestionItem Indexmemberandrewtruckle17 Jan '07 - 8:46 
Questionhow to have a combo in the header?memberArris713 Jan '07 - 23:16 
QuestionCGridCellCombo and itemData.memberlollinus17 Oct '06 - 3:04 
AnswerRe: CGridCellCombo and itemData.memberlollinus17 Oct '06 - 4:30 
QuestionHow a can display the list of the combo-boxmemberxkill27 Aug '06 - 8:57 
GeneralEditing Cells in MFCmemberSarah2Frog7 Jun '06 - 4:57 
GeneralRe: Editing Cells in MFCmemberSarah2Frog8 Jun '06 - 2:52 
GeneralRe: Editing Cells in MFCmemberHaiying8 Aug '06 - 9:47 
GeneralManage event on CGridCellCombomemberpocchio15 May '06 - 8:13 
I have a grid with two columns , one is a CGridCellCombo and one is a CGridCell.
I want put the combo selected item in the text in the CGridCell when I change the value on the CGridCellCombo.
How can I do this ?
Can someone help me ?
Mad | :mad:
GeneralRe: Manage event on CGridCellCombomemberHaiying8 Aug '06 - 9:50 
GeneralEntering non-list values in combo-boxmember3p5 Apr '06 - 0:19 
GeneralRe: Entering non-list values in combo-boxmemberandrewtruckle9 Feb '07 - 10:15 
GeneralYou can make editing flexgrid this waymemberthannerulavanya22 Mar '06 - 22:57 
QuestionChanging the style to "Sort" causes odd behaviormemberandrewtruckle11 Mar '06 - 5:27 
QuestionRightClick menu on the gridmemberMilind Wasaikar12 Jan '06 - 22:28 
QuestionGrey SpacememberMilind Wasaikar12 Jan '06 - 2:34 
GeneralCDateTimeCtrlmemberseher akdeniz4 Sep '05 - 22:00 
GeneralGetCurSelmembermartin beckett8 Aug '05 - 5:40 
QuestionStrange code?memberBob Stanneveld1 Apr '05 - 0:43 
AnswerRe: Strange code?memberAM8423 May '05 - 13:28 
GeneralRe: Strange code?memberBob Stanneveld23 May '05 - 20:14 
GeneralRe: Strange code?memberAnders Gustafsson9 Aug '06 - 9:07 
GeneralWord Wrappingmemberlarryp30 Dec '04 - 13:48 
GeneralWay to auto drop-down the combo boxesmemberCyndi7 Dec '04 - 8:31 
GeneralRe: No tab stop on combo boxmemberChristian Cheney2 Mar '04 - 11:34 
GeneralRe: No tab stop on combo boxmemberChristian Cheney2 Mar '04 - 11:43 
GeneralRe: No tab stop on combo boxmemberAnders Gustafsson23 Nov '07 - 10:06 
GeneralSetItemFontmemberkouchba1 Oct '03 - 2:43 
Generalitem data for ComboBoxmemberpalim21 May '03 - 3:08 
GeneralMask input in the cellsmemberjarribas14 May '03 - 8:36 
QuestionSlider?sussMark Ping21 Mar '03 - 13:26 
GeneralComboBox & VirtualModesussGianluca Nastasi12 Mar '03 - 0:35 
QuestionHow to put MFCGridCtrl into a property page.memberHanney Wang7 Mar '03 - 14:21 
AnswerRe: How to put MFCGridCtrl into a property page.memberHanney Wang15 Mar '03 - 2:18 
GeneralEmpty cellsussGianluca Nastasi28 Feb '03 - 5:06 
GeneralRe: Empty cellmemberHanney Wang15 Mar '03 - 2:42 
GeneralRe: Empty cellmemberxuyuhua20001 Jun '06 - 17:17 
Questiononly seleting is allowed.but how?memberwithoutdruck5 Jan '03 - 1:18 
AnswerRe: only seleting is allowed.but how?memberRolando E. Cruz-Marshall10 Jan '03 - 9:04 
General[Code change-proposal] SetOptions(CStringArray) to SetOptions(CTupleArray)memberoglimmer25 Nov '02 - 21:57 
QuestionHow to use the grid with the new HTML class ?memberAppstmd15 Sep '02 - 7:56 
GeneralMerging CellsmemberKP2410 Sep '02 - 18:50 
GeneralNESTED GRIDSmemberJi Gua6 Aug '02 - 6:39 
GeneralPlease tell me it can combine multi- cell and .....memberliangzhiguo2 Jun '02 - 14:43 
QuestionCan this work for WinCE?memberAnonymous23 Feb '02 - 5:06 
GeneralGrid's row selectionmemberConfused Ai20 Dec '01 - 3:52 

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

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