Click here to Skip to main content
Licence 
First Posted 6 Aug 2006
Views 80,796
Bookmarked 50 times

A combobox that looks decent when it is disabled

By | 6 Aug 2006 | Article
This article describes a method to ownerdraw a combo box so that the text is readable when the combobox is disabled.

Introduction

One of the things I have never liked about the combobox is that it is almost unreadable when it is disabled. I have an application that does a lot of databinding and enables or disables controls behind the scenes based upon access rules. The comboboxes show the right values, but it is grey on beige, and very hard to read. I don't want to make them textboxes because it would make my code considerably more complex (the GUI is generated at runtime).

I did a lot of research on the web and didn't find too many articles that dealt specifically with changing the appearance of the edit box. There are lots of articles that show an ownerdrawn list, but that only is helpful when the listbox is enabled.

What it looks like

As you can see in the first entry, the disabled combo is pretty ugly. The second entry shows my control in a disabled state - much more readable! The third entry shows that the control works normally when enabled.

I implemented the code as a simple subclass from combobox, so it can be dropped in as a replacement anywhere a combobox is used.

Key parts of the code

First, to allow your combo to be owner drawn, you have to add a DrawItemEventHandler and also set the DrawMode to OwnerDrawFixed. You can use OwnerDrawVariable if you want to change the size of the list, but I am not doing that here. I also added a handler for EnabledChanged event.

// Required for ownerdraw
this.DrawItem += new DrawItemEventHandler(EnableDisplayCombo_DrawItem);

// Required for ownerdraw
this.DrawMode = DrawMode.OwnerDrawFixed;
this.EnabledChanged += new EventHandler(EnableDisplayCombo_EnabledChanged);

Next, in the EnabledChanged event, I change the DropDownStyle based upon whether the control is enabled or not. A DropDownStyle of just DropDown will allow the control to be typed into, but DropDownList will only allow the use of the mouse. The catch is that you can only control the appearance of the EditBox if you set the DropDownStyle to DropDownList. For a disabled control, it doesn't matter since you can't select it anyway.

void EnableDisplayCombo_EnabledChanged(object sender, EventArgs e)
{
    if (this.Enabled)
        this.DropDownStyle = ComboBoxStyle.DropDown;
    else
        this.DropDownStyle = ComboBoxStyle.DropDownList;
}

Finally, here is the ownerdraw routine. Basically, you just key on the state of the item that needs to be drawn, and draw it however you want.

void EnableDisplayCombo_DrawItem(object sender, DrawItemEventArgs e)
{    
    System.Drawing.Graphics g = e.Graphics;    
    Rectangle r = e.Bounds;    
    if (e.Index >= 0)    
    {        
        string label = this.Items[e.Index].ToString();        
        // This is how we draw a disabled control        
        if (e.State == (DrawItemState.Disabled | DrawItemState.NoAccelerator 
                | DrawItemState.NoFocusRect | DrawItemState.ComboBoxEdit))   
        {            
            e.Graphics.FillRectangle(new SolidBrush(Color.White), r);      
            g.DrawString(label, e.Font, Brushes.Black, r);            
            e.DrawFocusRectangle();        
        }        
        
        // This is how we draw the items in an enabled control that aren't<BR>        // in focus        
        else if (e.State == (DrawItemState.NoAccelerator | <BR>                             DrawItemState.NoFocusRect))        
        {            
            e.Graphics.FillRectangle(new SolidBrush(Color.White), r); 
            g.DrawString(label, e.Font, Brushes.Black, r);            
            e.DrawFocusRectangle();        
        }        
        
        // This is how we draw the focused items        
        else        
        {            
            e.Graphics.FillRectangle(new SolidBrush(Color.Blue), r); 
            g.DrawString(label, e.Font, Brushes.White, r);            
            e.DrawFocusRectangle();        
        }    
    }    
    g.Dispose();
}

Note that if you use databinding you will have to use a little more complex code in the line where the label is assigned. Maybe something like this:

string label = Convert.ToString(item[this.DisplayMember]);

I hope this is useful to somebody. The solution is pretty simple, but it took me a long time to piece together from various sources. Any comments or suggestions are very welcome.

License

This article has no explicit license attached to it but may contain usage terms in the article text or the download files themselves. If in doubt please contact the author via the discussion board below.

A list of licenses authors might use can be found here

About the Author

AlexKilpatrick

Web Developer

United States United States

Member

Alex Kilpatrick started writing software in 1981, modifying computer games written in basic. Since then, he has dabbled off and on in about 15 different languages, but currently prefers C#.

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

 
You must Sign In to use this message board. (secure sign-in)
 
Search this forum  
 FAQ
    Noise  Layout  Per page   
  Refresh
GeneralUnneeded Dispose PinmemberMichael Ehrt0:54 24 May '11  
GeneralUnreleased resources PinmemberMember 733103411:40 27 Aug '10  
GeneralMissing Row Highlight PinmemberMember 733103411:26 27 Aug '10  
GeneralBug Update PinmemberBib3477022:10 16 Sep '09  
Questionif comboBox's DropDownStyle = ComboBoxStyle.Simple, It looks as regular comboBox when disabled. Pinmemberflyrain00016:08 22 Jul '09  
GeneralVery good tutorial! Thanks very much! Pinmembersti_sti23:36 12 Aug '08  
GeneralAwesome PinmemberRakesh B Singh0:28 2 Jun '07  
GeneralDatabinding revisited PinmemberJohnny J.4:53 26 Apr '07  
GeneralRe: Databinding revisited Pinmemberannielhy16:49 20 Jun '08  
GeneralDropDownList PinmemberJohnny J.0:16 24 Apr '07  
GeneralUpdate for WinXP and W2K3 PinmemberRine22:44 21 Mar '07  
QuestionHow to handle databound combobox Pinmembersjoske3:48 13 Sep '06  
GeneralNice, but a few comments though PinmemberHenk van der Geld6:07 8 Aug '06  
GeneralRe: Nice, but a few comments though PinmemberAbhlu16:22 10 Oct '06  
GeneralASP.NET Version Pinmemberrjpaulsen4:49 7 Aug '06  
GeneralRe: ASP.NET Version PinmemberAlexKilpatrick5:15 7 Aug '06  
GeneralExcellent Trick !! PinmemberMarcos Meli3:45 7 Aug '06  
GeneralI'll certainly have to take a look PinmemberIlíon3:18 7 Aug '06  
GeneralRe: I'll certainly have to take a look PinmemberAlexKilpatrick5:17 7 Aug '06  
GeneralRe: I'll certainly have to take a look PinmemberIlíon9:10 7 Aug '06  
GeneralRe: I'll certainly have to take a look PinmemberIlíon13:42 18 Aug '06  

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

Use Ctrl+Left/Right to switch messages, Ctrl+Up/Down to switch threads, Ctrl+Shift+Left/Right to switch pages.

Permalink | Advertise | Privacy | Mobile
Web04 | 2.5.120517.1 | Last Updated 6 Aug 2006
Article Copyright 2006 by AlexKilpatrick
Everything else Copyright © CodeProject, 1999-2012
Terms of Use
Layout: fixed | fluid