Code:
/ Dotnetfx_Vista_SP2 / Dotnetfx_Vista_SP2 / 8.0.50727.4016 / DEVDIV / depot / DevDiv / releases / whidbey / NetFxQFE / ndp / fx / src / WinForms / Managed / System / WinForms / ComboBox.cs / 1 / ComboBox.cs
//------------------------------------------------------------------------------ //// Copyright (c) Microsoft Corporation. All rights reserved. // //----------------------------------------------------------------------------- /* */ namespace System.Windows.Forms { using Accessibility; using System.Runtime.Serialization.Formatters; using System.Threading; using System.Runtime.Remoting; using System.Runtime.InteropServices; using System.Diagnostics; using System; using System.Globalization; using System.Security; using System.Security.Permissions; using System.Windows.Forms.Layout; using System.Windows.Forms.ComponentModel; using System.ComponentModel; using System.ComponentModel.Design; using System.Collections; using System.Drawing; using System.Windows.Forms.Internal; using System.Drawing.Design; using Microsoft.Win32; using System.Reflection; using System.Collections.Specialized; using System.Text; using System.Diagnostics.CodeAnalysis; ////// /// [ ComVisible(true), ClassInterface(ClassInterfaceType.AutoDispatch), DefaultEvent("SelectedIndexChanged"), DefaultProperty("Items"), DefaultBindingProperty("Text"), Designer("System.Windows.Forms.Design.ComboBoxDesigner, " + AssemblyRef.SystemDesign), SRDescription(SR.DescriptionComboBox) ] public class ComboBox : ListControl { private static readonly object EVENT_DROPDOWN = new object(); private static readonly object EVENT_DRAWITEM = new object(); private static readonly object EVENT_MEASUREITEM = new object(); private static readonly object EVENT_SELECTEDINDEXCHANGED = new object(); private static readonly object EVENT_SELECTIONCHANGECOMMITTED = new object(); private static readonly object EVENT_SELECTEDITEMCHANGED = new object(); private static readonly object EVENT_DROPDOWNSTYLE = new object(); private static readonly object EVENT_TEXTUPDATE = new object(); private static readonly object EVENT_DROPDOWNCLOSED = new object(); private static readonly int PropMaxLength = PropertyStore.CreateKey(); private static readonly int PropItemHeight = PropertyStore.CreateKey(); private static readonly int PropDropDownWidth = PropertyStore.CreateKey(); private static readonly int PropDropDownHeight = PropertyStore.CreateKey(); private static readonly int PropStyle = PropertyStore.CreateKey(); private static readonly int PropDrawMode = PropertyStore.CreateKey(); private static readonly int PropMatchingText = PropertyStore.CreateKey(); private static readonly int PropFlatComboAdapter = PropertyStore.CreateKey(); private const int DefaultSimpleStyleHeight = 150; private const int DefaultDropDownHeight = 106; private const int AutoCompleteTimeout = 10000000; // 1 second timeout for resetting the MatchingText private bool autoCompleteDroppedDown = false; private FlatStyle flatStyle = FlatStyle.Standard; private int updateCount; //Timestamp of the last keystroke. Used for auto-completion // in DropDownList style. private long autoCompleteTimeStamp; private int selectedIndex = -1; // used when we don't have a handle. private bool allowCommit = true; // When the style is "simple", the requested height is used // for the actual height of the control. // When the style is non-simple, the height of the control // is determined by the OS. // This fixes bug #20966 private int requestedHeight; private ComboBoxChildNativeWindow childEdit; private ComboBoxChildNativeWindow childListBox; private IntPtr dropDownHandle; private ObjectCollection itemsCollection; private short prefHeightCache = -1; private short maxDropDownItems = 8; private bool integralHeight = true; private bool mousePressed; private bool mouseEvents; private bool mouseInEdit; private bool sorted; private bool fireSetFocus = true; private bool fireLostFocus = true; private bool mouseOver; private bool suppressNextWindosPos; private bool canFireLostFocus; // When the user types a letter and drops the dropdown... // the combobox itself auto-searches the matching item... // and selects the item in the edit... // thus changing the windowText... // hence we should Fire the TextChanged event in such a scenario.. // The string below is used for checking the window Text before and after the dropdown. private string currentText = ""; private string lastTextChangedValue; private bool dropDown; private AutoCompleteDropDownFinder finder = new AutoCompleteDropDownFinder(); private bool selectedValueChangedFired; ////// Displays an editing field and a list, allowing the user to select from the /// list or to enter new text. Displays only the editing field until the user /// explicitly displays the list. /// /// ////// /// This stores the value for the autocomplete mode which can be either /// None, AutoSuggest, AutoAppend or AutoSuggestAppend. /// private AutoCompleteMode autoCompleteMode = AutoCompleteMode.None; ////// /// This stores the value for the autoCompleteSource mode which can be one of the values /// from AutoCompleteSource enum. /// private AutoCompleteSource autoCompleteSource = AutoCompleteSource.None; ////// /// This stores the custom StringCollection required for the autoCompleteSource when its set to CustomSource. /// private AutoCompleteStringCollection autoCompleteCustomSource; private StringSource stringSource; private bool fromHandleCreate = false; ////// /// Creates a new ComboBox control. The default style for the combo is /// a regular DropDown Combo. /// public ComboBox() { SetStyle(ControlStyles.UserPaint | ControlStyles.UseTextForAccessibility | ControlStyles.StandardClick, false); requestedHeight = DefaultSimpleStyleHeight; // this class overrides GetPreferredSizeCore, let Control automatically cache the result SetState2(STATE2_USEPREFERREDSIZECACHE, true); } ////// /// This is the AutoCompleteMode which can be either /// None, AutoSuggest, AutoAppend or AutoSuggestAppend. /// This property in conjunction with AutoCompleteSource enables the AutoComplete feature for ComboBox. /// [ DefaultValue(AutoCompleteMode.None), SRDescription(SR.ComboBoxAutoCompleteModeDescr), Browsable(true), EditorBrowsable(EditorBrowsableState.Always) ] public AutoCompleteMode AutoCompleteMode { get { return autoCompleteMode; } set { //valid values are 0x0 to 0x3 if (!ClientUtils.IsEnumValid(value, (int)value, (int)AutoCompleteMode.None, (int)AutoCompleteMode.SuggestAppend)) { throw new InvalidEnumArgumentException("value", (int)value, typeof(AutoCompleteMode)); } if (this.DropDownStyle == ComboBoxStyle.DropDownList && this.AutoCompleteSource != AutoCompleteSource.ListItems && value != AutoCompleteMode.None) { throw new NotSupportedException(SR.GetString(SR.ComboBoxAutoCompleteModeOnlyNoneAllowed)); } if (Application.OleRequired() != System.Threading.ApartmentState.STA) { throw new ThreadStateException(SR.GetString(SR.ThreadMustBeSTA)); } bool resetAutoComplete = false; if (autoCompleteMode != AutoCompleteMode.None && value == AutoCompleteMode.None) { resetAutoComplete = true; } autoCompleteMode = value; SetAutoComplete(resetAutoComplete, true); } } ////// /// This is the AutoCompleteSource which can be one of the /// values from AutoCompleteSource enumeration. /// This property in conjunction with AutoCompleteMode enables the AutoComplete feature for ComboBox. /// [ DefaultValue(AutoCompleteSource.None), SRDescription(SR.ComboBoxAutoCompleteSourceDescr), Browsable(true), EditorBrowsable(EditorBrowsableState.Always) ] public AutoCompleteSource AutoCompleteSource { get { return autoCompleteSource; } set { // FxCop: Avoid usage of Enum.IsDefined - this looks like an enum that could grow if (!ClientUtils.IsEnumValid_NotSequential(value, (int)value, (int)AutoCompleteSource.None, (int)AutoCompleteSource.AllSystemSources, (int)AutoCompleteSource.AllUrl, (int)AutoCompleteSource.CustomSource, (int)AutoCompleteSource.FileSystem, (int)AutoCompleteSource.FileSystemDirectories, (int)AutoCompleteSource.HistoryList, (int)AutoCompleteSource.ListItems, (int)AutoCompleteSource.RecentlyUsedList)) { throw new InvalidEnumArgumentException("value", (int)value, typeof(AutoCompleteSource)); } if (this.DropDownStyle == ComboBoxStyle.DropDownList && this.AutoCompleteMode != AutoCompleteMode.None && value != AutoCompleteSource.ListItems) { throw new NotSupportedException(SR.GetString(SR.ComboBoxAutoCompleteSourceOnlyListItemsAllowed)); } if (Application.OleRequired() != System.Threading.ApartmentState.STA) { throw new ThreadStateException(SR.GetString(SR.ThreadMustBeSTA)); } // VSWhidbey 466300 if (value != AutoCompleteSource.None && value != AutoCompleteSource.CustomSource && value != AutoCompleteSource.ListItems) { FileIOPermission fiop = new FileIOPermission(PermissionState.Unrestricted); fiop.AllFiles = FileIOPermissionAccess.PathDiscovery; fiop.Demand(); } autoCompleteSource = value; SetAutoComplete(false, true); } } ////// /// This is the AutoCompleteCustomSource which is custom StringCollection used when the /// AutoCompleteSource is CustomSource. /// [ DesignerSerializationVisibility(DesignerSerializationVisibility.Content), Localizable(true), SRDescription(SR.ComboBoxAutoCompleteCustomSourceDescr), Editor("System.Windows.Forms.Design.ListControlStringCollectionEditor, " + AssemblyRef.SystemDesign, typeof(UITypeEditor)), Browsable(true), EditorBrowsable(EditorBrowsableState.Always) ] public AutoCompleteStringCollection AutoCompleteCustomSource { get { if (autoCompleteCustomSource == null) { autoCompleteCustomSource = new AutoCompleteStringCollection(); autoCompleteCustomSource.CollectionChanged += new CollectionChangeEventHandler(this.OnAutoCompleteCustomSourceChanged); } return autoCompleteCustomSource; } set { if (autoCompleteCustomSource != value) { if (autoCompleteCustomSource != null) { autoCompleteCustomSource.CollectionChanged -= new CollectionChangeEventHandler(this.OnAutoCompleteCustomSourceChanged); } autoCompleteCustomSource = value; if (autoCompleteCustomSource != null) { autoCompleteCustomSource.CollectionChanged += new CollectionChangeEventHandler(this.OnAutoCompleteCustomSourceChanged); } SetAutoComplete(false, true); } } } ////// /// The background color of this control. This is an ambient property and /// will always return a non-null value. /// public override Color BackColor { get { if (ShouldSerializeBackColor()) { return base.BackColor; } else { return SystemColors.Window; } } set { base.BackColor = value; } } ////// /// [Browsable(false), EditorBrowsable(EditorBrowsableState.Never)] public override Image BackgroundImage { get { return base.BackgroundImage; } set { base.BackgroundImage = value; } } ///[To be supplied.] ////// /// [Browsable(false), EditorBrowsable(EditorBrowsableState.Never)] public override ImageLayout BackgroundImageLayout { get { return base.BackgroundImageLayout; } set { base.BackgroundImageLayout = value; } } ///[To be supplied.] ////// [Browsable(false), EditorBrowsable(EditorBrowsableState.Never)] new public event EventHandler BackgroundImageChanged { add { base.BackgroundImageChanged += value; } remove { base.BackgroundImageChanged -= value; } } /// /// [Browsable(false), EditorBrowsable(EditorBrowsableState.Never)] new public event EventHandler BackgroundImageLayoutChanged { add { base.BackgroundImageLayoutChanged += value; } remove { base.BackgroundImageLayoutChanged -= value; } } /// /// /// Returns the parameters needed to create the handle. Inheriting classes /// can override this to provide extra functionality. They should not, /// however, forget to call base.CreateParams() first to get the struct /// filled up with the basic info. /// ///protected override CreateParams CreateParams { [SecurityPermission(SecurityAction.LinkDemand, Flags = SecurityPermissionFlag.UnmanagedCode)] get { CreateParams cp = base.CreateParams; cp.ClassName = "COMBOBOX"; cp.Style |= NativeMethods.WS_VSCROLL | NativeMethods.CBS_HASSTRINGS | NativeMethods.CBS_AUTOHSCROLL; cp.ExStyle |= NativeMethods.WS_EX_CLIENTEDGE; if (!integralHeight) cp.Style |= NativeMethods.CBS_NOINTEGRALHEIGHT; switch (DropDownStyle) { case ComboBoxStyle.Simple: cp.Style |= NativeMethods.CBS_SIMPLE; break; case ComboBoxStyle.DropDown: cp.Style |= NativeMethods.CBS_DROPDOWN; // Make sure we put the height back or we won't be able to size the dropdown! cp.Height = PreferredHeight; break; case ComboBoxStyle.DropDownList: cp.Style |= NativeMethods.CBS_DROPDOWNLIST; // Comment above... cp.Height = PreferredHeight; break; } switch (DrawMode) { case DrawMode.OwnerDrawFixed: cp.Style |= NativeMethods.CBS_OWNERDRAWFIXED; break; case DrawMode.OwnerDrawVariable: cp.Style |= NativeMethods.CBS_OWNERDRAWVARIABLE; break; } return cp; } } /// /// /// Deriving classes can override this to configure a default size for their control. /// This is more efficient than setting the size in the control's constructor. /// protected override Size DefaultSize { get { return new Size(121, PreferredHeight); } } ////// /// The ListSource to consume as this ListBox's source of data. /// When set, a user can not modify the Items collection. /// [ SRCategory(SR.CatData), DefaultValue(null), RefreshProperties(RefreshProperties.Repaint), AttributeProvider(typeof(IListSource)), SRDescription(SR.ListControlDataSourceDescr) ] public new object DataSource { get { return base.DataSource; } set { base.DataSource = value; } } ////// /// Retrieves the value of the DrawMode property. The DrawMode property /// controls whether the control is drawn by Windows or by the user. /// [ SRCategory(SR.CatBehavior), DefaultValue(DrawMode.Normal), SRDescription(SR.ComboBoxDrawModeDescr), RefreshProperties(RefreshProperties.Repaint) ] public DrawMode DrawMode { get { bool found; int drawMode = Properties.GetInteger(PropDrawMode, out found); if (found) { return (DrawMode)drawMode; } return DrawMode.Normal; } set { if (DrawMode != value) { //valid values are 0x0 to 0x2. if (!ClientUtils.IsEnumValid(value, (int)value, (int)DrawMode.Normal, (int)DrawMode.OwnerDrawVariable)) { throw new InvalidEnumArgumentException("value", (int)value, typeof(DrawMode)); } ResetHeightCache(); Properties.SetInteger(PropDrawMode, (int)value); RecreateHandle(); } } } ////// /// Returns the width of the drop down box in a combo box. /// [ SRCategory(SR.CatBehavior), SRDescription(SR.ComboBoxDropDownWidthDescr) ] public int DropDownWidth { get { bool found; int dropDownWidth = Properties.GetInteger(PropDropDownWidth, out found); if (found) { return dropDownWidth; } else { return Width; } } set { if (value < 1) { throw new ArgumentOutOfRangeException("DropDownWidth", SR.GetString(SR.InvalidArgument, "DropDownWidth", (value).ToString(CultureInfo.CurrentCulture))); } if (Properties.GetInteger(PropDropDownWidth) != value) { Properties.SetInteger(PropDropDownWidth, value); if (IsHandleCreated) { SendMessage(NativeMethods.CB_SETDROPPEDWIDTH, value, 0); } } } } ////// /// Sets the Height of the drop down box in a combo box. /// [ SRCategory(SR.CatBehavior), SRDescription(SR.ComboBoxDropDownHeightDescr), Browsable(true), EditorBrowsable(EditorBrowsableState.Always), DefaultValue(106) ] public int DropDownHeight { get { bool found; int dropDownHeight = Properties.GetInteger(PropDropDownHeight, out found); if (found) { return dropDownHeight; } else { return DefaultDropDownHeight; } } set { if (value < 1) { throw new ArgumentOutOfRangeException("DropDownHeight", SR.GetString(SR.InvalidArgument, "DropDownHeight", (value).ToString(CultureInfo.CurrentCulture))); } if (Properties.GetInteger(PropDropDownHeight) != value) { Properties.SetInteger(PropDropDownHeight, value); // The dropDownHeight is not reflected unless the // combobox integralHeight == false.. IntegralHeight = false; } } } ////// /// Indicates whether the DropDown of the combo is currently dropped down. /// [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden), SRDescription(SR.ComboBoxDroppedDownDescr) ] public bool DroppedDown { get { if (IsHandleCreated) { return (int)SendMessage(NativeMethods.CB_GETDROPPEDSTATE, 0, 0) != 0; } else { return false; } } set { if (!IsHandleCreated) { CreateHandle(); } SendMessage(NativeMethods.CB_SHOWDROPDOWN, value ? -1 : 0, 0); } } ////// /// [ SRCategory(SR.CatAppearance), DefaultValue(FlatStyle.Standard), Localizable(true), SRDescription(SR.ComboBoxFlatStyleDescr) ] public FlatStyle FlatStyle { get { return flatStyle; } set { //valid values are 0x0 to 0x3 if (!ClientUtils.IsEnumValid(value, (int)value, (int)FlatStyle.Flat, (int)FlatStyle.System)) { throw new InvalidEnumArgumentException("value", (int)value, typeof(FlatStyle)); } flatStyle = value; Invalidate(); } } ////// Gets or /// sets /// the flat style appearance of the button control. /// ////// /// Returns true if this control has focus. /// public override bool Focused { get { if (base.Focused) return true; IntPtr focus = UnsafeNativeMethods.GetFocus(); return focus != IntPtr.Zero && ((childEdit != null && focus == childEdit.Handle) || (childListBox != null && focus == childListBox.Handle)); } } ////// /// public override Color ForeColor { get { if (ShouldSerializeForeColor()) { return base.ForeColor; } else { return SystemColors.WindowText; } } set { base.ForeColor = value; } } ////// Gets or sets the foreground color of the control. /// ////// /// Indicates if the combo should avoid showing partial Items. If so, /// then only full items will be displayed, and the list portion will be resized /// to prevent partial items from being shown. Otherwise, they will be /// shown /// [ SRCategory(SR.CatBehavior), DefaultValue(true), Localizable(true), SRDescription(SR.ComboBoxIntegralHeightDescr) ] public bool IntegralHeight { get { return integralHeight; } set { if (integralHeight != value) { integralHeight = value; RecreateHandle(); } } } ////// /// Returns the height of an item in the combo box. When drawMode is Normal /// or OwnerDrawFixed, all items have the same height. When drawMode is /// OwnerDrawVariable, this method returns the height that will be given /// to new items added to the combo box. To determine the actual height of /// an item, use the GetItemHeight() method with an integer parameter. /// [ SRCategory(SR.CatBehavior), Localizable(true), SRDescription(SR.ComboBoxItemHeightDescr) ] public int ItemHeight { get { DrawMode drawMode = DrawMode; if (drawMode == DrawMode.OwnerDrawFixed || drawMode == DrawMode.OwnerDrawVariable || !IsHandleCreated) { bool found; int itemHeight = Properties.GetInteger(PropItemHeight, out found); if (found) { return itemHeight; } else { return FontHeight + 2; // bug (90774)+2 for the 1 pixel gap on each side (up and Bottom) of the Text. } } // Note that the above if clause deals with the case when the handle has not yet been created Debug.Assert(IsHandleCreated, "Handle should be created at this point"); int h = (int)SendMessage(NativeMethods.CB_GETITEMHEIGHT, 0, 0); if (h == -1) { throw new Win32Exception(); } return h; } set { if (value < 1) { throw new ArgumentOutOfRangeException("ItemHeight", SR.GetString(SR.InvalidArgument, "ItemHeight", (value).ToString(CultureInfo.CurrentCulture))); } ResetHeightCache(); if (Properties.GetInteger(PropItemHeight) != value) { Properties.SetInteger(PropItemHeight, value); if (DrawMode != DrawMode.Normal) { UpdateItemHeight(); } } } } ////// /// Collection of the items contained in this ComboBox. /// [ SRCategory(SR.CatData), DesignerSerializationVisibility(DesignerSerializationVisibility.Content), Localizable(true), SRDescription(SR.ComboBoxItemsDescr), Editor("System.Windows.Forms.Design.ListControlStringCollectionEditor, " + AssemblyRef.SystemDesign, typeof(UITypeEditor)), MergableProperty(false) ] public ObjectCollection Items { get { if (itemsCollection == null) { itemsCollection = new ObjectCollection(this); } return itemsCollection; } } // Text used to match an item in the list when auto-completion // is used in DropDownList style. private string MatchingText { get { string matchingText = (string)Properties.GetObject(PropMatchingText); return (matchingText == null) ? string.Empty : matchingText; } set { if (value != null || this.Properties.ContainsObject(PropMatchingText)) { Properties.SetObject(PropMatchingText, value); } } } ////// /// The maximum number of items to be shown in the dropdown portion /// of the ComboBox. This number can be between 1 and 100. /// [ SRCategory(SR.CatBehavior), DefaultValue(8), Localizable(true), SRDescription(SR.ComboBoxMaxDropDownItemsDescr) ] public int MaxDropDownItems { get { return maxDropDownItems; } set { if (value < 1 || value > 100) { throw new ArgumentOutOfRangeException("MaxDropDownItems", SR.GetString(SR.InvalidBoundArgument, "MaxDropDownItems", (value).ToString(CultureInfo.CurrentCulture), (1).ToString(CultureInfo.CurrentCulture), (100).ToString(CultureInfo.CurrentCulture))); } maxDropDownItems = (short)value; } } public override Size MaximumSize { get { return base.MaximumSize; } set { base.MaximumSize = new Size(value.Width, 0); } } public override Size MinimumSize { get { return base.MinimumSize; } set { base.MinimumSize = new Size(value.Width, 0); } } ////// /// The maximum length of the text the user may type into the edit control /// of a combo box. /// [ SRCategory(SR.CatBehavior), DefaultValue(0), Localizable(true), SRDescription(SR.ComboBoxMaxLengthDescr) ] public int MaxLength { get { return Properties.GetInteger(PropMaxLength); } set { if (value < 0) value = 0; if (MaxLength != value) { Properties.SetInteger(PropMaxLength, value); if (IsHandleCreated) SendMessage(NativeMethods.CB_LIMITTEXT, value, 0); } } } ////// /// If the mouse is over the combobox, draw selection rect. /// internal bool MouseIsOver { get { return mouseOver; } set { if (mouseOver != value) { mouseOver = value; // Nothing to see here... Just keep on walking... VSWhidbey 504477. // Turns out that with Theming off, we don't get quite the same messages as with theming on, so // our drawing gets a little messed up. So in case theming is off, force a draw here. if ((!ContainsFocus || !Application.RenderWithVisualStyles) && this.FlatStyle == FlatStyle.Popup) { Invalidate(); Update(); } } } } ////// /// [ Browsable(false), EditorBrowsable(EditorBrowsableState.Never), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden) ] public new Padding Padding { get { return base.Padding; } set { base.Padding = value; } } [ Browsable(false), EditorBrowsable(EditorBrowsableState.Never) ] public new event EventHandler PaddingChanged { add { base.PaddingChanged += value; } remove { base.PaddingChanged -= value; } } ////// ///[To be supplied.] ////// /// ApplySizeConstraints calls into this method when DropDownStyles is DropDown and DropDownList. /// This causes PreferredSize to be bounded by PreferredHeight in these two cases only. /// [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden), SRDescription(SR.ComboBoxPreferredHeightDescr) ] public int PreferredHeight { get { if (!FormattingEnabled) { //see #465057 for why I did this //do preferred height the old broken way for everett apps //we need this for compat reasons because (get this) // (a) everett preferredheight was always wrong. // (b) so, when combobox1.Size = actualdefaultsize was called, it would enter setboundscore // (c) this updated requestedheight // (d) if the user then changed the combo to simple style, the height did not change. // We simply cannot match this behavior if preferredheight is corrected so that (b) never // occurs. We simply do not know when Size was set. // So in whidbey, the behavior will be: // (1) user uses default size = setting dropdownstyle=simple will revert to simple height // (2) user uses nondefault size = setting dropdownstyle=simple will not change height from this value //In everett // if the user manually sets Size = (121, 20) in code (usually height gets forced to 21), then he will see Whidey.(1) above // user usually uses nondefault size and will experience whidbey.(2) above Size textSize = TextRenderer.MeasureText(LayoutUtils.TestString, this.Font, new Size(Int16.MaxValue, (int)(FontHeight * 1.25)), TextFormatFlags.SingleLine); prefHeightCache = (short)(textSize.Height + SystemInformation.BorderSize.Height * 8 + Padding.Size.Height); return prefHeightCache ; } else { // Normally we do this sort of calculation in GetPreferredSizeCore which has builtin // caching, but in this case we can not because PreferredHeight is used in ApplySizeConstraints // which is used by GetPreferredSize (infinite loop). if (prefHeightCache < 0) { Size textSize = TextRenderer.MeasureText(LayoutUtils.TestString, this.Font, new Size(Int16.MaxValue, (int)(FontHeight * 1.25)), TextFormatFlags.SingleLine); // For a "simple" style combobox, the preferred height depends on the // number of items in the combobox. if (DropDownStyle == ComboBoxStyle.Simple) { int itemCount = Items.Count + 1; prefHeightCache = (short)(textSize.Height * itemCount + SystemInformation.BorderSize.Height * 16 + Padding.Size.Height); } else { // We do this old school rather than use SizeFromClientSize because CreateParams calls this // method and SizeFromClientSize calls CreateParams (another infinite loop.) prefHeightCache = (short)GetComboHeight(); } } return prefHeightCache; } } } // VSWhidbey 194386 - ComboBox.PreferredHeight returns incorrect values // This is translated from windows implementation. Since we cannot control the size // of the combo box, we need to use the same calculation they do. [SuppressMessage("Microsoft.Globalization", "CA1303:DoNotPassLiteralsAsLocalizedParameters")] // "0" is what Windows uses in all languages. private int GetComboHeight() { int cyCombo = 0; // Add on CYEDGE just for some extra space in the edit field/static item. // It's really only for static text items, but we want static & editable // controls to be the same height. Size textExtent = Size.Empty; using (WindowsFont font = WindowsFont.FromFont(this.Font)) { // this is the character that Windows uses to determine the extent textExtent = WindowsGraphicsCacheManager.MeasurementGraphics.GetTextExtent("0", font); } int dyEdit = textExtent.Height + SystemInformation.Border3DSize.Height; if (DrawMode != DrawMode.Normal) { // This is an ownerdraw combo. Have the owner tell us how tall this // item is. dyEdit = ItemHeight; } // Set the initial width to be the combo box rect. Later we will shorten it // if there is a dropdown button. Size fixedFrameBoderSize = SystemInformation.FixedFrameBorderSize; cyCombo = 2 * fixedFrameBoderSize.Height + dyEdit; return cyCombo; } private string[] GetStringsForAutoComplete(IList collection) { if (collection is AutoCompleteStringCollection) { string[] strings = new string[AutoCompleteCustomSource.Count]; for (int i = 0; i < AutoCompleteCustomSource.Count; i++) { strings[i] = AutoCompleteCustomSource[i]; } return strings; } else if (collection is ObjectCollection) { string[] strings = new string[itemsCollection.Count]; for (int i = 0; i < itemsCollection.Count; i++) { strings[i] = GetItemText(itemsCollection[i]); } return strings; } return new string[0]; } ////// /// The [zero based] index of the currently selected item in the combos list. /// Note If the value of index is -1, then the ComboBox is /// set to have no selection. /// [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden), SRDescription(SR.ComboBoxSelectedIndexDescr) ] public override int SelectedIndex { get { if (IsHandleCreated) { return (int)SendMessage(NativeMethods.CB_GETCURSEL, 0, 0); } else { return selectedIndex; } } set { if (SelectedIndex != value) { int itemCount = 0; if (itemsCollection != null) { itemCount = itemsCollection.Count; } if (value < -1 || value >= itemCount) { throw new ArgumentOutOfRangeException("SelectedIndex", SR.GetString(SR.InvalidArgument, "SelectedIndex", (value).ToString(CultureInfo.CurrentCulture))); } if (IsHandleCreated) { SendMessage(NativeMethods.CB_SETCURSEL, value, 0); } else { selectedIndex = value; } UpdateText(); if (IsHandleCreated) { OnTextChanged(EventArgs.Empty); } OnSelectedItemChanged(EventArgs.Empty); OnSelectedIndexChanged(EventArgs.Empty); } } } ////// /// The handle to the object that is currently selected in the /// combos list. /// [ Browsable(false), Bindable(true), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden), SRDescription(SR.ComboBoxSelectedItemDescr) ] public object SelectedItem { get { int index = SelectedIndex; return (index == -1) ? null : Items[index]; } set { int x = -1; if (itemsCollection != null) { //bug (82115) if (value != null) x = itemsCollection.IndexOf(value); else SelectedIndex = -1; } if (x != -1) { SelectedIndex = x; } } } ////// /// The selected text in the edit component of the ComboBox. If the /// ComboBox has ComboBoxStyle.DROPDOWNLIST, the return is an empty /// string (""). /// [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden), SRDescription(SR.ComboBoxSelectedTextDescr) ] public string SelectedText { get { if (DropDownStyle == ComboBoxStyle.DropDownList) return ""; return Text.Substring(SelectionStart, SelectionLength); } set { if (DropDownStyle != ComboBoxStyle.DropDownList) { //guard against null string, since otherwise we will throw an //AccessViolation exception, which is bad string str = (value == null ? "" : value); CreateControl(); if (IsHandleCreated) { Debug.Assert(childEdit != null); if (childEdit != null) { UnsafeNativeMethods.SendMessage(new HandleRef(this, childEdit.Handle), NativeMethods.EM_REPLACESEL, NativeMethods.InvalidIntPtr, str); } } } } } ////// /// The length, in characters, of the selection in the editbox. /// [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden), SRDescription(SR.ComboBoxSelectionLengthDescr) ] public int SelectionLength { get { int[] end = new int[] { 0 }; int[] start = new int[] { 0 }; UnsafeNativeMethods.SendMessage(new HandleRef(this, Handle), NativeMethods.CB_GETEDITSEL, start, end); return end[0] - start[0]; } set { // SelectionLength can be negtive... Select(SelectionStart, value); } } ////// /// The [zero-based] index of the first character in the current text selection. /// [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden), SRDescription(SR.ComboBoxSelectionStartDescr) ] public int SelectionStart { get { int[] value = new int[] { 0 }; UnsafeNativeMethods.SendMessage(new HandleRef(this, Handle), NativeMethods.CB_GETEDITSEL, value, (int[])null); return value[0]; } set { if (value < 0) { throw new ArgumentOutOfRangeException("SelectionStart", SR.GetString(SR.InvalidArgument, "SelectionStart", value.ToString(CultureInfo.CurrentCulture))); } Select(value, SelectionLength); } } ////// /// Indicates if the Combos list is sorted or not. /// [ SRCategory(SR.CatBehavior), DefaultValue(false), SRDescription(SR.ComboBoxSortedDescr) ] public bool Sorted { get { return sorted; } set { if (sorted != value) { if (this.DataSource != null && value) { throw new ArgumentException(SR.GetString(SR.ComboBoxSortWithDataSource)); } sorted = value; RefreshItems(); SelectedIndex = -1; } } } ////// /// The type of combo that we are right now. The value would come /// from the System.Windows.Forms.ComboBoxStyle enumeration. /// [ SRCategory(SR.CatAppearance), DefaultValue(ComboBoxStyle.DropDown), SRDescription(SR.ComboBoxStyleDescr), RefreshPropertiesAttribute(RefreshProperties.Repaint) ] public ComboBoxStyle DropDownStyle { get { bool found; int style = Properties.GetInteger(PropStyle, out found); if (found) { return (ComboBoxStyle)style; } return ComboBoxStyle.DropDown; } set { if (DropDownStyle != value) { // verify that 'value' is a valid enum type... //valid values are 0x0 to 0x2 if (!ClientUtils.IsEnumValid(value, (int)value, (int)ComboBoxStyle.Simple, (int)ComboBoxStyle.DropDownList)) { throw new InvalidEnumArgumentException("value", (int)value, typeof(ComboBoxStyle)); } if (value == ComboBoxStyle.DropDownList && this.AutoCompleteSource != AutoCompleteSource.ListItems && this.AutoCompleteMode != AutoCompleteMode.None) { this.AutoCompleteMode = AutoCompleteMode.None; } // reset preferred height. ResetHeightCache(); Properties.SetInteger(PropStyle, (int)value); if (IsHandleCreated) { RecreateHandle(); } OnDropDownStyleChanged(EventArgs.Empty); } } } ////// /// [ Localizable(true), Bindable(true) ] public override string Text { get { if (SelectedItem != null && !BindingFieldEmpty) { //preserve everett behavior if "formatting enabled == false" -- just return selecteditem text. if (FormattingEnabled) { string candidate = GetItemText(SelectedItem); if (!String.IsNullOrEmpty(candidate)) { if (String.Compare(candidate, base.Text, true, CultureInfo.CurrentCulture) == 0) { return candidate; //for whidbey, if we only differ by case -- return the candidate; } } } else { return FilterItemOnProperty(SelectedItem).ToString(); //heinous. } } return base.Text; } set { if (DropDownStyle == ComboBoxStyle.DropDownList && !IsHandleCreated && !String.IsNullOrEmpty(value) && FindStringExact(value) == -1) return; base.Text = value; object selectedItem = null; selectedItem = SelectedItem; if (!DesignMode) { //bug <70650> Subhag removed 'value.Length == 0' check to handle String.Empty. // if (value == null) { SelectedIndex = -1; } else if (value != null && (selectedItem == null || (String.Compare(value, GetItemText(selectedItem), false, CultureInfo.CurrentCulture) != 0))) { int index = FindStringIgnoreCase(value); //we cannot set the index to -1 unless we want to do a hack and save/restore text //because the native control will erase the text when we change the index to -1 if (index != -1) { SelectedIndex = index; } } } } } private int FindStringIgnoreCase(string value) { //look for an exact match and then a case insensitive match if that fails. int index = FindStringExact(value, -1, false); if (index == -1) { index = FindStringExact(value, -1, true); } return index; } // Special AutoComplete notification handling // If the text changed, this will fire TextChanged // If it matches an item in the list, this will fire SIC and SVC private void NotifyAutoComplete() { NotifyAutoComplete(true); } // Special AutoComplete notification handling // If the text changed, this will fire TextChanged // If it matches an item in the list, this will fire SIC and SVC private void NotifyAutoComplete(bool setSelectedIndex) { string text = this.Text; bool textChanged = (text != this.lastTextChangedValue); bool selectedIndexSet = false; if (setSelectedIndex) { // Process return key. This is sent by the AutoComplete DropDown when a // selection is made from the DropDown // Check to see if the Text Changed. If so, at least fire a TextChanged int index = FindStringIgnoreCase(text); if ((index != -1) && (index != SelectedIndex)) { // We found a match, do the full monty SelectedIndex = index; // Put the cursor at the end SelectionStart = 0; SelectionLength = text.Length; selectedIndexSet = true; } } //don't fire textch if we had set the selectedindex -- because it was already fired if so. if (textChanged && !selectedIndexSet) { // No match, just fire a TextChagned OnTextChanged(EventArgs.Empty); } // Save the new value this.lastTextChangedValue = text; } // Returns true if using System AutoComplete private bool SystemAutoCompleteEnabled { get { return ((this.autoCompleteMode != AutoCompleteMode.None) && (this.DropDownStyle != ComboBoxStyle.DropDownList)); } } // VSWhidbey 95691: Prevent this event from being displayed in the Property Grid. ///[To be supplied.] ////// [Browsable(false), EditorBrowsable(EditorBrowsableState.Never)] public new event EventHandler DoubleClick { add { base.DoubleClick += value; } remove { base.DoubleClick -= value; } } /// /// /// [SRCategory(SR.CatBehavior), SRDescription(SR.drawItemEventDescr)] public event DrawItemEventHandler DrawItem { add { Events.AddHandler(EVENT_DRAWITEM, value); } remove { Events.RemoveHandler(EVENT_DRAWITEM, value); } } ///[To be supplied.] ////// /// [SRCategory(SR.CatBehavior), SRDescription(SR.ComboBoxOnDropDownDescr)] public event EventHandler DropDown { add { Events.AddHandler(EVENT_DROPDOWN, value); } remove { Events.RemoveHandler(EVENT_DROPDOWN, value); } } ///[To be supplied.] ////// /// [SRCategory(SR.CatBehavior), SRDescription(SR.measureItemEventDescr)] public event MeasureItemEventHandler MeasureItem { add { Events.AddHandler(EVENT_MEASUREITEM, value); UpdateItemHeight(); } remove { Events.RemoveHandler(EVENT_MEASUREITEM, value); UpdateItemHeight(); } } ///[To be supplied.] ////// /// [SRCategory(SR.CatBehavior), SRDescription(SR.selectedIndexChangedEventDescr)] public event EventHandler SelectedIndexChanged { add { Events.AddHandler(EVENT_SELECTEDINDEXCHANGED, value); } remove { Events.RemoveHandler(EVENT_SELECTEDINDEXCHANGED, value); } } ///[To be supplied.] ////// /// [SRCategory(SR.CatBehavior), SRDescription(SR.selectionChangeCommittedEventDescr)] public event EventHandler SelectionChangeCommitted { add { Events.AddHandler(EVENT_SELECTIONCHANGECOMMITTED, value); } remove { Events.RemoveHandler(EVENT_SELECTIONCHANGECOMMITTED, value); } } ///[To be supplied.] ////// /// [SRCategory(SR.CatBehavior), SRDescription(SR.ComboBoxDropDownStyleChangedDescr)] public event EventHandler DropDownStyleChanged { add { Events.AddHandler(EVENT_DROPDOWNSTYLE, value); } remove { Events.RemoveHandler(EVENT_DROPDOWNSTYLE, value); } } ///[To be supplied.] ////// /// ComboBox Onpaint. /// ///[Browsable(false), EditorBrowsable(EditorBrowsableState.Never)] public new event PaintEventHandler Paint { add { base.Paint += value; } remove { base.Paint -= value; } } /// /// /// [SRCategory(SR.CatBehavior), SRDescription(SR.ComboBoxOnTextUpdateDescr)] public event EventHandler TextUpdate { add { Events.AddHandler(EVENT_TEXTUPDATE, value); } remove { Events.RemoveHandler(EVENT_TEXTUPDATE, value); } } ///This will fire the TextUpdate Event on the ComboBox. This events fires when the Combobox gets the /// CBN_EDITUPDATE notification. // ////// /// /// [SRCategory(SR.CatBehavior), SRDescription(SR.ComboBoxOnDropDownClosedDescr)] public event EventHandler DropDownClosed { add { Events.AddHandler(EVENT_DROPDOWNCLOSED, value); } remove { Events.RemoveHandler(EVENT_DROPDOWNCLOSED, value); } } ///This will fire the DropDownClosed Event on the ComboBox. This events fires when the Combobox gets the /// CBN_CLOSEUP notification. This happens when the DropDown closes. // ////// /// /// Performs the work of adding the specified items to the combobox /// [Obsolete("This method has been deprecated. There is no replacement. http://go.microsoft.com/fwlink/?linkid=14202")] protected virtual void AddItemsCore(object[] value) { int count = value == null ? 0 : value.Length; if (count == 0) { return; } BeginUpdate(); try { Items.AddRangeInternal(value); } finally { EndUpdate(); } } ////// /// Disables redrawing of the combo box. A call to beginUpdate() must be /// balanced by a following call to endUpdate(). Following a call to /// beginUpdate(), any redrawing caused by operations performed on the /// combo box is deferred until the call to endUpdate(). /// public void BeginUpdate() { updateCount++; BeginUpdateInternal(); } private void CheckNoDataSource() { if (DataSource != null) { throw new ArgumentException(SR.GetString(SR.DataSourceLocksItems)); } } ////// protected override AccessibleObject CreateAccessibilityInstance() { return new ComboBoxAccessibleObject(this); } ////// internal bool UpdateNeeded() { return (updateCount == 0); } ////// This procedure takes in the message, converts the Edit handle coordinates into Combo Box Coordinates /// ///internal Point EditToComboboxMapping(Message m) { if (childEdit == null) { return new Point(0, 0); } // Get the Combox Rect ... // NativeMethods.RECT comboRectMid = new NativeMethods.RECT(); UnsafeNativeMethods.GetWindowRect(new HandleRef(this, Handle), ref comboRectMid); // //Get the Edit Rectangle... // NativeMethods.RECT editRectMid = new NativeMethods.RECT(); UnsafeNativeMethods.GetWindowRect(new HandleRef(this, childEdit.Handle), ref editRectMid); //get the delta int comboXMid = NativeMethods.Util.SignedLOWORD(m.LParam) + (editRectMid.left - comboRectMid.left); int comboYMid = NativeMethods.Util.SignedHIWORD(m.LParam) + (editRectMid.top - comboRectMid.top); return (new Point(comboXMid, comboYMid)); } /// /// Subclassed window procedure for the edit and list child controls of the /// combo box. /// ///private void ChildWndProc(ref Message m) { switch (m.Msg) { case NativeMethods.WM_CHAR: if (DropDownStyle == ComboBoxStyle.Simple && m.HWnd == childListBox.Handle) { DefChildWndProc(ref m); } else { if (PreProcessControlMessage(ref m) != PreProcessControlState.MessageProcessed) { if (ProcessKeyMessage(ref m)) { return; } DefChildWndProc(ref m); } } break; case NativeMethods.WM_SYSCHAR: if (DropDownStyle == ComboBoxStyle.Simple && m.HWnd == childListBox.Handle) { DefChildWndProc(ref m); } else { if (PreProcessControlMessage(ref m) != PreProcessControlState.MessageProcessed) { if (ProcessKeyEventArgs(ref m)) { return; } DefChildWndProc(ref m); } } break; case NativeMethods.WM_KEYDOWN: case NativeMethods.WM_SYSKEYDOWN: if (SystemAutoCompleteEnabled && !ACNativeWindow.AutoCompleteActive) { finder.FindDropDowns(false); } if (AutoCompleteMode != AutoCompleteMode.None) { char keyChar = unchecked((char)(long)m.WParam); if (keyChar == (char)(int)Keys.Escape) { this.DroppedDown = false; } else if (keyChar == (char)(int)Keys.Return && this.DroppedDown) { UpdateText(); OnSelectionChangeCommittedInternal(EventArgs.Empty); this.DroppedDown = false; } } if (DropDownStyle == ComboBoxStyle.Simple && m.HWnd == childListBox.Handle) { DefChildWndProc(ref m); } else { if (PreProcessControlMessage(ref m) != PreProcessControlState.MessageProcessed) { if (ProcessKeyMessage(ref m)) { return; } DefChildWndProc(ref m); } } break; case NativeMethods.WM_INPUTLANGCHANGE: DefChildWndProc( ref m ); break; case NativeMethods.WM_KEYUP: case NativeMethods.WM_SYSKEYUP: if (DropDownStyle == ComboBoxStyle.Simple && m.HWnd == childListBox.Handle) { DefChildWndProc(ref m); } else { if (PreProcessControlMessage(ref m) != PreProcessControlState.MessageProcessed) { if (ProcessKeyMessage(ref m)) { return; } DefChildWndProc(ref m); } } if (SystemAutoCompleteEnabled && !ACNativeWindow.AutoCompleteActive) { finder.FindDropDowns(); } break; case NativeMethods.WM_KILLFOCUS: // Consider - If we dont' have a childwndproc, then we don't get here, so we don't // update the cache. Do we need to? This happens when we have a DropDownList. if (!DesignMode) { OnImeContextStatusChanged( m.HWnd ); } DefChildWndProc(ref m); // We don't want to fire the focus events twice - // once in the combobox and once here. if (fireLostFocus) { OnLostFocus(EventArgs.Empty); } if (FlatStyle == FlatStyle.Popup) { this.Invalidate(); } break; case NativeMethods.WM_SETFOCUS: // Consider - If we dont' have a childwndproc, then we don't get here, so we don't // set the status. Do we need to? This happens when we have a DropDownList. if (!DesignMode) { ImeContext.SetImeStatus(CachedImeMode, m.HWnd); } if (!HostedInWin32DialogManager) { IContainerControl c = GetContainerControlInternal(); if (c != null) { ContainerControl container = c as ContainerControl; if (container != null) { if (!container.ActivateControlInternal(this, false)) { return; } } } } DefChildWndProc(ref m); // We don't want to fire the focus events twice - // once in the combobox and once here. if (fireSetFocus) { OnGotFocus(EventArgs.Empty); } if (FlatStyle == FlatStyle.Popup) { this.Invalidate(); } break; case NativeMethods.WM_SETFONT: DefChildWndProc(ref m); if (childEdit != null && m.HWnd == childEdit.Handle) { UnsafeNativeMethods.SendMessage(new HandleRef(this, childEdit.Handle), NativeMethods.EM_SETMARGINS, NativeMethods.EC_LEFTMARGIN | NativeMethods.EC_RIGHTMARGIN, 0); } break; case NativeMethods.WM_LBUTTONDBLCLK: //the Listbox gets WM_LBUTTONDOWN - WM_LBUTTONUP -WM_LBUTTONDBLCLK - WM_LBUTTONUP... //sequence for doubleclick... //Set MouseEvents... mousePressed = true; mouseEvents = true; CaptureInternal = true; //Call the DefWndProc() so that mousemove messages get to the windows edit(112079) // DefChildWndProc(ref m); //the up gets fired from "Combo-box's WndPrc --- So Convert these Coordinates to Combobox coordianate... // Point Ptlc = EditToComboboxMapping(m); OnMouseDown(new MouseEventArgs(MouseButtons.Left, 1, Ptlc.X, Ptlc.Y, 0)); break; case NativeMethods.WM_MBUTTONDBLCLK: //the Listbox gets WM_LBUTTONDOWN - WM_LBUTTONUP -WM_LBUTTONDBLCLK - WM_LBUTTONUP... //sequence for doubleclick... //Set MouseEvents... mousePressed = true; mouseEvents = true; CaptureInternal = true; //Call the DefWndProc() so that mousemove messages get to the windows edit(112079) // DefChildWndProc(ref m); //the up gets fired from "Combo-box's WndPrc --- So Convert these Coordinates to Combobox coordianate... // Point Ptmc = EditToComboboxMapping(m); OnMouseDown(new MouseEventArgs(MouseButtons.Middle, 1, Ptmc.X, Ptmc.Y, 0)); break; case NativeMethods.WM_RBUTTONDBLCLK: //the Listbox gets WM_LBUTTONDOWN - WM_LBUTTONUP -WM_LBUTTONDBLCLK - WM_LBUTTONUP... //sequence for doubleclick... //Set MouseEvents... mousePressed = true; mouseEvents = true; CaptureInternal = true; //Call the DefWndProc() so that mousemove messages get to the windows edit(112079) // DefChildWndProc(ref m); //the up gets fired from "Combo-box's WndPrc --- So Convert these Coordinates to Combobox coordianate... // Point Ptrc = EditToComboboxMapping(m); OnMouseDown(new MouseEventArgs(MouseButtons.Right, 1, Ptrc.X, Ptrc.Y, 0)); break; case NativeMethods.WM_LBUTTONDOWN: mousePressed = true; mouseEvents = true; //set the mouse capture .. this is the Child Wndproc.. // CaptureInternal = true; DefChildWndProc(ref m); //the up gets fired from "Combo-box's WndPrc --- So Convert these Coordinates to Combobox coordianate... // Point Ptl = EditToComboboxMapping(m); OnMouseDown(new MouseEventArgs(MouseButtons.Left, 1, Ptl.X, Ptl.Y, 0)); break; case NativeMethods.WM_LBUTTONUP: // Get the mouse location // NativeMethods.RECT r = new NativeMethods.RECT(); UnsafeNativeMethods.GetWindowRect(new HandleRef(this, Handle), ref r); Rectangle ClientRect = new Rectangle(r.left, r.top, r.right - r.left, r.bottom - r.top); // Get the mouse location // int x = NativeMethods.Util.SignedLOWORD(m.LParam); int y = NativeMethods.Util.SignedHIWORD(m.LParam); Point pt = new Point(x, y); pt = PointToScreen(pt); // combo box gets a WM_LBUTTONUP for focus change ... // So check MouseEvents.... if (mouseEvents && !ValidationCancelled) { mouseEvents = false; if (mousePressed) { if (ClientRect.Contains(pt)) { mousePressed = false; OnClick(new MouseEventArgs(MouseButtons.Left, 1, NativeMethods.Util.SignedLOWORD(m.LParam), NativeMethods.Util.SignedHIWORD(m.LParam), 0)); OnMouseClick(new MouseEventArgs(MouseButtons.Left, 1, NativeMethods.Util.SignedLOWORD(m.LParam), NativeMethods.Util.SignedHIWORD(m.LParam), 0)); } else { mousePressed = false; mouseInEdit = false; OnMouseLeave(EventArgs.Empty); } } } DefChildWndProc(ref m); CaptureInternal = false; //the up gets fired from "Combo-box's WndPrc --- So Convert these Coordinates to Combobox coordianate... // pt = EditToComboboxMapping(m); OnMouseUp(new MouseEventArgs(MouseButtons.Left, 1, pt.X, pt.Y, 0)); break; case NativeMethods.WM_MBUTTONDOWN: mousePressed = true; mouseEvents = true; //set the mouse capture .. this is the Child Wndproc.. // CaptureInternal = true; DefChildWndProc(ref m); //the up gets fired from "Combo-box's WndPrc --- So Convert these Coordinates to Combobox coordianate... // Point P = EditToComboboxMapping(m); OnMouseDown(new MouseEventArgs(MouseButtons.Middle, 1, P.X, P.Y, 0)); break; case NativeMethods.WM_RBUTTONDOWN: mousePressed = true; mouseEvents = true; //set the mouse capture .. this is the Child Wndproc.. // Bug# 112108: If I set the capture=true here, the // DefWndProc() never fires the WM_CONTEXTMENU that would show // the default context menu. // if (this.ContextMenu != null || this.ContextMenuStrip != null) CaptureInternal = true; DefChildWndProc(ref m); //the up gets fired from "Combo-box's WndPrc --- So Convert these Coordinates to Combobox coordianate... // Point Pt = EditToComboboxMapping(m); OnMouseDown(new MouseEventArgs(MouseButtons.Right, 1, Pt.X, Pt.Y, 0)); break; case NativeMethods.WM_MBUTTONUP: mousePressed = false; mouseEvents = false; //set the mouse capture .. this is the Child Wndproc.. // CaptureInternal = false; DefChildWndProc(ref m); OnMouseUp(new MouseEventArgs(MouseButtons.Middle, 1, NativeMethods.Util.SignedLOWORD(m.LParam), NativeMethods.Util.SignedHIWORD(m.LParam), 0)); break; case NativeMethods.WM_RBUTTONUP: mousePressed = false; mouseEvents = false; //set the mouse capture .. this is the Child Wndproc.. // if (this.ContextMenu != null) CaptureInternal = false; DefChildWndProc(ref m); //the up gets fired from "Combo-box's WndPrc --- So Convert these Coordinates to Combobox coordianate... // Point ptRBtnUp = EditToComboboxMapping(m); OnMouseUp(new MouseEventArgs(MouseButtons.Right, 1, ptRBtnUp.X, ptRBtnUp.Y, 0)); break; case NativeMethods.WM_CONTEXTMENU: // Forward context menu messages to the parent control if (this.ContextMenu != null || this.ContextMenuStrip != null) { UnsafeNativeMethods.SendMessage(new HandleRef(this, Handle), NativeMethods.WM_CONTEXTMENU, m.WParam, m.LParam); } else { DefChildWndProc(ref m); } break; case NativeMethods.WM_MOUSEMOVE: Point point = EditToComboboxMapping(m); //Call the DefWndProc() so that mousemove messages get to the windows edit(112079) // DefChildWndProc(ref m); OnMouseEnterInternal(EventArgs.Empty); OnMouseMove(new MouseEventArgs(MouseButtons, 0, point.X, point.Y, 0)); break; case NativeMethods.WM_SETCURSOR: if (Cursor != DefaultCursor && childEdit != null && m.HWnd == childEdit.Handle && NativeMethods.Util.LOWORD(m.LParam) == NativeMethods.HTCLIENT) { Cursor.CurrentInternal = Cursor; } else { DefChildWndProc(ref m); } break; case NativeMethods.WM_MOUSELEAVE: DefChildWndProc(ref m); OnMouseLeaveInternal(EventArgs.Empty); break; default: DefChildWndProc(ref m); break; } } /// /// Helper to handle MouseEnter. /// /// private void OnMouseEnterInternal(EventArgs args) { if (!mouseInEdit) { OnMouseEnter(args); mouseInEdit = true; } } ////// Helper to handle mouseleave /// /// private void OnMouseLeaveInternal(EventArgs args) { NativeMethods.RECT rect = new NativeMethods.RECT(); UnsafeNativeMethods.GetWindowRect(new HandleRef(this, Handle), ref rect); Rectangle Rect = new Rectangle(rect.left, rect.top, rect.right - rect.left, rect.bottom - rect.top); Point p = MousePosition; if (!Rect.Contains(p)) { OnMouseLeave(args); mouseInEdit = false; } } ////// ///private void DefChildWndProc(ref Message m) { if (childEdit != null) { NativeWindow childWindow = (m.HWnd == childEdit.Handle) ? childEdit : childListBox; //childwindow could be null if the handle was recreated while within a message handler // and then whoever recreated the handle allowed the message to continue to be processed //we cannot really be sure the new child will properly handle this window message, so we eat it. if (childWindow != null) { childWindow.DefWndProc(ref m); } } } /// protected override void Dispose(bool disposing) { if (disposing) { if (autoCompleteCustomSource != null) { autoCompleteCustomSource.CollectionChanged -= new CollectionChangeEventHandler(this.OnAutoCompleteCustomSourceChanged); } if (stringSource != null) { stringSource.ReleaseAutoComplete(); stringSource = null; } } base.Dispose(disposing); } /// /// /// Reenables redrawing of the combo box. A call to beginUpdate() must be /// balanced by a following call to endUpdate(). Following a call to /// beginUpdate(), any redrawing caused by operations performed on the /// combo box is deferred until the call to endUpdate(). /// public void EndUpdate() { updateCount--; if (updateCount == 0 && AutoCompleteSource == AutoCompleteSource.ListItems) { SetAutoComplete(false, false); } if (EndUpdateInternal()) { if (childEdit != null && childEdit.Handle != IntPtr.Zero) { SafeNativeMethods.InvalidateRect(new HandleRef(this, childEdit.Handle), null, false); } if (childListBox != null && childListBox.Handle != IntPtr.Zero) { SafeNativeMethods.InvalidateRect(new HandleRef(this, childListBox.Handle), null, false); } } } ////// /// Finds the first item in the combo box that starts with the given string. /// The search is not case sensitive. /// public int FindString(string s) { return FindString(s, -1); } ////// /// Finds the first item after the given index which starts with the given /// string. The search is not case sensitive. /// public int FindString(string s, int startIndex) { if (s == null) { return -1; } if (itemsCollection == null || itemsCollection.Count == 0) { return -1; } // VSWhidbey 95158: The last item in the list is still a valid starting point for a search. if (startIndex < -1 || startIndex >= itemsCollection.Count) { throw new ArgumentOutOfRangeException("startIndex"); } // Always use the managed FindStringInternal instead of CB_FINDSTRING. // The managed version correctly handles Turkish I. // return FindStringInternal(s, Items, startIndex, false); } ////// /// Finds the first item in the combo box that matches the given string. /// The strings must match exactly, except for differences in casing. /// public int FindStringExact(string s) { return FindStringExact(s, -1, true); } ////// /// Finds the first item after the given index that matches the given /// string. The strings must match exactly, except for differences in /// casing. /// public int FindStringExact(string s, int startIndex) { return FindStringExact(s, startIndex, true); } ////// /// Finds the first item after the given index that matches the given /// string. The strings must match exactly, except for differences in /// casing. /// internal int FindStringExact(string s, int startIndex, bool ignorecase) { if (s == null) return -1; if (itemsCollection == null || itemsCollection.Count == 0) { return -1; } // VSWhidbey 95158: The last item in the list is still a valid starting point for a search. if (startIndex < -1 || startIndex >= itemsCollection.Count) { throw new ArgumentOutOfRangeException("startIndex"); } // Always use the managed FindStringInternal instead of CB_FINDSTRINGEXACT. // The managed version correctly handles Turkish I. // return FindStringInternal(s, Items, startIndex, true, ignorecase); } // GetPreferredSize and SetBoundsCore call this method to allow controls to self impose // constraints on their size. internal override Rectangle ApplyBoundsConstraints(int suggestedX, int suggestedY, int proposedWidth, int proposedHeight) { if (DropDownStyle == ComboBoxStyle.DropDown || DropDownStyle == ComboBoxStyle.DropDownList) { proposedHeight = PreferredHeight; } return base.ApplyBoundsConstraints(suggestedX,suggestedY, proposedWidth, proposedHeight); } [SuppressMessage("Microsoft.Portability", "CA1902:AvoidTestingForFloatingPointEquality")] protected override void ScaleControl(SizeF factor, BoundsSpecified specified) { if (factor.Width != 1F && factor.Height != 1F) { // VSWhidbey 440275 // we get called on scale before we get a notification that our font has changed. // in this case, we need to reset our height cache. ResetHeightCache(); } base.ScaleControl(factor, specified); } ////// /// Returns the height of the given item in an OwnerDrawVariable style /// combo box. This method should not be used for Normal or OwnerDrawFixed /// style combo boxes. /// public int GetItemHeight(int index) { // This function is only relevant for OwnerDrawVariable if (DrawMode != DrawMode.OwnerDrawVariable) { return ItemHeight; } if (index < 0 || itemsCollection == null || index >= itemsCollection.Count) { throw new ArgumentOutOfRangeException("index", SR.GetString(SR.InvalidArgument, "index", (index).ToString(CultureInfo.CurrentCulture))); } if (IsHandleCreated) { int h = (int)SendMessage(NativeMethods.CB_GETITEMHEIGHT, index, 0); if (h == -1) { throw new Win32Exception(); } return h; } return ItemHeight; } internal override IntPtr InitializeDCForWmCtlColor(IntPtr dc, int msg) { if ((msg == NativeMethods.WM_CTLCOLORSTATIC) && !ShouldSerializeBackColor()) { // Let the Win32 Edit control handle background colors itself. // This is necessary because a disabled edit control will display a different // BackColor than when enabled. return IntPtr.Zero; } else if ((msg == NativeMethods.WM_CTLCOLORLISTBOX) && GetStyle(ControlStyles.UserPaint)) { // VSWhidbey#93929. Base class returns hollow brush when UserPaint style is set, to avoid flicker in // main control. But when returning colors for child dropdown list, return normal ForeColor/BackColor, // since hollow brush leaves the list background unpainted. SafeNativeMethods.SetTextColor(new HandleRef(null, dc), ColorTranslator.ToWin32(ForeColor)); SafeNativeMethods.SetBkColor(new HandleRef(null, dc), ColorTranslator.ToWin32(BackColor)); return BackColorBrush; } else { return base.InitializeDCForWmCtlColor(dc, msg); } } // Returns true when the key processing needs to be intercepted to allow // auto-completion in DropDownList style. private bool InterceptAutoCompleteKeystroke(Message m) { if (m.Msg == NativeMethods.WM_KEYDOWN) { Debug.Assert((ModifierKeys & Keys.Alt) == 0); // Keys.Delete only triggers a WM_KEYDOWN and WM_KEYUP, and no WM_CHAR. That's why it's treated separately. if ((Keys)(int)m.WParam == Keys.Delete) { // Reset matching text and remove any selection this.MatchingText = ""; this.autoCompleteTimeStamp = DateTime.Now.Ticks; if (this.Items.Count > 0) { SelectedIndex = 0; } return false; } } else if (m.Msg == NativeMethods.WM_CHAR) { Debug.Assert((ModifierKeys & Keys.Alt) == 0); char keyChar = unchecked((char)(long)m.WParam); if (keyChar == (char)Keys.Back) { if (DateTime.Now.Ticks - this.autoCompleteTimeStamp > AutoCompleteTimeout || this.MatchingText.Length <= 1) { // Reset matching text and remove any selection this.MatchingText = ""; if (this.Items.Count > 0) { SelectedIndex = 0; } } else { // Remove one character from matching text and rematch this.MatchingText = this.MatchingText.Remove(this.MatchingText.Length - 1); this.SelectedIndex = FindString(this.MatchingText); } this.autoCompleteTimeStamp = DateTime.Now.Ticks; return false; } else if (keyChar == (char)Keys.Escape) { this.MatchingText = ""; } string newMatchingText; if (keyChar != (char)Keys.Escape && keyChar != (char)Keys.Return && !DroppedDown && AutoCompleteMode != AutoCompleteMode.Append) { DroppedDown = true; } if (DateTime.Now.Ticks - this.autoCompleteTimeStamp > AutoCompleteTimeout) { newMatchingText = new string(keyChar, 1); if (FindString(newMatchingText) != -1) { this.MatchingText = newMatchingText; // Select the found item } this.autoCompleteTimeStamp = DateTime.Now.Ticks; return false; } else { newMatchingText = this.MatchingText + keyChar; int itemFound = FindString(newMatchingText); if (itemFound != -1) { this.MatchingText = newMatchingText; if (itemFound != this.SelectedIndex) { this.SelectedIndex = itemFound; } } // Do not change the selection this.autoCompleteTimeStamp = DateTime.Now.Ticks; return true; } } return false; } // Invalidate the entire control, including child HWNDs and non-client areas private void InvalidateEverything() { SafeNativeMethods.RedrawWindow(new HandleRef(this, Handle), null, NativeMethods.NullHandleRef, NativeMethods.RDW_INVALIDATE | NativeMethods.RDW_FRAME | // Control.Invalidate(true) doesn't invalidate the non-client region NativeMethods.RDW_ERASE | NativeMethods.RDW_ALLCHILDREN); } ////// /// Determines if keyData is in input key that the control wants. /// Overridden to return true for RETURN and ESCAPE when the combo box is /// dropped down. /// ///protected override bool IsInputKey(Keys keyData) { Keys keyCode = keyData & (Keys.KeyCode | Keys.Alt); if (keyCode == Keys.Return || keyCode == Keys.Escape) { if (this.DroppedDown || autoCompleteDroppedDown) { //old behavior return true; } else if (SystemAutoCompleteEnabled && ACNativeWindow.AutoCompleteActive) { autoCompleteDroppedDown = true; return true; } } return base.IsInputKey(keyData); } /// /// Adds the given item to the native combo box. This asserts if the handle hasn't been /// created. /// private int NativeAdd(object item) { Debug.Assert(IsHandleCreated, "Shouldn't be calling Native methods before the handle is created."); int insertIndex = (int)SendMessage(NativeMethods.CB_ADDSTRING, 0, GetItemText(item)); if (insertIndex < 0) { throw new OutOfMemoryException(SR.GetString(SR.ComboBoxItemOverflow)); } return insertIndex; } ////// Clears the contents of the combo box. /// private void NativeClear() { Debug.Assert(IsHandleCreated, "Shouldn't be calling Native methods before the handle is created."); string saved = null; if (DropDownStyle != ComboBoxStyle.DropDownList) { saved = WindowText; } SendMessage(NativeMethods.CB_RESETCONTENT, 0, 0); if (saved != null) { WindowText = saved; } } ////// Get the text stored by the native control for the specified list item. /// private string NativeGetItemText(int index) { int len = (int)SendMessage(NativeMethods.CB_GETLBTEXTLEN, index, 0); StringBuilder sb = new StringBuilder(len + 1); UnsafeNativeMethods.SendMessage(new HandleRef(this, Handle), NativeMethods.CB_GETLBTEXT, index, sb); return sb.ToString(); } ////// Inserts the given item to the native combo box at the index. This asserts if the handle hasn't been /// created or if the resulting insert index doesn't match the passed in index. /// private int NativeInsert(int index, object item) { Debug.Assert(IsHandleCreated, "Shouldn't be calling Native methods before the handle is created."); int insertIndex = (int)SendMessage(NativeMethods.CB_INSERTSTRING, index, GetItemText(item)); if (insertIndex < 0) { throw new OutOfMemoryException(SR.GetString(SR.ComboBoxItemOverflow)); } Debug.Assert(insertIndex == index, "NativeComboBox inserted at " + insertIndex + " not the requested index of " + index); return insertIndex; } ////// Removes the native item from the given index. /// private void NativeRemoveAt(int index) { Debug.Assert(IsHandleCreated, "Shouldn't be calling Native methods before the handle is created."); // Windows combo does not invalidate the selected region if you remove the // currently selected item. Test for this and invalidate. Note that because // invalidate will lazy-paint we can actually invalidate before we send the // delete message. // if (DropDownStyle == ComboBoxStyle.DropDownList && SelectedIndex == index) { Invalidate(); } SendMessage(NativeMethods.CB_DELETESTRING, index, 0); } internal override void RecreateHandleCore() { string oldText = WindowText; base.RecreateHandleCore(); if (!String.IsNullOrEmpty(oldText) && String.IsNullOrEmpty(WindowText)) { WindowText = oldText; //restore the window text } } ////// /// Overridden to avoid multiple layouts during handle creation due to combobox size change /// see bug 458948 /// ///protected override void CreateHandle() { using (new LayoutTransaction(ParentInternal, this, PropertyNames.Bounds)) { base.CreateHandle(); } } /// /// /// Overridden to make sure all the items and styles get set up correctly. /// Inheriting classes should not forget to call /// base.OnHandleCreated() /// ///protected override void OnHandleCreated(EventArgs e) { base.OnHandleCreated(e); if (MaxLength > 0) { SendMessage(NativeMethods.CB_LIMITTEXT, MaxLength, 0); } // Get the handles and wndprocs of the ComboBox's child windows // Debug.Assert(childEdit == null, "Child edit window already attached"); Debug.Assert(childListBox == null, "Child listbox window already attached"); bool ok = childEdit == null && childListBox == null; if (ok && DropDownStyle != ComboBoxStyle.DropDownList) { IntPtr hwnd = UnsafeNativeMethods.GetWindow(new HandleRef(this, Handle), NativeMethods.GW_CHILD); if (hwnd != IntPtr.Zero) { // if it's a simple dropdown list, the first HWND is the list box. // if (DropDownStyle == ComboBoxStyle.Simple) { childListBox = new ComboBoxChildNativeWindow(this); childListBox.AssignHandle(hwnd); // get the edits hwnd... // hwnd = UnsafeNativeMethods.GetWindow(new HandleRef(this, hwnd), NativeMethods.GW_HWNDNEXT); } childEdit = new ComboBoxChildNativeWindow(this); childEdit.AssignHandle(hwnd); // set the initial margin for combobox to be zero (this is also done whenever the font is changed). // UnsafeNativeMethods.SendMessage(new HandleRef(this, childEdit.Handle), NativeMethods.EM_SETMARGINS, NativeMethods.EC_LEFTMARGIN | NativeMethods.EC_RIGHTMARGIN, 0); } } bool found; int dropDownWidth = Properties.GetInteger(PropDropDownWidth, out found); if (found) { SendMessage(NativeMethods.CB_SETDROPPEDWIDTH, dropDownWidth, 0); } found = false; int itemHeight = Properties.GetInteger(PropItemHeight, out found); if (found) { // someone has set the item height - update it UpdateItemHeight(); } // Resize a simple style combobox on handle creation // to respect the requested height. // if (DropDownStyle == ComboBoxStyle.Simple) { Height = requestedHeight; } //If HandleCreated set the AutoComplete... //this function checks if the correct properties are set to enable AutoComplete feature on combobox. try { fromHandleCreate = true; SetAutoComplete(false, false); } finally { fromHandleCreate = false; } if (itemsCollection != null) { foreach (object item in itemsCollection) { NativeAdd(item); } // Now udpate the current selection. // if (selectedIndex >= 0) { SendMessage(NativeMethods.CB_SETCURSEL, selectedIndex, 0); UpdateText(); selectedIndex = -1; } } // NOTE: Setting SelectedIndex must be the last thing we do. See ASURT 73949. } /// /// /// We need to un-subclasses everything here. Inheriting classes should /// not forget to call base.OnHandleDestroyed() /// ///protected override void OnHandleDestroyed(EventArgs e) { dropDownHandle = IntPtr.Zero; if (Disposing) { itemsCollection = null; selectedIndex = -1; } else { selectedIndex = SelectedIndex; } if (stringSource != null) { stringSource.ReleaseAutoComplete(); stringSource = null; } base.OnHandleDestroyed(e); } /// /// /// This is the code that actually fires the drawItem event. Don't /// forget to call base.onDrawItem() to ensure that drawItem events /// are correctly fired at all other times. /// protected virtual void OnDrawItem(DrawItemEventArgs e) { DrawItemEventHandler handler = (DrawItemEventHandler)Events[EVENT_DRAWITEM]; if (handler != null) handler(this, e); } ////// /// This is the code that actually fires the dropDown event. Don't /// forget to call base.onDropDown() to ensure that dropDown events /// are correctly fired at all other times. /// protected virtual void OnDropDown(EventArgs e) { EventHandler handler = (EventHandler)Events[EVENT_DROPDOWN]; if (handler != null) handler(this, e); } ////// /// [EditorBrowsable(EditorBrowsableState.Advanced)] protected override void OnKeyDown(KeyEventArgs e) { // Do Return/ESC handling if (SystemAutoCompleteEnabled) { if (e.KeyCode == Keys.Return) { // Set SelectedIndex NotifyAutoComplete(true); } else if ((e.KeyCode == Keys.Escape) && autoCompleteDroppedDown) { // Fire TextChanged Only NotifyAutoComplete(false); } autoCompleteDroppedDown = false; } // Base Handling base.OnKeyDown(e); } ///Raises the ///event. /// /// Key press event handler. Overridden to close up the combo box when the /// user presses RETURN or ESCAPE. /// ///protected override void OnKeyPress(KeyPressEventArgs e) { base.OnKeyPress(e); //return when dropped down already fires commit. if (!e.Handled && (e.KeyChar == (char)(int)Keys.Return || e.KeyChar == (char)(int)Keys.Escape) && DroppedDown) { dropDown = false; if (FormattingEnabled) { //Set the Text which would Compare the WindowText with the TEXT and change SelectedIndex. Text = WindowText; SelectAll(); e.Handled = false; } else { DroppedDown = false; e.Handled = true; } } } /// /// /// This is the code that actually fires the measuereItem event. Don't /// forget to call base.onMeasureItem() to ensure that measureItem /// events are correctly fired at all other times. /// protected virtual void OnMeasureItem(MeasureItemEventArgs e) { MeasureItemEventHandler handler = (MeasureItemEventHandler)Events[EVENT_MEASUREITEM]; if (handler != null) handler(this, e); } ////// /// If we have the style set to popup show mouse over /// protected override void OnMouseEnter(EventArgs e) { base.OnMouseEnter(e); MouseIsOver = true; } ////// /// If we have the style set to popup show mouse over /// protected override void OnMouseLeave(EventArgs e) { base.OnMouseLeave(e); MouseIsOver = false; } ////// This internal helper allows us to call the committed function multiple times without worrying about double firing. /// private void OnSelectionChangeCommittedInternal(EventArgs e) { //There are cases where if we disable the combo while in this event handler, it sends the message again. //This is a recursion gaurd to ensure we only send one commit per user action. if (allowCommit) { try { allowCommit = false; OnSelectionChangeCommitted(e); } finally { allowCommit = true; } } } ////// /// This is the code that actually fires the SelectionChangeCommitted event. /// Don't forget to call base.OnSelectionChangeCommitted() to ensure /// that SelectionChangeCommitted events are correctly fired at all other times. /// protected virtual void OnSelectionChangeCommitted(EventArgs e) { EventHandler handler = (EventHandler)Events[EVENT_SELECTIONCHANGECOMMITTED]; if (handler != null) handler(this, e); } ////// /// This is the code that actually fires the selectedIndexChanged event. /// Don't forget to call base.onSelectedIndexChanged() to ensure /// that selectedIndexChanged events are correctly fired at all other times. /// protected override void OnSelectedIndexChanged(EventArgs e) { base.OnSelectedIndexChanged(e); EventHandler handler = (EventHandler)Events[EVENT_SELECTEDINDEXCHANGED]; if (handler != null) handler(this, e); // set the position in the dataSource, if there is any // we will only set the position in the currencyManager if it is different // from the SelectedIndex. Setting CurrencyManager::Position (even w/o changing it) // calls CurrencyManager::EndCurrentEdit, and that will pull the dataFrom the controls // into the backEnd. We do not need to do that. // // don't change the position if SelectedIndex is -1 because this indicates a selection not from the list. if (this.DataManager != null && DataManager.Position != SelectedIndex) { //read this as "if everett or (whidbey and selindex is valid)" if (!FormattingEnabled || SelectedIndex != -1) { this.DataManager.Position = this.SelectedIndex; } } } ///protected override void OnSelectedValueChanged(EventArgs e) { base.OnSelectedValueChanged(e); selectedValueChangedFired = true; } /// /// /// This is the code that actually fires the selectedItemChanged event. /// Don't forget to call base.onSelectedItemChanged() to ensure /// that selectedItemChanged events are correctly fired at all other times. /// protected virtual void OnSelectedItemChanged(EventArgs e) { EventHandler handler = (EventHandler)Events[EVENT_SELECTEDITEMCHANGED]; if (handler != null) handler(this, e); } ////// /// This is the code that actually fires the DropDownStyleChanged event. /// protected virtual void OnDropDownStyleChanged(EventArgs e) { EventHandler handler = (EventHandler)Events[EVENT_DROPDOWNSTYLE]; if (handler != null) handler(this, e); } ////// /// This method is called by the parent control when any property /// changes on the parent. This can be overriden by inheriting /// classes, however they must call base.OnParentPropertyChanged. /// protected override void OnParentBackColorChanged(EventArgs e) { base.OnParentBackColorChanged(e); if (DropDownStyle == ComboBoxStyle.Simple) Invalidate(); } ////// /// Indicates that a critical property, such as color or font has /// changed. /// ///protected override void OnFontChanged(EventArgs e) { base.OnFontChanged(e); ResetHeightCache(); if (this.AutoCompleteMode == AutoCompleteMode.None) { UpdateControl(true); } else { //we always will recreate the handle when autocomplete mode is on RecreateHandle(); } CommonProperties.xClearPreferredSizeCache(this); } private void OnAutoCompleteCustomSourceChanged(object sender, CollectionChangeEventArgs e) { if (AutoCompleteSource == AutoCompleteSource.CustomSource) { if (AutoCompleteCustomSource.Count == 0) { SetAutoComplete(true, true /*recreate handle*/); } else { SetAutoComplete(true, false); } } } /// /// /// Indicates that a critical property, such as color or font has /// changed. /// ///protected override void OnBackColorChanged(EventArgs e) { base.OnBackColorChanged(e); UpdateControl(false); } /// /// /// Indicates that a critical property, such as color or font has /// changed. /// ///protected override void OnForeColorChanged(EventArgs e) { base.OnForeColorChanged(e); UpdateControl(false); } /// /// [EditorBrowsable(EditorBrowsableState.Advanced)] protected override void OnGotFocus(EventArgs e) { if (!canFireLostFocus) { base.OnGotFocus(e); canFireLostFocus = true; } } /// /// [EditorBrowsable(EditorBrowsableState.Advanced)] protected override void OnLostFocus(EventArgs e) { if (canFireLostFocus) { if (this.AutoCompleteMode != AutoCompleteMode.None && this.AutoCompleteSource == AutoCompleteSource.ListItems && this.DropDownStyle == ComboBoxStyle.DropDownList) { this.MatchingText = ""; } base.OnLostFocus(e); canFireLostFocus = false; } } /// /// /// [EditorBrowsable(EditorBrowsableState.Advanced)] protected override void OnTextChanged(EventArgs e) { if (SystemAutoCompleteEnabled) { string text = this.Text; // Prevent multiple TextChanges... if (text != this.lastTextChangedValue) { // Need to still fire a TextChanged base.OnTextChanged(e); // Save the new value this.lastTextChangedValue = text; } } else { // Call the base base.OnTextChanged(e); } } ///[To be supplied.] ////// /// [EditorBrowsable(EditorBrowsableState.Advanced)] protected override void OnValidating(CancelEventArgs e) { if (SystemAutoCompleteEnabled) { // Handle AutoComplete notification NotifyAutoComplete(); } // Call base base.OnValidating(e); } private void UpdateControl(bool recreate) { //clear the pref height cache ResetHeightCache(); if (IsHandleCreated) { if (DropDownStyle == ComboBoxStyle.Simple && recreate) { // Control forgets to add a scrollbar. See ASURT 19113. RecreateHandle(); } else { UpdateItemHeight(); // Force everything to repaint. See ASURT 19113. InvalidateEverything(); } } } ///Raises the ////// event. /// /// protected override void OnResize(EventArgs e) { base.OnResize(e); if (DropDownStyle == ComboBoxStyle.Simple) { // simple style combo boxes have more painting problems than you can shake a stick at (#19113) InvalidateEverything(); } } ///Raises the ///event. protected override void OnDataSourceChanged(EventArgs e) { if (Sorted) { if (DataSource != null && Created) { // we will only throw the exception when the control is already on the form. Debug.Assert(DisplayMember.Equals(String.Empty), "if this list is sorted it means that dataSource was null when Sorted first became true. at that point DisplayMember had to be String.Empty"); DataSource = null; throw new InvalidOperationException(SR.GetString(SR.ComboBoxDataSourceWithSort)); } } if (DataSource == null) { BeginUpdate(); SelectedIndex = -1; Items.ClearInternal(); EndUpdate(); } if (!Sorted && Created) base.OnDataSourceChanged(e); RefreshItems(); } /// protected override void OnDisplayMemberChanged(EventArgs e) { base.OnDisplayMemberChanged(e); // bug 63005: when we change the displayMember, we need to refresh the // items collection // RefreshItems(); } /// /// /// This event is fired when the dropdown portion of the combobox is hidden. /// protected virtual void OnDropDownClosed(EventArgs e) { EventHandler handler = (EventHandler)Events[EVENT_DROPDOWNCLOSED]; if (handler != null) handler(this, e); } ////// /// This event is fired when the edit portion of a combobox is about to display altered text. /// This event is NOT fired if the TEXT property is programatically changed. /// protected virtual void OnTextUpdate(EventArgs e) { EventHandler handler = (EventHandler)Events[EVENT_TEXTUPDATE]; if (handler != null) handler(this, e); } ///[ SecurityPermission(SecurityAction.LinkDemand, Flags = SecurityPermissionFlag.UnmanagedCode), ] protected override bool ProcessKeyEventArgs(ref Message m) { if (this.AutoCompleteMode != AutoCompleteMode.None && this.AutoCompleteSource == AutoCompleteSource.ListItems && this.DropDownStyle == ComboBoxStyle.DropDownList && InterceptAutoCompleteKeystroke(m)) { return true; } else { return base.ProcessKeyEventArgs(ref m); } } private void ResetHeightCache() { prefHeightCache = -1; } /// /// /// Reparses the objects, getting new text strings for them. /// ///protected override void RefreshItems() { // Save off the selection and the current collection. // int selectedIndex = SelectedIndex; ObjectCollection savedItems = itemsCollection; itemsCollection = null; object[] newItems = null; // if we have a dataSource and a DisplayMember, then use it // to populate the Items collection // if (this.DataManager != null && this.DataManager.Count != -1) { newItems = new object[this.DataManager.Count]; for (int i = 0; i < newItems.Length; i++) { newItems[i] = this.DataManager[i]; } } else if (savedItems != null) { newItems = new object[savedItems.Count]; savedItems.CopyTo(newItems, 0); } BeginUpdate(); try { // Clear the items. // if (IsHandleCreated) { NativeClear(); } // Store the current list of items // if (newItems != null) { Items.AddRangeInternal(newItems); } if (this.DataManager != null) { // put the selectedIndex in [....] w/ the position in the dataManager this.SelectedIndex = this.DataManager.Position; } else { SelectedIndex = selectedIndex; } } finally { EndUpdate(); } } /// /// /// Reparses the object at the given index, getting new text string for it. /// ///protected override void RefreshItem(int index) { Items.SetItemInternal(index, Items[index]); } /// /// Release the ChildWindow object by un-subclassing the child edit and /// list controls and freeing the root of the ChildWindow object. /// private void ReleaseChildWindow() { if (childEdit != null) { childEdit.ReleaseHandle(); childEdit = null; } if (childListBox != null) { childListBox.ReleaseHandle(); childListBox = null; } } private void ResetAutoCompleteCustomSource() { AutoCompleteCustomSource = null; } private void ResetDropDownWidth() { Properties.RemoveInteger(PropDropDownWidth); } private void ResetItemHeight() { Properties.RemoveInteger(PropItemHeight); } public override void ResetText() { base.ResetText(); } ////// /// Enables the AutoComplete feature for combobox depending on the properties set. /// These properties are namely AutoCompleteMode, AutoCompleteSource and AutoCompleteCustomSource. /// [SuppressMessage("Microsoft.Security", "CA2122:DoNotIndirectlyExposeMethodsWithLinkDemands")] private void SetAutoComplete(bool reset, bool recreate) { if (!IsHandleCreated || childEdit == null) { return; } if (AutoCompleteMode != AutoCompleteMode.None) { if (!fromHandleCreate && recreate && IsHandleCreated) { //RecreateHandle to avoid Leak. // notice the use of member variable to avoid re-entrancy AutoCompleteMode backUpMode = this.AutoCompleteMode; autoCompleteMode = AutoCompleteMode.None; RecreateHandle(); autoCompleteMode = backUpMode; } if (AutoCompleteSource == AutoCompleteSource.CustomSource) { if (AutoCompleteCustomSource != null) { if (AutoCompleteCustomSource.Count == 0) { int mode = NativeMethods.AUTOSUGGEST_OFF | NativeMethods.AUTOAPPEND_OFF; SafeNativeMethods.SHAutoComplete(new HandleRef(this, childEdit.Handle), mode); } else { if (stringSource == null) { stringSource = new StringSource(GetStringsForAutoComplete(AutoCompleteCustomSource)); if (!stringSource.Bind(new HandleRef(this, childEdit.Handle), (int)AutoCompleteMode)) { throw new ArgumentException(SR.GetString(SR.AutoCompleteFailure)); } } else { stringSource.RefreshList(GetStringsForAutoComplete(AutoCompleteCustomSource)); } } } } else if (AutoCompleteSource == AutoCompleteSource.ListItems) { if (DropDownStyle != ComboBoxStyle.DropDownList) { if (itemsCollection != null) { if (itemsCollection.Count == 0) { int mode = NativeMethods.AUTOSUGGEST_OFF | NativeMethods.AUTOAPPEND_OFF; SafeNativeMethods.SHAutoComplete(new HandleRef(this, childEdit.Handle), mode); } else { if (stringSource == null) { stringSource = new StringSource(GetStringsForAutoComplete(Items)); if (!stringSource.Bind(new HandleRef(this, childEdit.Handle), (int)AutoCompleteMode)) { throw new ArgumentException(SR.GetString(SR.AutoCompleteFailureListItems)); } } else { stringSource.RefreshList(GetStringsForAutoComplete(Items)); } } } } else { // Drop Down List special handling Debug.Assert(DropDownStyle == ComboBoxStyle.DropDownList); int mode = NativeMethods.AUTOSUGGEST_OFF | NativeMethods.AUTOAPPEND_OFF; SafeNativeMethods.SHAutoComplete(new HandleRef(this, childEdit.Handle), mode); } } else { try { int mode = 0; if (AutoCompleteMode == AutoCompleteMode.Suggest) { mode |= NativeMethods.AUTOSUGGEST | NativeMethods.AUTOAPPEND_OFF; } if (AutoCompleteMode == AutoCompleteMode.Append) { mode |= NativeMethods.AUTOAPPEND | NativeMethods.AUTOSUGGEST_OFF; } if (AutoCompleteMode == AutoCompleteMode.SuggestAppend) { mode |= NativeMethods.AUTOSUGGEST; mode |= NativeMethods.AUTOAPPEND; } int ret = SafeNativeMethods.SHAutoComplete(new HandleRef(this, childEdit.Handle), (int)AutoCompleteSource | mode); } catch (SecurityException) { // If we don't have full trust, degrade gracefully. Allow the control to // function without auto-complete. Allow the app to continue running. } } } else if (reset) { int mode = NativeMethods.AUTOSUGGEST_OFF | NativeMethods.AUTOAPPEND_OFF; SafeNativeMethods.SHAutoComplete(new HandleRef(this, childEdit.Handle), mode); } } ////// /// Selects the text in the editable portion of the ComboBox at the /// from the given start index to the given end index. /// public void Select(int start, int length) { if (start < 0) { throw new ArgumentOutOfRangeException("start", SR.GetString(SR.InvalidArgument, "start", start.ToString(CultureInfo.CurrentCulture))); } // the Length can be negative to support Selecting in the "reverse" direction.. int end = start + length; // but end cannot be negative... this means Length is far negative... if (end < 0) { throw new ArgumentOutOfRangeException("length", SR.GetString(SR.InvalidArgument, "length", length.ToString(CultureInfo.CurrentCulture))); } SendMessage(NativeMethods.CB_SETEDITSEL, 0, NativeMethods.Util.MAKELPARAM(start, end)); } ////// /// Selects all the text in the editable portion of the ComboBox. /// public void SelectAll() { Select(0, Int32.MaxValue); } ///protected override void SetBoundsCore(int x, int y, int width, int height, BoundsSpecified specified) { // If we are changing height, store the requested height. // Requested height is used if the style is changed to simple. // (Bug fix #20966) if ((specified & BoundsSpecified.Height) != BoundsSpecified.None) { requestedHeight = height; } base.SetBoundsCore(x, y, width, height, specified); } /// /// /// Performs the work of setting the specified items to the combobox /// protected override void SetItemsCore(IList value) { BeginUpdate(); Items.ClearInternal(); Items.AddRangeInternal(value); // if the list changed, we want to keep the same selected index // CurrencyManager will provide the PositionChanged event // it will be provided before changing the list though... if (this.DataManager != null) { if (this.DataSource is ICurrencyManagerProvider) { // Everett ListControl's had a bug where they would not fire // OnSelectedValueChanged if their list of items were refreshed. // We fix this post-Everett. // However, for APPCOMPAT reasons, we only want to fix it when binding to // Whidbey components. // vsw 547279. this.selectedValueChangedFired = false; } if (IsHandleCreated) { SendMessage(NativeMethods.CB_SETCURSEL, DataManager.Position, 0); } else { selectedIndex = DataManager.Position; } // if set_SelectedIndexChanged did not fire OnSelectedValueChanged // then we have to fire it ourselves, cos the list changed anyway if (!selectedValueChangedFired) { OnSelectedValueChanged(EventArgs.Empty); selectedValueChangedFired = false; } } EndUpdate(); } ///protected override void SetItemCore(int index, object value) { Items.SetItemInternal(index, value); } private bool ShouldSerializeAutoCompleteCustomSource() { return autoCompleteCustomSource != null && autoCompleteCustomSource.Count > 0; } internal bool ShouldSerializeDropDownWidth() { return (Properties.ContainsInteger(PropDropDownWidth)); } /// /// Indicates whether the itemHeight property should be persisted. /// internal bool ShouldSerializeItemHeight() { return (Properties.ContainsInteger(PropItemHeight)); } ////// /// Determines if the Text property needs to be persisted. /// internal override bool ShouldSerializeText() { return SelectedIndex == -1 && base.ShouldSerializeText(); } ////// /// Provides some interesting info about this control in String form. /// ///public override string ToString() { string s = base.ToString(); return s + ", Items.Count: " + ((itemsCollection == null) ? (0).ToString(CultureInfo.CurrentCulture) : itemsCollection.Count.ToString(CultureInfo.CurrentCulture)); } /// /// ///private void UpdateDropDownHeight() { if (dropDownHandle != IntPtr.Zero) { //Now use the DropDownHeight property instead of calculating the Height... int height = DropDownHeight; if (height == DefaultDropDownHeight) { int itemCount = (itemsCollection == null) ? 0 : itemsCollection.Count; int count = Math.Min(Math.Max(itemCount, 1), maxDropDownItems); height = (ItemHeight * count + 2); } SafeNativeMethods.SetWindowPos(new HandleRef(this, dropDownHandle), NativeMethods.NullHandleRef, 0, 0, DropDownWidth, height, NativeMethods.SWP_NOMOVE | NativeMethods.SWP_NOZORDER); } } /// /// Manufactures a MeasureItemEventArgs for each item in the list to simulate /// the combobox requesting the info. This gives the effect of allowing the /// measureitem info to be updated at anytime. /// ///private void UpdateItemHeight() { if (!IsHandleCreated) { // VSWhidbey 156992: if we don't create control here we report item heights incorrectly later on. CreateControl(); } if (DrawMode == DrawMode.OwnerDrawFixed) { SendMessage(NativeMethods.CB_SETITEMHEIGHT, -1, ItemHeight); SendMessage(NativeMethods.CB_SETITEMHEIGHT, 0, ItemHeight); } else if (DrawMode == DrawMode.OwnerDrawVariable) { SendMessage(NativeMethods.CB_SETITEMHEIGHT, -1, ItemHeight); Graphics graphics = CreateGraphicsInternal(); for (int i = 0; i < Items.Count; i++) { int original = (int)SendMessage(NativeMethods.CB_GETITEMHEIGHT, i, 0); MeasureItemEventArgs mievent = new MeasureItemEventArgs(graphics, i, original); OnMeasureItem(mievent); if (mievent.ItemHeight != original) { SendMessage(NativeMethods.CB_SETITEMHEIGHT, i, mievent.ItemHeight); } } graphics.Dispose(); } } /// /// Forces the text to be updated based on the current selection. /// ///private void UpdateText() { // V#45724 - Fire text changed for dropdown combos when the selection // changes, since the text really does change. We've got // to do this asynchronously because the actual edit text // isn't updated until a bit later (V#51240). // // QFE 2471: // v1.0 - ComboBox::set_Text compared items w/ "value" and set the SelectedIndex accordingly // v1.0 - null values can't correspond to String.Empty // v1.0 - SelectedIndex == -1 corresponds to Text == String.Emtpy // // v1.1 - ComboBox::set_Text compares FilterItemOnProperty(item) w/ "value" and set the SelectedIndex accordingly // v1.1 - null values correspond to String.Empty // v1.1 - SelectedIndex == -1 corresponds to Text == null string s = null; if (SelectedIndex != -1) { object item = Items[SelectedIndex]; if (item != null) { s = GetItemText(item); } } Text = s; if (DropDownStyle == ComboBoxStyle.DropDown) { if (childEdit != null && childEdit.Handle != IntPtr.Zero) { UnsafeNativeMethods.SendMessage(new HandleRef(this, childEdit.Handle), NativeMethods.WM_SETTEXT, IntPtr.Zero, s); } } } /// /// ///private void WmEraseBkgnd(ref Message m) { if ((DropDownStyle == ComboBoxStyle.Simple) && ParentInternal != null) { NativeMethods.RECT rect = new NativeMethods.RECT(); SafeNativeMethods.GetClientRect(new HandleRef(this, Handle), ref rect); Control p = ParentInternal; Graphics graphics = Graphics.FromHdcInternal(m.WParam); if (p != null) { Brush brush = new SolidBrush(p.BackColor); graphics.FillRectangle(brush, rect.left, rect.top, rect.right - rect.left, rect.bottom - rect.top); brush.Dispose(); } else { graphics.FillRectangle(SystemBrushes.Control, rect.left, rect.top, rect.right - rect.left, rect.bottom - rect.top); } graphics.Dispose(); m.Result = (IntPtr)1; return; } base.WndProc(ref m); } /// /// ///private void WmParentNotify(ref Message m) { base.WndProc(ref m); if (unchecked((int)(long)m.WParam) == (NativeMethods.WM_CREATE | 1000 << 16)) { dropDownHandle = m.LParam; } } /// /// Text change behavior. /// Here are the window messages corresponding to each user event. /// /// DropDown (free text window): /// Type in Text Window: /// CBN_EDITUPDATE /// CBN_EDITCHANGE /// Down/Up Arrow /// CBN_SELENDOK (text not changed yet) /// CBN_SELCHANGE /// with text set to beginning of a valid item -- drop and close dropdown (selects that item) /// CBN_DROPDOWN /// CBN_CLOSEUP /// Drop List, up/down arrow to select item, click away /// CBN_DROPDOWN /// CBN_SELCHANGE /// CBN_CLOSEUP /// Drop List, click on item /// CBN_DROPDOWN /// CBN_SELENDOK /// CBN_CLOSEUP /// CBN_SELCHANGE (text changes here via selected item) /// /// DropDownList (limited text window): /// /// Type text and arrow up/down: /// CBN_SELENDOK (text already changed) /// CBN_SELCHANGE /// Drop List, up/down arrow to select item, click away /// CBN_DROPDOWN /// CBN_SELCHANGE /// CBN_CLOSEUP /// Drop List, click on item /// CBN_DROPDOWN /// CBN_SELENDOK /// CBN_CLOSEUP /// CBN_SELCHANGE /// /// Simple (listbox visible): /// Type in Text Window: /// CBN_EDITUPDATE /// CBN_EDITCHANGE /// Down/Up Arrow /// CBN_SELENDOK (text not changed yet) /// CBN_SELCHANGE /// Click on item /// CBN_SELENDOK (text not changed yet) /// CBN_SELCHANGE /// /// /// What we do is fire textchange events in these messages: /// CBN_SELCHANGE /// CBN_EDITCHANGE /// CBN_CLOSEUP /// /// and we only actually call the real event if the Text is different than currentText. /// currentText is never changed outside this method. /// This internal version can be called from anywhere we might suspect text has changed /// it's fairly safe to call anywhere. /// /// ///private void WmReflectCommand(ref Message m) { switch (NativeMethods.Util.HIWORD(m.WParam)) { case NativeMethods.CBN_DBLCLK: //OnDoubleClick(EventArgs.Empty); break; case NativeMethods.CBN_EDITUPDATE: OnTextUpdate(EventArgs.Empty); break; case NativeMethods.CBN_CLOSEUP: OnDropDownClosed(EventArgs.Empty); if (FormattingEnabled && Text != currentText && dropDown) { OnTextChanged(EventArgs.Empty); } dropDown = false; break; case NativeMethods.CBN_DROPDOWN: currentText = Text; dropDown = true; OnDropDown(EventArgs.Empty); UpdateDropDownHeight(); break; case NativeMethods.CBN_EDITCHANGE: OnTextChanged(EventArgs.Empty); break; case NativeMethods.CBN_SELCHANGE: UpdateText(); OnSelectedIndexChanged(EventArgs.Empty); break; case NativeMethods.CBN_SELENDOK: OnSelectionChangeCommittedInternal(EventArgs.Empty); break; } } /// /// ///private void WmReflectDrawItem(ref Message m) { NativeMethods.DRAWITEMSTRUCT dis = (NativeMethods.DRAWITEMSTRUCT)m.GetLParam(typeof(NativeMethods.DRAWITEMSTRUCT)); IntPtr oldPal = SetUpPalette(dis.hDC, false /*force*/, false /*realize*/); try { Graphics g = Graphics.FromHdcInternal(dis.hDC); try { OnDrawItem(new DrawItemEventArgs(g, Font, Rectangle.FromLTRB(dis.rcItem.left, dis.rcItem.top, dis.rcItem.right, dis.rcItem.bottom), dis.itemID, (DrawItemState)dis.itemState, ForeColor, BackColor)); } finally { g.Dispose(); } } finally { if (oldPal != IntPtr.Zero) { SafeNativeMethods.SelectPalette(new HandleRef(this, dis.hDC), new HandleRef(null, oldPal), 0); } } m.Result = (IntPtr)1; } /// /// ///private void WmReflectMeasureItem(ref Message m) { NativeMethods.MEASUREITEMSTRUCT mis = (NativeMethods.MEASUREITEMSTRUCT)m.GetLParam(typeof(NativeMethods.MEASUREITEMSTRUCT)); // Determine if message was sent by a combo item or the combo edit field if (DrawMode == DrawMode.OwnerDrawVariable && mis.itemID >= 0) { Graphics graphics = CreateGraphicsInternal(); MeasureItemEventArgs mie = new MeasureItemEventArgs(graphics, mis.itemID, ItemHeight); OnMeasureItem(mie); mis.itemHeight = mie.ItemHeight; graphics.Dispose(); } else { // Message was sent by the combo edit field mis.itemHeight = ItemHeight; } Marshal.StructureToPtr(mis, m.LParam, false); m.Result = (IntPtr)1; } /// /// /// The comboboxs window procedure. Inheritng classes can override this /// to add extra functionality, but should not forget to call /// base.wndProc(m); to ensure the combo continues to function properly. /// ///[SecurityPermission(SecurityAction.LinkDemand, Flags = SecurityPermissionFlag.UnmanagedCode)] protected override void WndProc(ref Message m) { switch (m.Msg) { // We don't want to fire the focus events twice - // once in the combobox and once in the ChildWndProc. case NativeMethods.WM_SETFOCUS: try { fireSetFocus = false; base.WndProc(ref m); } finally { fireSetFocus = true; } break; case NativeMethods.WM_KILLFOCUS: try { fireLostFocus = false; base.WndProc(ref m); // Nothing to see here... Just keep on walking... VSWhidbey 504477. // Turns out that with Theming off, we don't get quite the same messages as with theming on. // With theming on we get a WM_MOUSELEAVE after a WM_KILLFOCUS even if you use the Tab key // to move focus. Our response to WM_MOUSELEAVE causes us to repaint everything correctly. // With theming off, we do not get a WM_MOUSELEAVE after a WM_KILLFOCUS, and since we don't have a childwndproc // when we are a Flat DropDownList, we need to force a repaint. The easiest way to do this is to send a // WM_MOUSELEAVE to ourselves, since that also sets up the right state. Or... at least the state is the same // as with Theming on. // This is such a @#$(*@$ hack. if (!Application.RenderWithVisualStyles && GetStyle(ControlStyles.UserPaint) == false && this.DropDownStyle == ComboBoxStyle.DropDownList && (FlatStyle == FlatStyle.Flat || FlatStyle == FlatStyle.Popup)) { UnsafeNativeMethods.PostMessage(new HandleRef(this, Handle), NativeMethods.WM_MOUSELEAVE, 0, 0); } } finally { fireLostFocus = true; } break; case NativeMethods.WM_CTLCOLOREDIT: case NativeMethods.WM_CTLCOLORLISTBOX: m.Result = InitializeDCForWmCtlColor(m.WParam, m.Msg); break; case NativeMethods.WM_ERASEBKGND: WmEraseBkgnd(ref m); break; case NativeMethods.WM_PARENTNOTIFY: WmParentNotify(ref m); break; case NativeMethods.WM_REFLECT + NativeMethods.WM_COMMAND: WmReflectCommand(ref m); break; case NativeMethods.WM_REFLECT + NativeMethods.WM_DRAWITEM: WmReflectDrawItem(ref m); break; case NativeMethods.WM_REFLECT + NativeMethods.WM_MEASUREITEM: WmReflectMeasureItem(ref m); break; case NativeMethods.WM_LBUTTONDOWN: mouseEvents = true; base.WndProc(ref m); break; case NativeMethods.WM_LBUTTONUP: // Get the mouse location // NativeMethods.RECT r = new NativeMethods.RECT(); UnsafeNativeMethods.GetWindowRect(new HandleRef(this, Handle), ref r); Rectangle ClientRect = new Rectangle(r.left, r.top, r.right - r.left, r.bottom - r.top); int x = NativeMethods.Util.SignedLOWORD(m.LParam); int y = NativeMethods.Util.SignedHIWORD(m.LParam); Point pt = new Point(x, y); pt = PointToScreen(pt); //mouseEvents is used to keep the check that we get the WM_LBUTTONUP after //WM_LBUTTONDOWN or WM_LBUTTONDBLBCLK // combo box gets a WM_LBUTTONUP for focus change ... // if (mouseEvents && !ValidationCancelled) { mouseEvents = false; bool captured = Capture; if (captured && ClientRect.Contains(pt)) { OnClick(new MouseEventArgs(MouseButtons.Left, 1, NativeMethods.Util.SignedLOWORD(m.LParam), NativeMethods.Util.SignedHIWORD(m.LParam), 0)); OnMouseClick(new MouseEventArgs(MouseButtons.Left, 1, NativeMethods.Util.SignedLOWORD(m.LParam), NativeMethods.Util.SignedHIWORD(m.LParam), 0)); } base.WndProc(ref m); } else { CaptureInternal = false; DefWndProc(ref m); } break; case NativeMethods.WM_MOUSELEAVE: DefWndProc(ref m); OnMouseLeaveInternal(EventArgs.Empty); break; case NativeMethods.WM_PAINT: if (GetStyle(ControlStyles.UserPaint) == false && (FlatStyle == FlatStyle.Flat || FlatStyle == FlatStyle.Popup)) { using (WindowsRegion dr = new WindowsRegion(FlatComboBoxAdapter.dropDownRect)) { using (WindowsRegion wr = new WindowsRegion(this.Bounds)) { // Stash off the region we have to update (the base is going to clear this off in BeginPaint) NativeMethods.RegionFlags updateRegionFlags = (NativeMethods.RegionFlags)SafeNativeMethods.GetUpdateRgn(new HandleRef(this, this.Handle), new HandleRef(this, wr.HRegion), true); dr.CombineRegion(wr, dr, RegionCombineMode.DIFF); Rectangle updateRegionBoundingRect = wr.ToRectangle(); FlatComboBoxAdapter.ValidateOwnerDrawRegions(this, updateRegionBoundingRect); // Call the base class to do its painting (with a clipped DC). NativeMethods.PAINTSTRUCT ps = new NativeMethods.PAINTSTRUCT(); IntPtr dc; bool disposeDc = false; if (m.WParam == IntPtr.Zero) { dc = UnsafeNativeMethods.BeginPaint(new HandleRef(this, Handle), ref ps); disposeDc = true; } else { dc = m.WParam; } using (DeviceContext mDC = DeviceContext.FromHdc(dc)) { using (WindowsGraphics wg = new WindowsGraphics(mDC)) { if (updateRegionFlags != NativeMethods.RegionFlags.ERROR) { wg.DeviceContext.SetClip(dr); } m.WParam = dc; DefWndProc(ref m); if (updateRegionFlags != NativeMethods.RegionFlags.ERROR) { wg.DeviceContext.SetClip(wr); } using (Graphics g = Graphics.FromHdcInternal(dc)) { FlatComboBoxAdapter.DrawFlatCombo(this, g); } } } if (disposeDc) { UnsafeNativeMethods.EndPaint(new HandleRef(this, Handle), ref ps); } } return; } } base.WndProc(ref m); break; case NativeMethods.WM_PRINTCLIENT: // all the fancy stuff we do in OnPaint has to happen again in OnPrint. if (GetStyle(ControlStyles.UserPaint) == false && FlatStyle == FlatStyle.Flat || FlatStyle == FlatStyle.Popup) { DefWndProc(ref m); if (((int)m.LParam & NativeMethods.PRF_CLIENT) == NativeMethods.PRF_CLIENT) { if (GetStyle(ControlStyles.UserPaint) == false && FlatStyle == FlatStyle.Flat || FlatStyle == FlatStyle.Popup) { using (Graphics g = Graphics.FromHdcInternal(m.WParam)) { FlatComboBoxAdapter.DrawFlatCombo(this,g); } } return; } } base.WndProc(ref m); return; case NativeMethods.WM_SETCURSOR: base.WndProc(ref m); break; case NativeMethods.WM_SETFONT: //(bug 119265) if (Width == 0) { suppressNextWindosPos = true; } base.WndProc(ref m); break; case NativeMethods.WM_WINDOWPOSCHANGED: if (!suppressNextWindosPos) { base.WndProc(ref m); } suppressNextWindosPos = false; break; case NativeMethods.WM_NCDESTROY: base.WndProc(ref m); ReleaseChildWindow(); break; default: if (m.Msg == NativeMethods.WM_MOUSEENTER) { DefWndProc(ref m); OnMouseEnterInternal(EventArgs.Empty); break; } base.WndProc(ref m); break; } } /// /// ///private class ComboBoxChildNativeWindow : NativeWindow { private ComboBox _owner; private InternalAccessibleObject _accessibilityObject; internal ComboBoxChildNativeWindow(ComboBox comboBox) { _owner = comboBox; } protected override void WndProc(ref Message m) { switch (m.Msg) { case NativeMethods.WM_GETOBJECT: WmGetObject(ref m); return; default: _owner.ChildWndProc(ref m); break; } } private void WmGetObject(ref Message m) { // See "How to Handle WM_GETOBJECT" in MSDN // if (NativeMethods.OBJID_CLIENT == unchecked((int)(long)m.LParam)) { // Get the IAccessible GUID // Guid IID_IAccessible = new Guid(NativeMethods.uuid_IAccessible); // Get an Lresult for the accessibility Object for this control // IntPtr punkAcc; try { AccessibleObject wfAccessibleObject = null; UnsafeNativeMethods.IAccessibleInternal iacc = null; if (_accessibilityObject == null) { IntSecurity.UnmanagedCode.Assert(); try { wfAccessibleObject = new ChildAccessibleObject(_owner, Handle); _accessibilityObject = new InternalAccessibleObject(wfAccessibleObject); } finally { CodeAccessPermission.RevertAssert(); } } iacc = (UnsafeNativeMethods.IAccessibleInternal)_accessibilityObject; // Obtain the Lresult // punkAcc = Marshal.GetIUnknownForObject(iacc); IntSecurity.UnmanagedCode.Assert(); try { m.Result = UnsafeNativeMethods.LresultFromObject(ref IID_IAccessible, m.WParam, new HandleRef(this, punkAcc)); } finally { CodeAccessPermission.RevertAssert(); Marshal.Release(punkAcc); } } catch (Exception e) { throw new InvalidOperationException(SR.GetString(SR.RichControlLresult), e); } } else { // m.lparam != OBJID_CLIENT, so do default message processing DefWndProc(ref m); } } } private sealed class ItemComparer : System.Collections.IComparer { private ComboBox comboBox; public ItemComparer(ComboBox comboBox) { this.comboBox = comboBox; } public int Compare(object item1, object item2) { if (item1 == null) { if (item2 == null) return 0; //both null, then they are equal return -1; //item1 is null, but item2 is valid (greater) } if (item2 == null) return 1; //item2 is null, so item 1 is greater String itemName1 = comboBox.GetItemText(item1); String itemName2 = comboBox.GetItemText(item2); CompareInfo compInfo = (Application.CurrentCulture).CompareInfo; return compInfo.Compare(itemName1, itemName2, CompareOptions.StringSort); } } /// /// /// [ListBindable(false)] public class ObjectCollection : IList { private ComboBox owner; private ArrayList innerList; private IComparer comparer; ///[To be supplied.] ////// /// public ObjectCollection(ComboBox owner) { this.owner = owner; } private IComparer Comparer { get { if (comparer == null) { comparer = new ItemComparer(owner); } return comparer; } } private ArrayList InnerList { get { if (innerList == null) { innerList = new ArrayList(); } return innerList; } } ///[To be supplied.] ////// /// Retrieves the number of items. /// public int Count { get { return InnerList.Count; } } ////// object ICollection.SyncRoot { get { return this; } } /// /// bool ICollection.IsSynchronized { get { return false; } } /// /// bool IList.IsFixedSize { get { return false; } } /// /// /// public bool IsReadOnly { get { return false; } } ///[To be supplied.] ////// /// Adds an item to the combo box. For an unsorted combo box, the item is /// added to the end of the existing list of items. For a sorted combo box, /// the item is inserted into the list according to its sorted position. /// The item's toString() method is called to obtain the string that is /// displayed in the combo box. /// A SystemException occurs if there is insufficient space available to /// store the new item. /// public int Add(object item) { owner.CheckNoDataSource(); int index = AddInternal(item); if (owner.UpdateNeeded() && owner.AutoCompleteSource == AutoCompleteSource.ListItems) { owner.SetAutoComplete(false, false); } return index; } private int AddInternal(object item) { if (item == null) { throw new ArgumentNullException("item"); } int index = -1; if (!owner.sorted) { InnerList.Add(item); } else { index = InnerList.BinarySearch(item, Comparer); if (index < 0) { index = ~index; // getting the index of the first element that is larger than the search value } Debug.Assert(index>=0 && index <= InnerList.Count, "Wrong index for insert"); InnerList.Insert(index, item); } bool successful = false; try { if (owner.sorted) { if (owner.IsHandleCreated) { owner.NativeInsert(index, item); } } else { index = InnerList.Count-1; if (owner.IsHandleCreated) { owner.NativeAdd(item); } } successful = true; } finally { if (!successful) { InnerList.Remove(item); } } return index; } ////// int IList.Add(object item) { return Add(item); } /// /// /// public void AddRange(object[] items) { owner.CheckNoDataSource(); owner.BeginUpdate(); try { AddRangeInternal(items); } finally { owner.EndUpdate(); } } internal void AddRangeInternal(IList items) { if (items == null) { throw new ArgumentNullException("items"); } foreach (object item in items) { // adding items one-by-one for performance (especially for sorted combobox) // we can not rely on ArrayList.Sort since its worst case complexity is n*n // AddInternal is based on BinarySearch and ensures n*log(n) complexity AddInternal(item); } if (owner.AutoCompleteSource == AutoCompleteSource.ListItems) { owner.SetAutoComplete(false, false); } } ///[To be supplied.] ////// /// Retrieves the item with the specified index. /// [Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)] public virtual object this[int index] { get { if (index < 0 || index >= InnerList.Count) { throw new ArgumentOutOfRangeException("index", SR.GetString(SR.InvalidArgument, "index", (index).ToString(CultureInfo.CurrentCulture))); } return InnerList[index]; } set { owner.CheckNoDataSource(); SetItemInternal(index, value); } } ////// /// Removes all items from the ComboBox. /// public void Clear() { owner.CheckNoDataSource(); ClearInternal(); } internal void ClearInternal() { if (owner.IsHandleCreated) { owner.NativeClear(); } InnerList.Clear(); owner.selectedIndex = -1; if (owner.AutoCompleteSource == AutoCompleteSource.ListItems) { owner.SetAutoComplete(false, true /*recreateHandle*/); } } ////// /// public bool Contains(object value) { return IndexOf(value) != -1; } ///[To be supplied.] ////// /// Copies the ComboBox Items collection to a destination array. /// public void CopyTo(object[] destination, int arrayIndex) { InnerList.CopyTo(destination, arrayIndex); } ////// void ICollection.CopyTo(Array destination, int index) { InnerList.CopyTo(destination, index); } /// /// /// Returns an enumerator for the ComboBox Items collection. /// public IEnumerator GetEnumerator() { return InnerList.GetEnumerator(); } ////// /// public int IndexOf(object value) { if (value == null) { throw new ArgumentNullException("value"); } return InnerList.IndexOf(value); } ///[To be supplied.] ////// /// Adds an item to the combo box. For an unsorted combo box, the item is /// added to the end of the existing list of items. For a sorted combo box, /// the item is inserted into the list according to its sorted position. /// The item's toString() method is called to obtain the string that is /// displayed in the combo box. /// A SystemException occurs if there is insufficient space available to /// store the new item. /// public void Insert(int index, object item) { owner.CheckNoDataSource(); if (item == null) { throw new ArgumentNullException("item"); } if (index < 0 || index > InnerList.Count) { throw new ArgumentOutOfRangeException("index", SR.GetString(SR.InvalidArgument, "index", (index).ToString(CultureInfo.CurrentCulture))); } // If the combo box is sorted, then nust treat this like an add // because we are going to twiddle the index anyway. // if (owner.sorted) { Add(item); } else { InnerList.Insert(index, item); if (owner.IsHandleCreated) { bool successful = false; try { owner.NativeInsert(index, item); successful = true; } finally { if (successful) { if (owner.AutoCompleteSource == AutoCompleteSource.ListItems) { owner.SetAutoComplete(false, false); } } else { InnerList.RemoveAt(index); } } } } } ////// /// Removes an item from the ComboBox at the given index. /// public void RemoveAt(int index) { owner.CheckNoDataSource(); if (index < 0 || index >= InnerList.Count) { throw new ArgumentOutOfRangeException("index", SR.GetString(SR.InvalidArgument, "index", (index).ToString(CultureInfo.CurrentCulture))); } if (owner.IsHandleCreated) { owner.NativeRemoveAt(index); } InnerList.RemoveAt(index); if (!owner.IsHandleCreated && index < owner.selectedIndex) { owner.selectedIndex--; } if (owner.AutoCompleteSource == AutoCompleteSource.ListItems) { owner.SetAutoComplete(false, false); } } ////// /// Removes the given item from the ComboBox, provided that it is /// actually in the list. /// public void Remove(object value) { int index = InnerList.IndexOf(value); if (index != -1) { RemoveAt(index); } } internal void SetItemInternal(int index, object value) { if (value == null) { throw new ArgumentNullException("value"); } if (index < 0 || index >= InnerList.Count) { throw new ArgumentOutOfRangeException("index", SR.GetString(SR.InvalidArgument, "index", (index).ToString(CultureInfo.CurrentCulture))); } InnerList[index] = value; // If the native control has been created, and the display text of the new list item object // is different to the current text in the native list item, recreate the native list item... if (owner.IsHandleCreated) { bool selected = (index == owner.SelectedIndex); if (String.Compare(this.owner.GetItemText(value), this.owner.NativeGetItemText(index), true, CultureInfo.CurrentCulture) != 0) { owner.NativeRemoveAt(index); owner.NativeInsert(index, value); if (selected) { owner.SelectedIndex = index; owner.UpdateText(); } if (owner.AutoCompleteSource == AutoCompleteSource.ListItems) { owner.SetAutoComplete(false, false); } } else { // NEW - FOR COMPATIBILITY REASONS // Minimum compatibility fix for VSWhidbey 377287/444903 if (selected) { owner.OnSelectedItemChanged(EventArgs.Empty); //we do this because set_SelectedIndex does this. (for consistency) owner.OnSelectedIndexChanged(EventArgs.Empty); } } } } } // end ObjectCollection ////// [ComVisible(true)] public class ChildAccessibleObject : AccessibleObject { ComboBox owner; /// [SecurityPermission(SecurityAction.Demand, Flags = SecurityPermissionFlag.UnmanagedCode)] public ChildAccessibleObject(ComboBox owner, IntPtr handle) { Debug.Assert(owner != null && owner.Handle != IntPtr.Zero, "ComboBox's handle hasn't been created"); this.owner = owner; UseStdAccessibleObjects(handle); } /// public override string Name { get { return owner.AccessibilityObject.Name; } } } /// /// [System.Runtime.InteropServices.ComVisible(true)] internal class ComboBoxAccessibleObject : ControlAccessibleObject { private const int COMBOBOX_ACC_ITEM_INDEX = 1; public ComboBoxAccessibleObject(Control ownerControl) : base(ownerControl) { } internal override string get_accNameInternal(object childID) { this.ValidateChildID(ref childID); if (childID != null && ((int)childID) == COMBOBOX_ACC_ITEM_INDEX) { return this.Name; } else { return base.get_accNameInternal(childID); } } internal override string get_accKeyboardShortcutInternal(object childID) { this.ValidateChildID(ref childID); if (childID != null && ((int)childID) == COMBOBOX_ACC_ITEM_INDEX) { return this.KeyboardShortcut; } else { return base.get_accKeyboardShortcutInternal(childID); } } } ////// This subclasses an autocomplete window so that we can determine if control is inside the AC wndproc. /// private sealed class ACNativeWindow : NativeWindow { static internal int inWndProcCnt; //this hashtable can contain null for those ACWindows we find, but are sure are not ours. static private Hashtable ACWindows = new Hashtable(); [ SuppressMessage("Microsoft.Usage", "CA2214:DoNotCallOverridableMethodsInConstructors") ] internal ACNativeWindow(IntPtr acHandle) { Debug.Assert(!ACWindows.ContainsKey(acHandle)); this.AssignHandle(acHandle); ACWindows.Add(acHandle, this); UnsafeNativeMethods.EnumChildWindows(new HandleRef(this, acHandle), new NativeMethods.EnumChildrenCallback(ACNativeWindow.RegisterACWindowRecursive), NativeMethods.NullHandleRef); } [SuppressMessage("Microsoft.Performance", "CA1806:DoNotIgnoreMethodResults")] private static bool RegisterACWindowRecursive(IntPtr handle, IntPtr lparam) { if (!ACWindows.ContainsKey(handle)) { ACNativeWindow newAC = new ACNativeWindow(handle); } return true; } internal bool Visible { get { return SafeNativeMethods.IsWindowVisible(new HandleRef(this, Handle)); } } static internal bool AutoCompleteActive { get { if (inWndProcCnt>0) { return true; } foreach(object o in ACWindows.Values) { ACNativeWindow window = o as ACNativeWindow; if (window != null && window.Visible) { return true; } } return false; } } protected override void WndProc(ref Message m) { inWndProcCnt++; try { base.WndProc(ref m); } finally { inWndProcCnt--; } if (m.Msg == NativeMethods.WM_NCDESTROY) { Debug.Assert(ACWindows.ContainsKey(this.Handle)); ACWindows.Remove(this.Handle); //so we do not leak ac windows. } } [SuppressMessage("Microsoft.Performance", "CA1806:DoNotIgnoreMethodResults")] internal static void RegisterACWindow(IntPtr acHandle, bool subclass) { if (subclass && ACWindows.ContainsKey(acHandle)) { if (ACWindows[acHandle] == null) { ACWindows.Remove(acHandle); //if an external handle got destroyed, dont let it stop us. } } if (!ACWindows.ContainsKey(acHandle)) { if (subclass) { ACNativeWindow newAC = new ACNativeWindow(acHandle); } else { ACWindows.Add(acHandle, null); } } } ////// This method clears out null entries so we get a clean BEFORE and AFTER snapshot /// null entries are ACWindows that belong to someone else. /// [System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Performance", "CA1808:AvoidCallsThatBoxValueTypes")] // Perfectly acceptible when dealing with PropertyDescriptors internal static void ClearNullACWindows() { ArrayList nulllist = new ArrayList(); foreach (DictionaryEntry e in ACWindows) { if (e.Value == null) { nulllist.Add(e.Key); } } foreach (IntPtr handle in nulllist) { ACWindows.Remove(handle); } } } ////// This finds all autcomplete windows that belong to the active thread. /// private class AutoCompleteDropDownFinder { private const int MaxClassName = 256; private const string AutoCompleteClassName = "Auto-Suggest Dropdown"; bool shouldSubClass = false; //nonstatic internal void FindDropDowns() { FindDropDowns(true); } internal void FindDropDowns(bool subclass) { if (!subclass) { //generating a before snapshot -- lets lose the null handles ACNativeWindow.ClearNullACWindows(); } // Look for a popped up dropdown shouldSubClass = subclass; UnsafeNativeMethods.EnumThreadWindows(SafeNativeMethods.GetCurrentThreadId(), new NativeMethods.EnumThreadWindowsCallback(this.Callback), new HandleRef(null, IntPtr.Zero)); } private bool Callback(IntPtr hWnd, IntPtr lParam) { HandleRef hRef = new HandleRef(null, hWnd); // Check class name and see if it's visible if (GetClassName(hRef) == AutoCompleteClassName) { ACNativeWindow.RegisterACWindow(hRef.Handle, shouldSubClass); } return true; } static string GetClassName(HandleRef hRef) { StringBuilder sb = new StringBuilder(MaxClassName); UnsafeNativeMethods.GetClassName(hRef, sb, MaxClassName); return sb.ToString(); } } private FlatComboAdapter FlatComboBoxAdapter { get { FlatComboAdapter comboAdapter = Properties.GetObject(PropFlatComboAdapter) as FlatComboAdapter; if (comboAdapter == null || !comboAdapter.IsValid(this)) { comboAdapter = CreateFlatComboAdapterInstance(); Properties.SetObject(PropFlatComboAdapter, comboAdapter); } return comboAdapter; } } internal virtual FlatComboAdapter CreateFlatComboAdapterInstance() { return new FlatComboAdapter(this,/*smallButton=*/false); } internal class FlatComboAdapter { Rectangle outerBorder; Rectangle innerBorder; Rectangle innerInnerBorder; internal Rectangle dropDownRect; Rectangle whiteFillRect; Rectangle clientRect; RightToLeft origRightToLeft; // The combo box's RTL value when we were created private const int WhiteFillRectWidth = 5; // used for making the button look smaller than it is public FlatComboAdapter(ComboBox comboBox, bool smallButton) { clientRect = comboBox.ClientRectangle; int dropDownButtonWidth = System.Windows.Forms.SystemInformation.HorizontalScrollBarArrowWidth; outerBorder = new Rectangle(clientRect.Location, new Size(clientRect.Width - 1, clientRect.Height - 1)); innerBorder = new Rectangle(outerBorder.X + 1, outerBorder.Y + 1, outerBorder.Width - dropDownButtonWidth - 2, outerBorder.Height - 2); innerInnerBorder = new Rectangle(innerBorder.X + 1, innerBorder.Y + 1, innerBorder.Width - 2, innerBorder.Height - 2); dropDownRect = new Rectangle(innerBorder.Right + 1, innerBorder.Y, dropDownButtonWidth, innerBorder.Height+1); // fill in several pixels of the dropdown rect with white so that it looks like the combo button is thinner. if (smallButton) { whiteFillRect = dropDownRect; whiteFillRect.Width = WhiteFillRectWidth; dropDownRect.X += WhiteFillRectWidth; dropDownRect.Width -= WhiteFillRectWidth; } origRightToLeft = comboBox.RightToLeft; if (origRightToLeft == RightToLeft.Yes) { innerBorder.X = clientRect.Width - innerBorder.Right; innerInnerBorder.X = clientRect.Width - innerInnerBorder.Right; dropDownRect.X = clientRect.Width - dropDownRect.Right; whiteFillRect.X = clientRect.Width - whiteFillRect.Right + 1; // since we're filling, we need to move over to the next px. } } public bool IsValid(ComboBox combo) { return (combo.ClientRectangle == clientRect && combo.RightToLeft == origRightToLeft); } ////// /// Paints over the edges of the combo box to make it appear flat. /// public virtual void DrawFlatCombo(ComboBox comboBox, Graphics g) { if (comboBox.DropDownStyle == ComboBoxStyle.Simple){ return; } Color outerBorderColor = GetOuterBorderColor(comboBox); Color innerBorderColor = GetInnerBorderColor(comboBox); bool rightToLeft = comboBox.RightToLeft == RightToLeft.Yes; // draw the drop down DrawFlatComboDropDown(comboBox, g, dropDownRect); // when we are disabled there is one line of color that seems to eek through if backcolor is set // so lets erase it. if (!LayoutUtils.IsZeroWidthOrHeight(whiteFillRect)) { // fill in two more pixels with white so it looks smaller. using (Brush b = new SolidBrush(innerBorderColor)) { g.FillRectangle(b, whiteFillRect); } } // Draw the outer border if (outerBorderColor.IsSystemColor) { Pen outerBorderPen = SystemPens.FromSystemColor(outerBorderColor); g.DrawRectangle(outerBorderPen, outerBorder); if (rightToLeft) { g.DrawRectangle(outerBorderPen, new Rectangle(outerBorder.X, outerBorder.Y, dropDownRect.Width + 1, outerBorder.Height)); } else { g.DrawRectangle(outerBorderPen, new Rectangle(dropDownRect.X, outerBorder.Y, outerBorder.Right - dropDownRect.X, outerBorder.Height)); } } else { using (Pen outerBorderPen = new Pen(outerBorderColor)) { g.DrawRectangle(outerBorderPen, outerBorder); if (rightToLeft) { g.DrawRectangle(outerBorderPen, new Rectangle(outerBorder.X, outerBorder.Y, dropDownRect.Width + 1, outerBorder.Height)); } else { g.DrawRectangle(outerBorderPen, new Rectangle(dropDownRect.X, outerBorder.Y, outerBorder.Right - dropDownRect.X, outerBorder.Height)); } } } // Draw the inner border if (innerBorderColor.IsSystemColor) { Pen innerBorderPen = SystemPens.FromSystemColor(innerBorderColor); g.DrawRectangle(innerBorderPen, innerBorder); g.DrawRectangle(innerBorderPen, innerInnerBorder); } else { using (Pen innerBorderPen = new Pen(innerBorderColor)) { g.DrawRectangle(innerBorderPen, innerBorder); g.DrawRectangle(innerBorderPen, innerInnerBorder); } } // Draw a dark border around everything if we're in popup mode if ((!comboBox.Enabled) ||(comboBox.FlatStyle == FlatStyle.Popup)) { bool focused = comboBox.ContainsFocus || comboBox.MouseIsOver; Color borderPenColor = GetPopupOuterBorderColor(comboBox, focused); using (Pen borderPen = new Pen(borderPenColor)) { Pen innerPen = (comboBox.Enabled) ? borderPen : SystemPens.Control; // around the dropdown if (rightToLeft) { g.DrawRectangle(innerPen, new Rectangle(outerBorder.X, outerBorder.Y, dropDownRect.Width + 1, outerBorder.Height)); } else { g.DrawRectangle(innerPen, new Rectangle(dropDownRect.X, outerBorder.Y, outerBorder.Right - dropDownRect.X, outerBorder.Height)); } // around the whole combobox. g.DrawRectangle(borderPen, outerBorder); } } } ////// /// Paints over the edges of the combo box to make it appear flat. /// protected virtual void DrawFlatComboDropDown(ComboBox comboBox, Graphics g, Rectangle dropDownRect) { g.FillRectangle(SystemBrushes.Control, dropDownRect); Brush brush = (comboBox.Enabled) ? SystemBrushes.ControlText : SystemBrushes.ControlDark; Point middle = new Point(dropDownRect.Left + dropDownRect.Width / 2, dropDownRect.Top + dropDownRect.Height / 2); if (origRightToLeft == RightToLeft.Yes) { // if the width is odd - favor pushing it over one pixel left. middle.X -= (dropDownRect.Width % 2); } else { // if the width is odd - favor pushing it over one pixel right. middle.X += (dropDownRect.Width % 2); } g.FillPolygon(brush, new Point[] { new Point(middle.X - 2, middle.Y - 1), new Point(middle.X + 3, middle.Y - 1), new Point(middle.X, middle.Y + 2) }); } protected virtual Color GetOuterBorderColor(ComboBox comboBox) { return (comboBox.Enabled) ? SystemColors.Window : SystemColors.ControlDark; } protected virtual Color GetPopupOuterBorderColor(ComboBox comboBox, bool focused) { if (!comboBox.Enabled) { return SystemColors.ControlDark; } return (focused) ? SystemColors.ControlDark : SystemColors.Window; } protected virtual Color GetInnerBorderColor(ComboBox comboBox) { return (comboBox.Enabled) ? comboBox.BackColor : SystemColors.Control; } // this eliminates flicker by removing the pieces we're going to paint ourselves from // the update region. Note the UpdateRegionBox is the bounding box of the actual update region. // this is just here so we can quickly eliminate rectangles that arent in the update region. public void ValidateOwnerDrawRegions(ComboBox comboBox, Rectangle updateRegionBox) { NativeMethods.RECT validRect; if (comboBox != null) { return; } Rectangle topOwnerDrawArea = new Rectangle(0,0,comboBox.Width, innerBorder.Top); Rectangle bottomOwnerDrawArea = new Rectangle(0,innerBorder.Bottom,comboBox.Width, comboBox.Height-innerBorder.Bottom); Rectangle leftOwnerDrawArea = new Rectangle(0,0,innerBorder.Left, comboBox.Height); Rectangle rightOwnerDrawArea = new Rectangle(innerBorder.Right,0,comboBox.Width - innerBorder.Right,comboBox.Height); if (topOwnerDrawArea.IntersectsWith(updateRegionBox)) { validRect = new NativeMethods.RECT(topOwnerDrawArea); SafeNativeMethods.ValidateRect(new HandleRef(comboBox, comboBox.Handle), ref validRect); } if (bottomOwnerDrawArea.IntersectsWith(updateRegionBox)) { validRect = new NativeMethods.RECT(bottomOwnerDrawArea); SafeNativeMethods.ValidateRect(new HandleRef(comboBox, comboBox.Handle), ref validRect); } if (leftOwnerDrawArea.IntersectsWith(updateRegionBox)) { validRect = new NativeMethods.RECT(leftOwnerDrawArea); SafeNativeMethods.ValidateRect(new HandleRef(comboBox, comboBox.Handle), ref validRect); } if (rightOwnerDrawArea.IntersectsWith(updateRegionBox)) { validRect = new NativeMethods.RECT(rightOwnerDrawArea); SafeNativeMethods.ValidateRect(new HandleRef(comboBox, comboBox.Handle), ref validRect); } } } } } // File provided for Reference Use Only by Microsoft Corporation (c) 2007. //------------------------------------------------------------------------------ //// Copyright (c) Microsoft Corporation. All rights reserved. // //----------------------------------------------------------------------------- /* */ namespace System.Windows.Forms { using Accessibility; using System.Runtime.Serialization.Formatters; using System.Threading; using System.Runtime.Remoting; using System.Runtime.InteropServices; using System.Diagnostics; using System; using System.Globalization; using System.Security; using System.Security.Permissions; using System.Windows.Forms.Layout; using System.Windows.Forms.ComponentModel; using System.ComponentModel; using System.ComponentModel.Design; using System.Collections; using System.Drawing; using System.Windows.Forms.Internal; using System.Drawing.Design; using Microsoft.Win32; using System.Reflection; using System.Collections.Specialized; using System.Text; using System.Diagnostics.CodeAnalysis; ////// /// [ ComVisible(true), ClassInterface(ClassInterfaceType.AutoDispatch), DefaultEvent("SelectedIndexChanged"), DefaultProperty("Items"), DefaultBindingProperty("Text"), Designer("System.Windows.Forms.Design.ComboBoxDesigner, " + AssemblyRef.SystemDesign), SRDescription(SR.DescriptionComboBox) ] public class ComboBox : ListControl { private static readonly object EVENT_DROPDOWN = new object(); private static readonly object EVENT_DRAWITEM = new object(); private static readonly object EVENT_MEASUREITEM = new object(); private static readonly object EVENT_SELECTEDINDEXCHANGED = new object(); private static readonly object EVENT_SELECTIONCHANGECOMMITTED = new object(); private static readonly object EVENT_SELECTEDITEMCHANGED = new object(); private static readonly object EVENT_DROPDOWNSTYLE = new object(); private static readonly object EVENT_TEXTUPDATE = new object(); private static readonly object EVENT_DROPDOWNCLOSED = new object(); private static readonly int PropMaxLength = PropertyStore.CreateKey(); private static readonly int PropItemHeight = PropertyStore.CreateKey(); private static readonly int PropDropDownWidth = PropertyStore.CreateKey(); private static readonly int PropDropDownHeight = PropertyStore.CreateKey(); private static readonly int PropStyle = PropertyStore.CreateKey(); private static readonly int PropDrawMode = PropertyStore.CreateKey(); private static readonly int PropMatchingText = PropertyStore.CreateKey(); private static readonly int PropFlatComboAdapter = PropertyStore.CreateKey(); private const int DefaultSimpleStyleHeight = 150; private const int DefaultDropDownHeight = 106; private const int AutoCompleteTimeout = 10000000; // 1 second timeout for resetting the MatchingText private bool autoCompleteDroppedDown = false; private FlatStyle flatStyle = FlatStyle.Standard; private int updateCount; //Timestamp of the last keystroke. Used for auto-completion // in DropDownList style. private long autoCompleteTimeStamp; private int selectedIndex = -1; // used when we don't have a handle. private bool allowCommit = true; // When the style is "simple", the requested height is used // for the actual height of the control. // When the style is non-simple, the height of the control // is determined by the OS. // This fixes bug #20966 private int requestedHeight; private ComboBoxChildNativeWindow childEdit; private ComboBoxChildNativeWindow childListBox; private IntPtr dropDownHandle; private ObjectCollection itemsCollection; private short prefHeightCache = -1; private short maxDropDownItems = 8; private bool integralHeight = true; private bool mousePressed; private bool mouseEvents; private bool mouseInEdit; private bool sorted; private bool fireSetFocus = true; private bool fireLostFocus = true; private bool mouseOver; private bool suppressNextWindosPos; private bool canFireLostFocus; // When the user types a letter and drops the dropdown... // the combobox itself auto-searches the matching item... // and selects the item in the edit... // thus changing the windowText... // hence we should Fire the TextChanged event in such a scenario.. // The string below is used for checking the window Text before and after the dropdown. private string currentText = ""; private string lastTextChangedValue; private bool dropDown; private AutoCompleteDropDownFinder finder = new AutoCompleteDropDownFinder(); private bool selectedValueChangedFired; ////// Displays an editing field and a list, allowing the user to select from the /// list or to enter new text. Displays only the editing field until the user /// explicitly displays the list. /// /// ////// /// This stores the value for the autocomplete mode which can be either /// None, AutoSuggest, AutoAppend or AutoSuggestAppend. /// private AutoCompleteMode autoCompleteMode = AutoCompleteMode.None; ////// /// This stores the value for the autoCompleteSource mode which can be one of the values /// from AutoCompleteSource enum. /// private AutoCompleteSource autoCompleteSource = AutoCompleteSource.None; ////// /// This stores the custom StringCollection required for the autoCompleteSource when its set to CustomSource. /// private AutoCompleteStringCollection autoCompleteCustomSource; private StringSource stringSource; private bool fromHandleCreate = false; ////// /// Creates a new ComboBox control. The default style for the combo is /// a regular DropDown Combo. /// public ComboBox() { SetStyle(ControlStyles.UserPaint | ControlStyles.UseTextForAccessibility | ControlStyles.StandardClick, false); requestedHeight = DefaultSimpleStyleHeight; // this class overrides GetPreferredSizeCore, let Control automatically cache the result SetState2(STATE2_USEPREFERREDSIZECACHE, true); } ////// /// This is the AutoCompleteMode which can be either /// None, AutoSuggest, AutoAppend or AutoSuggestAppend. /// This property in conjunction with AutoCompleteSource enables the AutoComplete feature for ComboBox. /// [ DefaultValue(AutoCompleteMode.None), SRDescription(SR.ComboBoxAutoCompleteModeDescr), Browsable(true), EditorBrowsable(EditorBrowsableState.Always) ] public AutoCompleteMode AutoCompleteMode { get { return autoCompleteMode; } set { //valid values are 0x0 to 0x3 if (!ClientUtils.IsEnumValid(value, (int)value, (int)AutoCompleteMode.None, (int)AutoCompleteMode.SuggestAppend)) { throw new InvalidEnumArgumentException("value", (int)value, typeof(AutoCompleteMode)); } if (this.DropDownStyle == ComboBoxStyle.DropDownList && this.AutoCompleteSource != AutoCompleteSource.ListItems && value != AutoCompleteMode.None) { throw new NotSupportedException(SR.GetString(SR.ComboBoxAutoCompleteModeOnlyNoneAllowed)); } if (Application.OleRequired() != System.Threading.ApartmentState.STA) { throw new ThreadStateException(SR.GetString(SR.ThreadMustBeSTA)); } bool resetAutoComplete = false; if (autoCompleteMode != AutoCompleteMode.None && value == AutoCompleteMode.None) { resetAutoComplete = true; } autoCompleteMode = value; SetAutoComplete(resetAutoComplete, true); } } ////// /// This is the AutoCompleteSource which can be one of the /// values from AutoCompleteSource enumeration. /// This property in conjunction with AutoCompleteMode enables the AutoComplete feature for ComboBox. /// [ DefaultValue(AutoCompleteSource.None), SRDescription(SR.ComboBoxAutoCompleteSourceDescr), Browsable(true), EditorBrowsable(EditorBrowsableState.Always) ] public AutoCompleteSource AutoCompleteSource { get { return autoCompleteSource; } set { // FxCop: Avoid usage of Enum.IsDefined - this looks like an enum that could grow if (!ClientUtils.IsEnumValid_NotSequential(value, (int)value, (int)AutoCompleteSource.None, (int)AutoCompleteSource.AllSystemSources, (int)AutoCompleteSource.AllUrl, (int)AutoCompleteSource.CustomSource, (int)AutoCompleteSource.FileSystem, (int)AutoCompleteSource.FileSystemDirectories, (int)AutoCompleteSource.HistoryList, (int)AutoCompleteSource.ListItems, (int)AutoCompleteSource.RecentlyUsedList)) { throw new InvalidEnumArgumentException("value", (int)value, typeof(AutoCompleteSource)); } if (this.DropDownStyle == ComboBoxStyle.DropDownList && this.AutoCompleteMode != AutoCompleteMode.None && value != AutoCompleteSource.ListItems) { throw new NotSupportedException(SR.GetString(SR.ComboBoxAutoCompleteSourceOnlyListItemsAllowed)); } if (Application.OleRequired() != System.Threading.ApartmentState.STA) { throw new ThreadStateException(SR.GetString(SR.ThreadMustBeSTA)); } // VSWhidbey 466300 if (value != AutoCompleteSource.None && value != AutoCompleteSource.CustomSource && value != AutoCompleteSource.ListItems) { FileIOPermission fiop = new FileIOPermission(PermissionState.Unrestricted); fiop.AllFiles = FileIOPermissionAccess.PathDiscovery; fiop.Demand(); } autoCompleteSource = value; SetAutoComplete(false, true); } } ////// /// This is the AutoCompleteCustomSource which is custom StringCollection used when the /// AutoCompleteSource is CustomSource. /// [ DesignerSerializationVisibility(DesignerSerializationVisibility.Content), Localizable(true), SRDescription(SR.ComboBoxAutoCompleteCustomSourceDescr), Editor("System.Windows.Forms.Design.ListControlStringCollectionEditor, " + AssemblyRef.SystemDesign, typeof(UITypeEditor)), Browsable(true), EditorBrowsable(EditorBrowsableState.Always) ] public AutoCompleteStringCollection AutoCompleteCustomSource { get { if (autoCompleteCustomSource == null) { autoCompleteCustomSource = new AutoCompleteStringCollection(); autoCompleteCustomSource.CollectionChanged += new CollectionChangeEventHandler(this.OnAutoCompleteCustomSourceChanged); } return autoCompleteCustomSource; } set { if (autoCompleteCustomSource != value) { if (autoCompleteCustomSource != null) { autoCompleteCustomSource.CollectionChanged -= new CollectionChangeEventHandler(this.OnAutoCompleteCustomSourceChanged); } autoCompleteCustomSource = value; if (autoCompleteCustomSource != null) { autoCompleteCustomSource.CollectionChanged += new CollectionChangeEventHandler(this.OnAutoCompleteCustomSourceChanged); } SetAutoComplete(false, true); } } } ////// /// The background color of this control. This is an ambient property and /// will always return a non-null value. /// public override Color BackColor { get { if (ShouldSerializeBackColor()) { return base.BackColor; } else { return SystemColors.Window; } } set { base.BackColor = value; } } ////// /// [Browsable(false), EditorBrowsable(EditorBrowsableState.Never)] public override Image BackgroundImage { get { return base.BackgroundImage; } set { base.BackgroundImage = value; } } ///[To be supplied.] ////// /// [Browsable(false), EditorBrowsable(EditorBrowsableState.Never)] public override ImageLayout BackgroundImageLayout { get { return base.BackgroundImageLayout; } set { base.BackgroundImageLayout = value; } } ///[To be supplied.] ////// [Browsable(false), EditorBrowsable(EditorBrowsableState.Never)] new public event EventHandler BackgroundImageChanged { add { base.BackgroundImageChanged += value; } remove { base.BackgroundImageChanged -= value; } } /// /// [Browsable(false), EditorBrowsable(EditorBrowsableState.Never)] new public event EventHandler BackgroundImageLayoutChanged { add { base.BackgroundImageLayoutChanged += value; } remove { base.BackgroundImageLayoutChanged -= value; } } /// /// /// Returns the parameters needed to create the handle. Inheriting classes /// can override this to provide extra functionality. They should not, /// however, forget to call base.CreateParams() first to get the struct /// filled up with the basic info. /// ///protected override CreateParams CreateParams { [SecurityPermission(SecurityAction.LinkDemand, Flags = SecurityPermissionFlag.UnmanagedCode)] get { CreateParams cp = base.CreateParams; cp.ClassName = "COMBOBOX"; cp.Style |= NativeMethods.WS_VSCROLL | NativeMethods.CBS_HASSTRINGS | NativeMethods.CBS_AUTOHSCROLL; cp.ExStyle |= NativeMethods.WS_EX_CLIENTEDGE; if (!integralHeight) cp.Style |= NativeMethods.CBS_NOINTEGRALHEIGHT; switch (DropDownStyle) { case ComboBoxStyle.Simple: cp.Style |= NativeMethods.CBS_SIMPLE; break; case ComboBoxStyle.DropDown: cp.Style |= NativeMethods.CBS_DROPDOWN; // Make sure we put the height back or we won't be able to size the dropdown! cp.Height = PreferredHeight; break; case ComboBoxStyle.DropDownList: cp.Style |= NativeMethods.CBS_DROPDOWNLIST; // Comment above... cp.Height = PreferredHeight; break; } switch (DrawMode) { case DrawMode.OwnerDrawFixed: cp.Style |= NativeMethods.CBS_OWNERDRAWFIXED; break; case DrawMode.OwnerDrawVariable: cp.Style |= NativeMethods.CBS_OWNERDRAWVARIABLE; break; } return cp; } } /// /// /// Deriving classes can override this to configure a default size for their control. /// This is more efficient than setting the size in the control's constructor. /// protected override Size DefaultSize { get { return new Size(121, PreferredHeight); } } ////// /// The ListSource to consume as this ListBox's source of data. /// When set, a user can not modify the Items collection. /// [ SRCategory(SR.CatData), DefaultValue(null), RefreshProperties(RefreshProperties.Repaint), AttributeProvider(typeof(IListSource)), SRDescription(SR.ListControlDataSourceDescr) ] public new object DataSource { get { return base.DataSource; } set { base.DataSource = value; } } ////// /// Retrieves the value of the DrawMode property. The DrawMode property /// controls whether the control is drawn by Windows or by the user. /// [ SRCategory(SR.CatBehavior), DefaultValue(DrawMode.Normal), SRDescription(SR.ComboBoxDrawModeDescr), RefreshProperties(RefreshProperties.Repaint) ] public DrawMode DrawMode { get { bool found; int drawMode = Properties.GetInteger(PropDrawMode, out found); if (found) { return (DrawMode)drawMode; } return DrawMode.Normal; } set { if (DrawMode != value) { //valid values are 0x0 to 0x2. if (!ClientUtils.IsEnumValid(value, (int)value, (int)DrawMode.Normal, (int)DrawMode.OwnerDrawVariable)) { throw new InvalidEnumArgumentException("value", (int)value, typeof(DrawMode)); } ResetHeightCache(); Properties.SetInteger(PropDrawMode, (int)value); RecreateHandle(); } } } ////// /// Returns the width of the drop down box in a combo box. /// [ SRCategory(SR.CatBehavior), SRDescription(SR.ComboBoxDropDownWidthDescr) ] public int DropDownWidth { get { bool found; int dropDownWidth = Properties.GetInteger(PropDropDownWidth, out found); if (found) { return dropDownWidth; } else { return Width; } } set { if (value < 1) { throw new ArgumentOutOfRangeException("DropDownWidth", SR.GetString(SR.InvalidArgument, "DropDownWidth", (value).ToString(CultureInfo.CurrentCulture))); } if (Properties.GetInteger(PropDropDownWidth) != value) { Properties.SetInteger(PropDropDownWidth, value); if (IsHandleCreated) { SendMessage(NativeMethods.CB_SETDROPPEDWIDTH, value, 0); } } } } ////// /// Sets the Height of the drop down box in a combo box. /// [ SRCategory(SR.CatBehavior), SRDescription(SR.ComboBoxDropDownHeightDescr), Browsable(true), EditorBrowsable(EditorBrowsableState.Always), DefaultValue(106) ] public int DropDownHeight { get { bool found; int dropDownHeight = Properties.GetInteger(PropDropDownHeight, out found); if (found) { return dropDownHeight; } else { return DefaultDropDownHeight; } } set { if (value < 1) { throw new ArgumentOutOfRangeException("DropDownHeight", SR.GetString(SR.InvalidArgument, "DropDownHeight", (value).ToString(CultureInfo.CurrentCulture))); } if (Properties.GetInteger(PropDropDownHeight) != value) { Properties.SetInteger(PropDropDownHeight, value); // The dropDownHeight is not reflected unless the // combobox integralHeight == false.. IntegralHeight = false; } } } ////// /// Indicates whether the DropDown of the combo is currently dropped down. /// [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden), SRDescription(SR.ComboBoxDroppedDownDescr) ] public bool DroppedDown { get { if (IsHandleCreated) { return (int)SendMessage(NativeMethods.CB_GETDROPPEDSTATE, 0, 0) != 0; } else { return false; } } set { if (!IsHandleCreated) { CreateHandle(); } SendMessage(NativeMethods.CB_SHOWDROPDOWN, value ? -1 : 0, 0); } } ////// /// [ SRCategory(SR.CatAppearance), DefaultValue(FlatStyle.Standard), Localizable(true), SRDescription(SR.ComboBoxFlatStyleDescr) ] public FlatStyle FlatStyle { get { return flatStyle; } set { //valid values are 0x0 to 0x3 if (!ClientUtils.IsEnumValid(value, (int)value, (int)FlatStyle.Flat, (int)FlatStyle.System)) { throw new InvalidEnumArgumentException("value", (int)value, typeof(FlatStyle)); } flatStyle = value; Invalidate(); } } ////// Gets or /// sets /// the flat style appearance of the button control. /// ////// /// Returns true if this control has focus. /// public override bool Focused { get { if (base.Focused) return true; IntPtr focus = UnsafeNativeMethods.GetFocus(); return focus != IntPtr.Zero && ((childEdit != null && focus == childEdit.Handle) || (childListBox != null && focus == childListBox.Handle)); } } ////// /// public override Color ForeColor { get { if (ShouldSerializeForeColor()) { return base.ForeColor; } else { return SystemColors.WindowText; } } set { base.ForeColor = value; } } ////// Gets or sets the foreground color of the control. /// ////// /// Indicates if the combo should avoid showing partial Items. If so, /// then only full items will be displayed, and the list portion will be resized /// to prevent partial items from being shown. Otherwise, they will be /// shown /// [ SRCategory(SR.CatBehavior), DefaultValue(true), Localizable(true), SRDescription(SR.ComboBoxIntegralHeightDescr) ] public bool IntegralHeight { get { return integralHeight; } set { if (integralHeight != value) { integralHeight = value; RecreateHandle(); } } } ////// /// Returns the height of an item in the combo box. When drawMode is Normal /// or OwnerDrawFixed, all items have the same height. When drawMode is /// OwnerDrawVariable, this method returns the height that will be given /// to new items added to the combo box. To determine the actual height of /// an item, use the GetItemHeight() method with an integer parameter. /// [ SRCategory(SR.CatBehavior), Localizable(true), SRDescription(SR.ComboBoxItemHeightDescr) ] public int ItemHeight { get { DrawMode drawMode = DrawMode; if (drawMode == DrawMode.OwnerDrawFixed || drawMode == DrawMode.OwnerDrawVariable || !IsHandleCreated) { bool found; int itemHeight = Properties.GetInteger(PropItemHeight, out found); if (found) { return itemHeight; } else { return FontHeight + 2; // bug (90774)+2 for the 1 pixel gap on each side (up and Bottom) of the Text. } } // Note that the above if clause deals with the case when the handle has not yet been created Debug.Assert(IsHandleCreated, "Handle should be created at this point"); int h = (int)SendMessage(NativeMethods.CB_GETITEMHEIGHT, 0, 0); if (h == -1) { throw new Win32Exception(); } return h; } set { if (value < 1) { throw new ArgumentOutOfRangeException("ItemHeight", SR.GetString(SR.InvalidArgument, "ItemHeight", (value).ToString(CultureInfo.CurrentCulture))); } ResetHeightCache(); if (Properties.GetInteger(PropItemHeight) != value) { Properties.SetInteger(PropItemHeight, value); if (DrawMode != DrawMode.Normal) { UpdateItemHeight(); } } } } ////// /// Collection of the items contained in this ComboBox. /// [ SRCategory(SR.CatData), DesignerSerializationVisibility(DesignerSerializationVisibility.Content), Localizable(true), SRDescription(SR.ComboBoxItemsDescr), Editor("System.Windows.Forms.Design.ListControlStringCollectionEditor, " + AssemblyRef.SystemDesign, typeof(UITypeEditor)), MergableProperty(false) ] public ObjectCollection Items { get { if (itemsCollection == null) { itemsCollection = new ObjectCollection(this); } return itemsCollection; } } // Text used to match an item in the list when auto-completion // is used in DropDownList style. private string MatchingText { get { string matchingText = (string)Properties.GetObject(PropMatchingText); return (matchingText == null) ? string.Empty : matchingText; } set { if (value != null || this.Properties.ContainsObject(PropMatchingText)) { Properties.SetObject(PropMatchingText, value); } } } ////// /// The maximum number of items to be shown in the dropdown portion /// of the ComboBox. This number can be between 1 and 100. /// [ SRCategory(SR.CatBehavior), DefaultValue(8), Localizable(true), SRDescription(SR.ComboBoxMaxDropDownItemsDescr) ] public int MaxDropDownItems { get { return maxDropDownItems; } set { if (value < 1 || value > 100) { throw new ArgumentOutOfRangeException("MaxDropDownItems", SR.GetString(SR.InvalidBoundArgument, "MaxDropDownItems", (value).ToString(CultureInfo.CurrentCulture), (1).ToString(CultureInfo.CurrentCulture), (100).ToString(CultureInfo.CurrentCulture))); } maxDropDownItems = (short)value; } } public override Size MaximumSize { get { return base.MaximumSize; } set { base.MaximumSize = new Size(value.Width, 0); } } public override Size MinimumSize { get { return base.MinimumSize; } set { base.MinimumSize = new Size(value.Width, 0); } } ////// /// The maximum length of the text the user may type into the edit control /// of a combo box. /// [ SRCategory(SR.CatBehavior), DefaultValue(0), Localizable(true), SRDescription(SR.ComboBoxMaxLengthDescr) ] public int MaxLength { get { return Properties.GetInteger(PropMaxLength); } set { if (value < 0) value = 0; if (MaxLength != value) { Properties.SetInteger(PropMaxLength, value); if (IsHandleCreated) SendMessage(NativeMethods.CB_LIMITTEXT, value, 0); } } } ////// /// If the mouse is over the combobox, draw selection rect. /// internal bool MouseIsOver { get { return mouseOver; } set { if (mouseOver != value) { mouseOver = value; // Nothing to see here... Just keep on walking... VSWhidbey 504477. // Turns out that with Theming off, we don't get quite the same messages as with theming on, so // our drawing gets a little messed up. So in case theming is off, force a draw here. if ((!ContainsFocus || !Application.RenderWithVisualStyles) && this.FlatStyle == FlatStyle.Popup) { Invalidate(); Update(); } } } } ////// /// [ Browsable(false), EditorBrowsable(EditorBrowsableState.Never), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden) ] public new Padding Padding { get { return base.Padding; } set { base.Padding = value; } } [ Browsable(false), EditorBrowsable(EditorBrowsableState.Never) ] public new event EventHandler PaddingChanged { add { base.PaddingChanged += value; } remove { base.PaddingChanged -= value; } } ////// ///[To be supplied.] ////// /// ApplySizeConstraints calls into this method when DropDownStyles is DropDown and DropDownList. /// This causes PreferredSize to be bounded by PreferredHeight in these two cases only. /// [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden), SRDescription(SR.ComboBoxPreferredHeightDescr) ] public int PreferredHeight { get { if (!FormattingEnabled) { //see #465057 for why I did this //do preferred height the old broken way for everett apps //we need this for compat reasons because (get this) // (a) everett preferredheight was always wrong. // (b) so, when combobox1.Size = actualdefaultsize was called, it would enter setboundscore // (c) this updated requestedheight // (d) if the user then changed the combo to simple style, the height did not change. // We simply cannot match this behavior if preferredheight is corrected so that (b) never // occurs. We simply do not know when Size was set. // So in whidbey, the behavior will be: // (1) user uses default size = setting dropdownstyle=simple will revert to simple height // (2) user uses nondefault size = setting dropdownstyle=simple will not change height from this value //In everett // if the user manually sets Size = (121, 20) in code (usually height gets forced to 21), then he will see Whidey.(1) above // user usually uses nondefault size and will experience whidbey.(2) above Size textSize = TextRenderer.MeasureText(LayoutUtils.TestString, this.Font, new Size(Int16.MaxValue, (int)(FontHeight * 1.25)), TextFormatFlags.SingleLine); prefHeightCache = (short)(textSize.Height + SystemInformation.BorderSize.Height * 8 + Padding.Size.Height); return prefHeightCache ; } else { // Normally we do this sort of calculation in GetPreferredSizeCore which has builtin // caching, but in this case we can not because PreferredHeight is used in ApplySizeConstraints // which is used by GetPreferredSize (infinite loop). if (prefHeightCache < 0) { Size textSize = TextRenderer.MeasureText(LayoutUtils.TestString, this.Font, new Size(Int16.MaxValue, (int)(FontHeight * 1.25)), TextFormatFlags.SingleLine); // For a "simple" style combobox, the preferred height depends on the // number of items in the combobox. if (DropDownStyle == ComboBoxStyle.Simple) { int itemCount = Items.Count + 1; prefHeightCache = (short)(textSize.Height * itemCount + SystemInformation.BorderSize.Height * 16 + Padding.Size.Height); } else { // We do this old school rather than use SizeFromClientSize because CreateParams calls this // method and SizeFromClientSize calls CreateParams (another infinite loop.) prefHeightCache = (short)GetComboHeight(); } } return prefHeightCache; } } } // VSWhidbey 194386 - ComboBox.PreferredHeight returns incorrect values // This is translated from windows implementation. Since we cannot control the size // of the combo box, we need to use the same calculation they do. [SuppressMessage("Microsoft.Globalization", "CA1303:DoNotPassLiteralsAsLocalizedParameters")] // "0" is what Windows uses in all languages. private int GetComboHeight() { int cyCombo = 0; // Add on CYEDGE just for some extra space in the edit field/static item. // It's really only for static text items, but we want static & editable // controls to be the same height. Size textExtent = Size.Empty; using (WindowsFont font = WindowsFont.FromFont(this.Font)) { // this is the character that Windows uses to determine the extent textExtent = WindowsGraphicsCacheManager.MeasurementGraphics.GetTextExtent("0", font); } int dyEdit = textExtent.Height + SystemInformation.Border3DSize.Height; if (DrawMode != DrawMode.Normal) { // This is an ownerdraw combo. Have the owner tell us how tall this // item is. dyEdit = ItemHeight; } // Set the initial width to be the combo box rect. Later we will shorten it // if there is a dropdown button. Size fixedFrameBoderSize = SystemInformation.FixedFrameBorderSize; cyCombo = 2 * fixedFrameBoderSize.Height + dyEdit; return cyCombo; } private string[] GetStringsForAutoComplete(IList collection) { if (collection is AutoCompleteStringCollection) { string[] strings = new string[AutoCompleteCustomSource.Count]; for (int i = 0; i < AutoCompleteCustomSource.Count; i++) { strings[i] = AutoCompleteCustomSource[i]; } return strings; } else if (collection is ObjectCollection) { string[] strings = new string[itemsCollection.Count]; for (int i = 0; i < itemsCollection.Count; i++) { strings[i] = GetItemText(itemsCollection[i]); } return strings; } return new string[0]; } ////// /// The [zero based] index of the currently selected item in the combos list. /// Note If the value of index is -1, then the ComboBox is /// set to have no selection. /// [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden), SRDescription(SR.ComboBoxSelectedIndexDescr) ] public override int SelectedIndex { get { if (IsHandleCreated) { return (int)SendMessage(NativeMethods.CB_GETCURSEL, 0, 0); } else { return selectedIndex; } } set { if (SelectedIndex != value) { int itemCount = 0; if (itemsCollection != null) { itemCount = itemsCollection.Count; } if (value < -1 || value >= itemCount) { throw new ArgumentOutOfRangeException("SelectedIndex", SR.GetString(SR.InvalidArgument, "SelectedIndex", (value).ToString(CultureInfo.CurrentCulture))); } if (IsHandleCreated) { SendMessage(NativeMethods.CB_SETCURSEL, value, 0); } else { selectedIndex = value; } UpdateText(); if (IsHandleCreated) { OnTextChanged(EventArgs.Empty); } OnSelectedItemChanged(EventArgs.Empty); OnSelectedIndexChanged(EventArgs.Empty); } } } ////// /// The handle to the object that is currently selected in the /// combos list. /// [ Browsable(false), Bindable(true), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden), SRDescription(SR.ComboBoxSelectedItemDescr) ] public object SelectedItem { get { int index = SelectedIndex; return (index == -1) ? null : Items[index]; } set { int x = -1; if (itemsCollection != null) { //bug (82115) if (value != null) x = itemsCollection.IndexOf(value); else SelectedIndex = -1; } if (x != -1) { SelectedIndex = x; } } } ////// /// The selected text in the edit component of the ComboBox. If the /// ComboBox has ComboBoxStyle.DROPDOWNLIST, the return is an empty /// string (""). /// [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden), SRDescription(SR.ComboBoxSelectedTextDescr) ] public string SelectedText { get { if (DropDownStyle == ComboBoxStyle.DropDownList) return ""; return Text.Substring(SelectionStart, SelectionLength); } set { if (DropDownStyle != ComboBoxStyle.DropDownList) { //guard against null string, since otherwise we will throw an //AccessViolation exception, which is bad string str = (value == null ? "" : value); CreateControl(); if (IsHandleCreated) { Debug.Assert(childEdit != null); if (childEdit != null) { UnsafeNativeMethods.SendMessage(new HandleRef(this, childEdit.Handle), NativeMethods.EM_REPLACESEL, NativeMethods.InvalidIntPtr, str); } } } } } ////// /// The length, in characters, of the selection in the editbox. /// [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden), SRDescription(SR.ComboBoxSelectionLengthDescr) ] public int SelectionLength { get { int[] end = new int[] { 0 }; int[] start = new int[] { 0 }; UnsafeNativeMethods.SendMessage(new HandleRef(this, Handle), NativeMethods.CB_GETEDITSEL, start, end); return end[0] - start[0]; } set { // SelectionLength can be negtive... Select(SelectionStart, value); } } ////// /// The [zero-based] index of the first character in the current text selection. /// [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden), SRDescription(SR.ComboBoxSelectionStartDescr) ] public int SelectionStart { get { int[] value = new int[] { 0 }; UnsafeNativeMethods.SendMessage(new HandleRef(this, Handle), NativeMethods.CB_GETEDITSEL, value, (int[])null); return value[0]; } set { if (value < 0) { throw new ArgumentOutOfRangeException("SelectionStart", SR.GetString(SR.InvalidArgument, "SelectionStart", value.ToString(CultureInfo.CurrentCulture))); } Select(value, SelectionLength); } } ////// /// Indicates if the Combos list is sorted or not. /// [ SRCategory(SR.CatBehavior), DefaultValue(false), SRDescription(SR.ComboBoxSortedDescr) ] public bool Sorted { get { return sorted; } set { if (sorted != value) { if (this.DataSource != null && value) { throw new ArgumentException(SR.GetString(SR.ComboBoxSortWithDataSource)); } sorted = value; RefreshItems(); SelectedIndex = -1; } } } ////// /// The type of combo that we are right now. The value would come /// from the System.Windows.Forms.ComboBoxStyle enumeration. /// [ SRCategory(SR.CatAppearance), DefaultValue(ComboBoxStyle.DropDown), SRDescription(SR.ComboBoxStyleDescr), RefreshPropertiesAttribute(RefreshProperties.Repaint) ] public ComboBoxStyle DropDownStyle { get { bool found; int style = Properties.GetInteger(PropStyle, out found); if (found) { return (ComboBoxStyle)style; } return ComboBoxStyle.DropDown; } set { if (DropDownStyle != value) { // verify that 'value' is a valid enum type... //valid values are 0x0 to 0x2 if (!ClientUtils.IsEnumValid(value, (int)value, (int)ComboBoxStyle.Simple, (int)ComboBoxStyle.DropDownList)) { throw new InvalidEnumArgumentException("value", (int)value, typeof(ComboBoxStyle)); } if (value == ComboBoxStyle.DropDownList && this.AutoCompleteSource != AutoCompleteSource.ListItems && this.AutoCompleteMode != AutoCompleteMode.None) { this.AutoCompleteMode = AutoCompleteMode.None; } // reset preferred height. ResetHeightCache(); Properties.SetInteger(PropStyle, (int)value); if (IsHandleCreated) { RecreateHandle(); } OnDropDownStyleChanged(EventArgs.Empty); } } } ////// /// [ Localizable(true), Bindable(true) ] public override string Text { get { if (SelectedItem != null && !BindingFieldEmpty) { //preserve everett behavior if "formatting enabled == false" -- just return selecteditem text. if (FormattingEnabled) { string candidate = GetItemText(SelectedItem); if (!String.IsNullOrEmpty(candidate)) { if (String.Compare(candidate, base.Text, true, CultureInfo.CurrentCulture) == 0) { return candidate; //for whidbey, if we only differ by case -- return the candidate; } } } else { return FilterItemOnProperty(SelectedItem).ToString(); //heinous. } } return base.Text; } set { if (DropDownStyle == ComboBoxStyle.DropDownList && !IsHandleCreated && !String.IsNullOrEmpty(value) && FindStringExact(value) == -1) return; base.Text = value; object selectedItem = null; selectedItem = SelectedItem; if (!DesignMode) { //bug <70650> Subhag removed 'value.Length == 0' check to handle String.Empty. // if (value == null) { SelectedIndex = -1; } else if (value != null && (selectedItem == null || (String.Compare(value, GetItemText(selectedItem), false, CultureInfo.CurrentCulture) != 0))) { int index = FindStringIgnoreCase(value); //we cannot set the index to -1 unless we want to do a hack and save/restore text //because the native control will erase the text when we change the index to -1 if (index != -1) { SelectedIndex = index; } } } } } private int FindStringIgnoreCase(string value) { //look for an exact match and then a case insensitive match if that fails. int index = FindStringExact(value, -1, false); if (index == -1) { index = FindStringExact(value, -1, true); } return index; } // Special AutoComplete notification handling // If the text changed, this will fire TextChanged // If it matches an item in the list, this will fire SIC and SVC private void NotifyAutoComplete() { NotifyAutoComplete(true); } // Special AutoComplete notification handling // If the text changed, this will fire TextChanged // If it matches an item in the list, this will fire SIC and SVC private void NotifyAutoComplete(bool setSelectedIndex) { string text = this.Text; bool textChanged = (text != this.lastTextChangedValue); bool selectedIndexSet = false; if (setSelectedIndex) { // Process return key. This is sent by the AutoComplete DropDown when a // selection is made from the DropDown // Check to see if the Text Changed. If so, at least fire a TextChanged int index = FindStringIgnoreCase(text); if ((index != -1) && (index != SelectedIndex)) { // We found a match, do the full monty SelectedIndex = index; // Put the cursor at the end SelectionStart = 0; SelectionLength = text.Length; selectedIndexSet = true; } } //don't fire textch if we had set the selectedindex -- because it was already fired if so. if (textChanged && !selectedIndexSet) { // No match, just fire a TextChagned OnTextChanged(EventArgs.Empty); } // Save the new value this.lastTextChangedValue = text; } // Returns true if using System AutoComplete private bool SystemAutoCompleteEnabled { get { return ((this.autoCompleteMode != AutoCompleteMode.None) && (this.DropDownStyle != ComboBoxStyle.DropDownList)); } } // VSWhidbey 95691: Prevent this event from being displayed in the Property Grid. ///[To be supplied.] ////// [Browsable(false), EditorBrowsable(EditorBrowsableState.Never)] public new event EventHandler DoubleClick { add { base.DoubleClick += value; } remove { base.DoubleClick -= value; } } /// /// /// [SRCategory(SR.CatBehavior), SRDescription(SR.drawItemEventDescr)] public event DrawItemEventHandler DrawItem { add { Events.AddHandler(EVENT_DRAWITEM, value); } remove { Events.RemoveHandler(EVENT_DRAWITEM, value); } } ///[To be supplied.] ////// /// [SRCategory(SR.CatBehavior), SRDescription(SR.ComboBoxOnDropDownDescr)] public event EventHandler DropDown { add { Events.AddHandler(EVENT_DROPDOWN, value); } remove { Events.RemoveHandler(EVENT_DROPDOWN, value); } } ///[To be supplied.] ////// /// [SRCategory(SR.CatBehavior), SRDescription(SR.measureItemEventDescr)] public event MeasureItemEventHandler MeasureItem { add { Events.AddHandler(EVENT_MEASUREITEM, value); UpdateItemHeight(); } remove { Events.RemoveHandler(EVENT_MEASUREITEM, value); UpdateItemHeight(); } } ///[To be supplied.] ////// /// [SRCategory(SR.CatBehavior), SRDescription(SR.selectedIndexChangedEventDescr)] public event EventHandler SelectedIndexChanged { add { Events.AddHandler(EVENT_SELECTEDINDEXCHANGED, value); } remove { Events.RemoveHandler(EVENT_SELECTEDINDEXCHANGED, value); } } ///[To be supplied.] ////// /// [SRCategory(SR.CatBehavior), SRDescription(SR.selectionChangeCommittedEventDescr)] public event EventHandler SelectionChangeCommitted { add { Events.AddHandler(EVENT_SELECTIONCHANGECOMMITTED, value); } remove { Events.RemoveHandler(EVENT_SELECTIONCHANGECOMMITTED, value); } } ///[To be supplied.] ////// /// [SRCategory(SR.CatBehavior), SRDescription(SR.ComboBoxDropDownStyleChangedDescr)] public event EventHandler DropDownStyleChanged { add { Events.AddHandler(EVENT_DROPDOWNSTYLE, value); } remove { Events.RemoveHandler(EVENT_DROPDOWNSTYLE, value); } } ///[To be supplied.] ////// /// ComboBox Onpaint. /// ///[Browsable(false), EditorBrowsable(EditorBrowsableState.Never)] public new event PaintEventHandler Paint { add { base.Paint += value; } remove { base.Paint -= value; } } /// /// /// [SRCategory(SR.CatBehavior), SRDescription(SR.ComboBoxOnTextUpdateDescr)] public event EventHandler TextUpdate { add { Events.AddHandler(EVENT_TEXTUPDATE, value); } remove { Events.RemoveHandler(EVENT_TEXTUPDATE, value); } } ///This will fire the TextUpdate Event on the ComboBox. This events fires when the Combobox gets the /// CBN_EDITUPDATE notification. // ////// /// /// [SRCategory(SR.CatBehavior), SRDescription(SR.ComboBoxOnDropDownClosedDescr)] public event EventHandler DropDownClosed { add { Events.AddHandler(EVENT_DROPDOWNCLOSED, value); } remove { Events.RemoveHandler(EVENT_DROPDOWNCLOSED, value); } } ///This will fire the DropDownClosed Event on the ComboBox. This events fires when the Combobox gets the /// CBN_CLOSEUP notification. This happens when the DropDown closes. // ////// /// /// Performs the work of adding the specified items to the combobox /// [Obsolete("This method has been deprecated. There is no replacement. http://go.microsoft.com/fwlink/?linkid=14202")] protected virtual void AddItemsCore(object[] value) { int count = value == null ? 0 : value.Length; if (count == 0) { return; } BeginUpdate(); try { Items.AddRangeInternal(value); } finally { EndUpdate(); } } ////// /// Disables redrawing of the combo box. A call to beginUpdate() must be /// balanced by a following call to endUpdate(). Following a call to /// beginUpdate(), any redrawing caused by operations performed on the /// combo box is deferred until the call to endUpdate(). /// public void BeginUpdate() { updateCount++; BeginUpdateInternal(); } private void CheckNoDataSource() { if (DataSource != null) { throw new ArgumentException(SR.GetString(SR.DataSourceLocksItems)); } } ////// protected override AccessibleObject CreateAccessibilityInstance() { return new ComboBoxAccessibleObject(this); } ////// internal bool UpdateNeeded() { return (updateCount == 0); } ////// This procedure takes in the message, converts the Edit handle coordinates into Combo Box Coordinates /// ///internal Point EditToComboboxMapping(Message m) { if (childEdit == null) { return new Point(0, 0); } // Get the Combox Rect ... // NativeMethods.RECT comboRectMid = new NativeMethods.RECT(); UnsafeNativeMethods.GetWindowRect(new HandleRef(this, Handle), ref comboRectMid); // //Get the Edit Rectangle... // NativeMethods.RECT editRectMid = new NativeMethods.RECT(); UnsafeNativeMethods.GetWindowRect(new HandleRef(this, childEdit.Handle), ref editRectMid); //get the delta int comboXMid = NativeMethods.Util.SignedLOWORD(m.LParam) + (editRectMid.left - comboRectMid.left); int comboYMid = NativeMethods.Util.SignedHIWORD(m.LParam) + (editRectMid.top - comboRectMid.top); return (new Point(comboXMid, comboYMid)); } /// /// Subclassed window procedure for the edit and list child controls of the /// combo box. /// ///private void ChildWndProc(ref Message m) { switch (m.Msg) { case NativeMethods.WM_CHAR: if (DropDownStyle == ComboBoxStyle.Simple && m.HWnd == childListBox.Handle) { DefChildWndProc(ref m); } else { if (PreProcessControlMessage(ref m) != PreProcessControlState.MessageProcessed) { if (ProcessKeyMessage(ref m)) { return; } DefChildWndProc(ref m); } } break; case NativeMethods.WM_SYSCHAR: if (DropDownStyle == ComboBoxStyle.Simple && m.HWnd == childListBox.Handle) { DefChildWndProc(ref m); } else { if (PreProcessControlMessage(ref m) != PreProcessControlState.MessageProcessed) { if (ProcessKeyEventArgs(ref m)) { return; } DefChildWndProc(ref m); } } break; case NativeMethods.WM_KEYDOWN: case NativeMethods.WM_SYSKEYDOWN: if (SystemAutoCompleteEnabled && !ACNativeWindow.AutoCompleteActive) { finder.FindDropDowns(false); } if (AutoCompleteMode != AutoCompleteMode.None) { char keyChar = unchecked((char)(long)m.WParam); if (keyChar == (char)(int)Keys.Escape) { this.DroppedDown = false; } else if (keyChar == (char)(int)Keys.Return && this.DroppedDown) { UpdateText(); OnSelectionChangeCommittedInternal(EventArgs.Empty); this.DroppedDown = false; } } if (DropDownStyle == ComboBoxStyle.Simple && m.HWnd == childListBox.Handle) { DefChildWndProc(ref m); } else { if (PreProcessControlMessage(ref m) != PreProcessControlState.MessageProcessed) { if (ProcessKeyMessage(ref m)) { return; } DefChildWndProc(ref m); } } break; case NativeMethods.WM_INPUTLANGCHANGE: DefChildWndProc( ref m ); break; case NativeMethods.WM_KEYUP: case NativeMethods.WM_SYSKEYUP: if (DropDownStyle == ComboBoxStyle.Simple && m.HWnd == childListBox.Handle) { DefChildWndProc(ref m); } else { if (PreProcessControlMessage(ref m) != PreProcessControlState.MessageProcessed) { if (ProcessKeyMessage(ref m)) { return; } DefChildWndProc(ref m); } } if (SystemAutoCompleteEnabled && !ACNativeWindow.AutoCompleteActive) { finder.FindDropDowns(); } break; case NativeMethods.WM_KILLFOCUS: // Consider - If we dont' have a childwndproc, then we don't get here, so we don't // update the cache. Do we need to? This happens when we have a DropDownList. if (!DesignMode) { OnImeContextStatusChanged( m.HWnd ); } DefChildWndProc(ref m); // We don't want to fire the focus events twice - // once in the combobox and once here. if (fireLostFocus) { OnLostFocus(EventArgs.Empty); } if (FlatStyle == FlatStyle.Popup) { this.Invalidate(); } break; case NativeMethods.WM_SETFOCUS: // Consider - If we dont' have a childwndproc, then we don't get here, so we don't // set the status. Do we need to? This happens when we have a DropDownList. if (!DesignMode) { ImeContext.SetImeStatus(CachedImeMode, m.HWnd); } if (!HostedInWin32DialogManager) { IContainerControl c = GetContainerControlInternal(); if (c != null) { ContainerControl container = c as ContainerControl; if (container != null) { if (!container.ActivateControlInternal(this, false)) { return; } } } } DefChildWndProc(ref m); // We don't want to fire the focus events twice - // once in the combobox and once here. if (fireSetFocus) { OnGotFocus(EventArgs.Empty); } if (FlatStyle == FlatStyle.Popup) { this.Invalidate(); } break; case NativeMethods.WM_SETFONT: DefChildWndProc(ref m); if (childEdit != null && m.HWnd == childEdit.Handle) { UnsafeNativeMethods.SendMessage(new HandleRef(this, childEdit.Handle), NativeMethods.EM_SETMARGINS, NativeMethods.EC_LEFTMARGIN | NativeMethods.EC_RIGHTMARGIN, 0); } break; case NativeMethods.WM_LBUTTONDBLCLK: //the Listbox gets WM_LBUTTONDOWN - WM_LBUTTONUP -WM_LBUTTONDBLCLK - WM_LBUTTONUP... //sequence for doubleclick... //Set MouseEvents... mousePressed = true; mouseEvents = true; CaptureInternal = true; //Call the DefWndProc() so that mousemove messages get to the windows edit(112079) // DefChildWndProc(ref m); //the up gets fired from "Combo-box's WndPrc --- So Convert these Coordinates to Combobox coordianate... // Point Ptlc = EditToComboboxMapping(m); OnMouseDown(new MouseEventArgs(MouseButtons.Left, 1, Ptlc.X, Ptlc.Y, 0)); break; case NativeMethods.WM_MBUTTONDBLCLK: //the Listbox gets WM_LBUTTONDOWN - WM_LBUTTONUP -WM_LBUTTONDBLCLK - WM_LBUTTONUP... //sequence for doubleclick... //Set MouseEvents... mousePressed = true; mouseEvents = true; CaptureInternal = true; //Call the DefWndProc() so that mousemove messages get to the windows edit(112079) // DefChildWndProc(ref m); //the up gets fired from "Combo-box's WndPrc --- So Convert these Coordinates to Combobox coordianate... // Point Ptmc = EditToComboboxMapping(m); OnMouseDown(new MouseEventArgs(MouseButtons.Middle, 1, Ptmc.X, Ptmc.Y, 0)); break; case NativeMethods.WM_RBUTTONDBLCLK: //the Listbox gets WM_LBUTTONDOWN - WM_LBUTTONUP -WM_LBUTTONDBLCLK - WM_LBUTTONUP... //sequence for doubleclick... //Set MouseEvents... mousePressed = true; mouseEvents = true; CaptureInternal = true; //Call the DefWndProc() so that mousemove messages get to the windows edit(112079) // DefChildWndProc(ref m); //the up gets fired from "Combo-box's WndPrc --- So Convert these Coordinates to Combobox coordianate... // Point Ptrc = EditToComboboxMapping(m); OnMouseDown(new MouseEventArgs(MouseButtons.Right, 1, Ptrc.X, Ptrc.Y, 0)); break; case NativeMethods.WM_LBUTTONDOWN: mousePressed = true; mouseEvents = true; //set the mouse capture .. this is the Child Wndproc.. // CaptureInternal = true; DefChildWndProc(ref m); //the up gets fired from "Combo-box's WndPrc --- So Convert these Coordinates to Combobox coordianate... // Point Ptl = EditToComboboxMapping(m); OnMouseDown(new MouseEventArgs(MouseButtons.Left, 1, Ptl.X, Ptl.Y, 0)); break; case NativeMethods.WM_LBUTTONUP: // Get the mouse location // NativeMethods.RECT r = new NativeMethods.RECT(); UnsafeNativeMethods.GetWindowRect(new HandleRef(this, Handle), ref r); Rectangle ClientRect = new Rectangle(r.left, r.top, r.right - r.left, r.bottom - r.top); // Get the mouse location // int x = NativeMethods.Util.SignedLOWORD(m.LParam); int y = NativeMethods.Util.SignedHIWORD(m.LParam); Point pt = new Point(x, y); pt = PointToScreen(pt); // combo box gets a WM_LBUTTONUP for focus change ... // So check MouseEvents.... if (mouseEvents && !ValidationCancelled) { mouseEvents = false; if (mousePressed) { if (ClientRect.Contains(pt)) { mousePressed = false; OnClick(new MouseEventArgs(MouseButtons.Left, 1, NativeMethods.Util.SignedLOWORD(m.LParam), NativeMethods.Util.SignedHIWORD(m.LParam), 0)); OnMouseClick(new MouseEventArgs(MouseButtons.Left, 1, NativeMethods.Util.SignedLOWORD(m.LParam), NativeMethods.Util.SignedHIWORD(m.LParam), 0)); } else { mousePressed = false; mouseInEdit = false; OnMouseLeave(EventArgs.Empty); } } } DefChildWndProc(ref m); CaptureInternal = false; //the up gets fired from "Combo-box's WndPrc --- So Convert these Coordinates to Combobox coordianate... // pt = EditToComboboxMapping(m); OnMouseUp(new MouseEventArgs(MouseButtons.Left, 1, pt.X, pt.Y, 0)); break; case NativeMethods.WM_MBUTTONDOWN: mousePressed = true; mouseEvents = true; //set the mouse capture .. this is the Child Wndproc.. // CaptureInternal = true; DefChildWndProc(ref m); //the up gets fired from "Combo-box's WndPrc --- So Convert these Coordinates to Combobox coordianate... // Point P = EditToComboboxMapping(m); OnMouseDown(new MouseEventArgs(MouseButtons.Middle, 1, P.X, P.Y, 0)); break; case NativeMethods.WM_RBUTTONDOWN: mousePressed = true; mouseEvents = true; //set the mouse capture .. this is the Child Wndproc.. // Bug# 112108: If I set the capture=true here, the // DefWndProc() never fires the WM_CONTEXTMENU that would show // the default context menu. // if (this.ContextMenu != null || this.ContextMenuStrip != null) CaptureInternal = true; DefChildWndProc(ref m); //the up gets fired from "Combo-box's WndPrc --- So Convert these Coordinates to Combobox coordianate... // Point Pt = EditToComboboxMapping(m); OnMouseDown(new MouseEventArgs(MouseButtons.Right, 1, Pt.X, Pt.Y, 0)); break; case NativeMethods.WM_MBUTTONUP: mousePressed = false; mouseEvents = false; //set the mouse capture .. this is the Child Wndproc.. // CaptureInternal = false; DefChildWndProc(ref m); OnMouseUp(new MouseEventArgs(MouseButtons.Middle, 1, NativeMethods.Util.SignedLOWORD(m.LParam), NativeMethods.Util.SignedHIWORD(m.LParam), 0)); break; case NativeMethods.WM_RBUTTONUP: mousePressed = false; mouseEvents = false; //set the mouse capture .. this is the Child Wndproc.. // if (this.ContextMenu != null) CaptureInternal = false; DefChildWndProc(ref m); //the up gets fired from "Combo-box's WndPrc --- So Convert these Coordinates to Combobox coordianate... // Point ptRBtnUp = EditToComboboxMapping(m); OnMouseUp(new MouseEventArgs(MouseButtons.Right, 1, ptRBtnUp.X, ptRBtnUp.Y, 0)); break; case NativeMethods.WM_CONTEXTMENU: // Forward context menu messages to the parent control if (this.ContextMenu != null || this.ContextMenuStrip != null) { UnsafeNativeMethods.SendMessage(new HandleRef(this, Handle), NativeMethods.WM_CONTEXTMENU, m.WParam, m.LParam); } else { DefChildWndProc(ref m); } break; case NativeMethods.WM_MOUSEMOVE: Point point = EditToComboboxMapping(m); //Call the DefWndProc() so that mousemove messages get to the windows edit(112079) // DefChildWndProc(ref m); OnMouseEnterInternal(EventArgs.Empty); OnMouseMove(new MouseEventArgs(MouseButtons, 0, point.X, point.Y, 0)); break; case NativeMethods.WM_SETCURSOR: if (Cursor != DefaultCursor && childEdit != null && m.HWnd == childEdit.Handle && NativeMethods.Util.LOWORD(m.LParam) == NativeMethods.HTCLIENT) { Cursor.CurrentInternal = Cursor; } else { DefChildWndProc(ref m); } break; case NativeMethods.WM_MOUSELEAVE: DefChildWndProc(ref m); OnMouseLeaveInternal(EventArgs.Empty); break; default: DefChildWndProc(ref m); break; } } /// /// Helper to handle MouseEnter. /// /// private void OnMouseEnterInternal(EventArgs args) { if (!mouseInEdit) { OnMouseEnter(args); mouseInEdit = true; } } ////// Helper to handle mouseleave /// /// private void OnMouseLeaveInternal(EventArgs args) { NativeMethods.RECT rect = new NativeMethods.RECT(); UnsafeNativeMethods.GetWindowRect(new HandleRef(this, Handle), ref rect); Rectangle Rect = new Rectangle(rect.left, rect.top, rect.right - rect.left, rect.bottom - rect.top); Point p = MousePosition; if (!Rect.Contains(p)) { OnMouseLeave(args); mouseInEdit = false; } } ////// ///private void DefChildWndProc(ref Message m) { if (childEdit != null) { NativeWindow childWindow = (m.HWnd == childEdit.Handle) ? childEdit : childListBox; //childwindow could be null if the handle was recreated while within a message handler // and then whoever recreated the handle allowed the message to continue to be processed //we cannot really be sure the new child will properly handle this window message, so we eat it. if (childWindow != null) { childWindow.DefWndProc(ref m); } } } /// protected override void Dispose(bool disposing) { if (disposing) { if (autoCompleteCustomSource != null) { autoCompleteCustomSource.CollectionChanged -= new CollectionChangeEventHandler(this.OnAutoCompleteCustomSourceChanged); } if (stringSource != null) { stringSource.ReleaseAutoComplete(); stringSource = null; } } base.Dispose(disposing); } /// /// /// Reenables redrawing of the combo box. A call to beginUpdate() must be /// balanced by a following call to endUpdate(). Following a call to /// beginUpdate(), any redrawing caused by operations performed on the /// combo box is deferred until the call to endUpdate(). /// public void EndUpdate() { updateCount--; if (updateCount == 0 && AutoCompleteSource == AutoCompleteSource.ListItems) { SetAutoComplete(false, false); } if (EndUpdateInternal()) { if (childEdit != null && childEdit.Handle != IntPtr.Zero) { SafeNativeMethods.InvalidateRect(new HandleRef(this, childEdit.Handle), null, false); } if (childListBox != null && childListBox.Handle != IntPtr.Zero) { SafeNativeMethods.InvalidateRect(new HandleRef(this, childListBox.Handle), null, false); } } } ////// /// Finds the first item in the combo box that starts with the given string. /// The search is not case sensitive. /// public int FindString(string s) { return FindString(s, -1); } ////// /// Finds the first item after the given index which starts with the given /// string. The search is not case sensitive. /// public int FindString(string s, int startIndex) { if (s == null) { return -1; } if (itemsCollection == null || itemsCollection.Count == 0) { return -1; } // VSWhidbey 95158: The last item in the list is still a valid starting point for a search. if (startIndex < -1 || startIndex >= itemsCollection.Count) { throw new ArgumentOutOfRangeException("startIndex"); } // Always use the managed FindStringInternal instead of CB_FINDSTRING. // The managed version correctly handles Turkish I. // return FindStringInternal(s, Items, startIndex, false); } ////// /// Finds the first item in the combo box that matches the given string. /// The strings must match exactly, except for differences in casing. /// public int FindStringExact(string s) { return FindStringExact(s, -1, true); } ////// /// Finds the first item after the given index that matches the given /// string. The strings must match exactly, except for differences in /// casing. /// public int FindStringExact(string s, int startIndex) { return FindStringExact(s, startIndex, true); } ////// /// Finds the first item after the given index that matches the given /// string. The strings must match exactly, except for differences in /// casing. /// internal int FindStringExact(string s, int startIndex, bool ignorecase) { if (s == null) return -1; if (itemsCollection == null || itemsCollection.Count == 0) { return -1; } // VSWhidbey 95158: The last item in the list is still a valid starting point for a search. if (startIndex < -1 || startIndex >= itemsCollection.Count) { throw new ArgumentOutOfRangeException("startIndex"); } // Always use the managed FindStringInternal instead of CB_FINDSTRINGEXACT. // The managed version correctly handles Turkish I. // return FindStringInternal(s, Items, startIndex, true, ignorecase); } // GetPreferredSize and SetBoundsCore call this method to allow controls to self impose // constraints on their size. internal override Rectangle ApplyBoundsConstraints(int suggestedX, int suggestedY, int proposedWidth, int proposedHeight) { if (DropDownStyle == ComboBoxStyle.DropDown || DropDownStyle == ComboBoxStyle.DropDownList) { proposedHeight = PreferredHeight; } return base.ApplyBoundsConstraints(suggestedX,suggestedY, proposedWidth, proposedHeight); } [SuppressMessage("Microsoft.Portability", "CA1902:AvoidTestingForFloatingPointEquality")] protected override void ScaleControl(SizeF factor, BoundsSpecified specified) { if (factor.Width != 1F && factor.Height != 1F) { // VSWhidbey 440275 // we get called on scale before we get a notification that our font has changed. // in this case, we need to reset our height cache. ResetHeightCache(); } base.ScaleControl(factor, specified); } ////// /// Returns the height of the given item in an OwnerDrawVariable style /// combo box. This method should not be used for Normal or OwnerDrawFixed /// style combo boxes. /// public int GetItemHeight(int index) { // This function is only relevant for OwnerDrawVariable if (DrawMode != DrawMode.OwnerDrawVariable) { return ItemHeight; } if (index < 0 || itemsCollection == null || index >= itemsCollection.Count) { throw new ArgumentOutOfRangeException("index", SR.GetString(SR.InvalidArgument, "index", (index).ToString(CultureInfo.CurrentCulture))); } if (IsHandleCreated) { int h = (int)SendMessage(NativeMethods.CB_GETITEMHEIGHT, index, 0); if (h == -1) { throw new Win32Exception(); } return h; } return ItemHeight; } internal override IntPtr InitializeDCForWmCtlColor(IntPtr dc, int msg) { if ((msg == NativeMethods.WM_CTLCOLORSTATIC) && !ShouldSerializeBackColor()) { // Let the Win32 Edit control handle background colors itself. // This is necessary because a disabled edit control will display a different // BackColor than when enabled. return IntPtr.Zero; } else if ((msg == NativeMethods.WM_CTLCOLORLISTBOX) && GetStyle(ControlStyles.UserPaint)) { // VSWhidbey#93929. Base class returns hollow brush when UserPaint style is set, to avoid flicker in // main control. But when returning colors for child dropdown list, return normal ForeColor/BackColor, // since hollow brush leaves the list background unpainted. SafeNativeMethods.SetTextColor(new HandleRef(null, dc), ColorTranslator.ToWin32(ForeColor)); SafeNativeMethods.SetBkColor(new HandleRef(null, dc), ColorTranslator.ToWin32(BackColor)); return BackColorBrush; } else { return base.InitializeDCForWmCtlColor(dc, msg); } } // Returns true when the key processing needs to be intercepted to allow // auto-completion in DropDownList style. private bool InterceptAutoCompleteKeystroke(Message m) { if (m.Msg == NativeMethods.WM_KEYDOWN) { Debug.Assert((ModifierKeys & Keys.Alt) == 0); // Keys.Delete only triggers a WM_KEYDOWN and WM_KEYUP, and no WM_CHAR. That's why it's treated separately. if ((Keys)(int)m.WParam == Keys.Delete) { // Reset matching text and remove any selection this.MatchingText = ""; this.autoCompleteTimeStamp = DateTime.Now.Ticks; if (this.Items.Count > 0) { SelectedIndex = 0; } return false; } } else if (m.Msg == NativeMethods.WM_CHAR) { Debug.Assert((ModifierKeys & Keys.Alt) == 0); char keyChar = unchecked((char)(long)m.WParam); if (keyChar == (char)Keys.Back) { if (DateTime.Now.Ticks - this.autoCompleteTimeStamp > AutoCompleteTimeout || this.MatchingText.Length <= 1) { // Reset matching text and remove any selection this.MatchingText = ""; if (this.Items.Count > 0) { SelectedIndex = 0; } } else { // Remove one character from matching text and rematch this.MatchingText = this.MatchingText.Remove(this.MatchingText.Length - 1); this.SelectedIndex = FindString(this.MatchingText); } this.autoCompleteTimeStamp = DateTime.Now.Ticks; return false; } else if (keyChar == (char)Keys.Escape) { this.MatchingText = ""; } string newMatchingText; if (keyChar != (char)Keys.Escape && keyChar != (char)Keys.Return && !DroppedDown && AutoCompleteMode != AutoCompleteMode.Append) { DroppedDown = true; } if (DateTime.Now.Ticks - this.autoCompleteTimeStamp > AutoCompleteTimeout) { newMatchingText = new string(keyChar, 1); if (FindString(newMatchingText) != -1) { this.MatchingText = newMatchingText; // Select the found item } this.autoCompleteTimeStamp = DateTime.Now.Ticks; return false; } else { newMatchingText = this.MatchingText + keyChar; int itemFound = FindString(newMatchingText); if (itemFound != -1) { this.MatchingText = newMatchingText; if (itemFound != this.SelectedIndex) { this.SelectedIndex = itemFound; } } // Do not change the selection this.autoCompleteTimeStamp = DateTime.Now.Ticks; return true; } } return false; } // Invalidate the entire control, including child HWNDs and non-client areas private void InvalidateEverything() { SafeNativeMethods.RedrawWindow(new HandleRef(this, Handle), null, NativeMethods.NullHandleRef, NativeMethods.RDW_INVALIDATE | NativeMethods.RDW_FRAME | // Control.Invalidate(true) doesn't invalidate the non-client region NativeMethods.RDW_ERASE | NativeMethods.RDW_ALLCHILDREN); } ////// /// Determines if keyData is in input key that the control wants. /// Overridden to return true for RETURN and ESCAPE when the combo box is /// dropped down. /// ///protected override bool IsInputKey(Keys keyData) { Keys keyCode = keyData & (Keys.KeyCode | Keys.Alt); if (keyCode == Keys.Return || keyCode == Keys.Escape) { if (this.DroppedDown || autoCompleteDroppedDown) { //old behavior return true; } else if (SystemAutoCompleteEnabled && ACNativeWindow.AutoCompleteActive) { autoCompleteDroppedDown = true; return true; } } return base.IsInputKey(keyData); } /// /// Adds the given item to the native combo box. This asserts if the handle hasn't been /// created. /// private int NativeAdd(object item) { Debug.Assert(IsHandleCreated, "Shouldn't be calling Native methods before the handle is created."); int insertIndex = (int)SendMessage(NativeMethods.CB_ADDSTRING, 0, GetItemText(item)); if (insertIndex < 0) { throw new OutOfMemoryException(SR.GetString(SR.ComboBoxItemOverflow)); } return insertIndex; } ////// Clears the contents of the combo box. /// private void NativeClear() { Debug.Assert(IsHandleCreated, "Shouldn't be calling Native methods before the handle is created."); string saved = null; if (DropDownStyle != ComboBoxStyle.DropDownList) { saved = WindowText; } SendMessage(NativeMethods.CB_RESETCONTENT, 0, 0); if (saved != null) { WindowText = saved; } } ////// Get the text stored by the native control for the specified list item. /// private string NativeGetItemText(int index) { int len = (int)SendMessage(NativeMethods.CB_GETLBTEXTLEN, index, 0); StringBuilder sb = new StringBuilder(len + 1); UnsafeNativeMethods.SendMessage(new HandleRef(this, Handle), NativeMethods.CB_GETLBTEXT, index, sb); return sb.ToString(); } ////// Inserts the given item to the native combo box at the index. This asserts if the handle hasn't been /// created or if the resulting insert index doesn't match the passed in index. /// private int NativeInsert(int index, object item) { Debug.Assert(IsHandleCreated, "Shouldn't be calling Native methods before the handle is created."); int insertIndex = (int)SendMessage(NativeMethods.CB_INSERTSTRING, index, GetItemText(item)); if (insertIndex < 0) { throw new OutOfMemoryException(SR.GetString(SR.ComboBoxItemOverflow)); } Debug.Assert(insertIndex == index, "NativeComboBox inserted at " + insertIndex + " not the requested index of " + index); return insertIndex; } ////// Removes the native item from the given index. /// private void NativeRemoveAt(int index) { Debug.Assert(IsHandleCreated, "Shouldn't be calling Native methods before the handle is created."); // Windows combo does not invalidate the selected region if you remove the // currently selected item. Test for this and invalidate. Note that because // invalidate will lazy-paint we can actually invalidate before we send the // delete message. // if (DropDownStyle == ComboBoxStyle.DropDownList && SelectedIndex == index) { Invalidate(); } SendMessage(NativeMethods.CB_DELETESTRING, index, 0); } internal override void RecreateHandleCore() { string oldText = WindowText; base.RecreateHandleCore(); if (!String.IsNullOrEmpty(oldText) && String.IsNullOrEmpty(WindowText)) { WindowText = oldText; //restore the window text } } ////// /// Overridden to avoid multiple layouts during handle creation due to combobox size change /// see bug 458948 /// ///protected override void CreateHandle() { using (new LayoutTransaction(ParentInternal, this, PropertyNames.Bounds)) { base.CreateHandle(); } } /// /// /// Overridden to make sure all the items and styles get set up correctly. /// Inheriting classes should not forget to call /// base.OnHandleCreated() /// ///protected override void OnHandleCreated(EventArgs e) { base.OnHandleCreated(e); if (MaxLength > 0) { SendMessage(NativeMethods.CB_LIMITTEXT, MaxLength, 0); } // Get the handles and wndprocs of the ComboBox's child windows // Debug.Assert(childEdit == null, "Child edit window already attached"); Debug.Assert(childListBox == null, "Child listbox window already attached"); bool ok = childEdit == null && childListBox == null; if (ok && DropDownStyle != ComboBoxStyle.DropDownList) { IntPtr hwnd = UnsafeNativeMethods.GetWindow(new HandleRef(this, Handle), NativeMethods.GW_CHILD); if (hwnd != IntPtr.Zero) { // if it's a simple dropdown list, the first HWND is the list box. // if (DropDownStyle == ComboBoxStyle.Simple) { childListBox = new ComboBoxChildNativeWindow(this); childListBox.AssignHandle(hwnd); // get the edits hwnd... // hwnd = UnsafeNativeMethods.GetWindow(new HandleRef(this, hwnd), NativeMethods.GW_HWNDNEXT); } childEdit = new ComboBoxChildNativeWindow(this); childEdit.AssignHandle(hwnd); // set the initial margin for combobox to be zero (this is also done whenever the font is changed). // UnsafeNativeMethods.SendMessage(new HandleRef(this, childEdit.Handle), NativeMethods.EM_SETMARGINS, NativeMethods.EC_LEFTMARGIN | NativeMethods.EC_RIGHTMARGIN, 0); } } bool found; int dropDownWidth = Properties.GetInteger(PropDropDownWidth, out found); if (found) { SendMessage(NativeMethods.CB_SETDROPPEDWIDTH, dropDownWidth, 0); } found = false; int itemHeight = Properties.GetInteger(PropItemHeight, out found); if (found) { // someone has set the item height - update it UpdateItemHeight(); } // Resize a simple style combobox on handle creation // to respect the requested height. // if (DropDownStyle == ComboBoxStyle.Simple) { Height = requestedHeight; } //If HandleCreated set the AutoComplete... //this function checks if the correct properties are set to enable AutoComplete feature on combobox. try { fromHandleCreate = true; SetAutoComplete(false, false); } finally { fromHandleCreate = false; } if (itemsCollection != null) { foreach (object item in itemsCollection) { NativeAdd(item); } // Now udpate the current selection. // if (selectedIndex >= 0) { SendMessage(NativeMethods.CB_SETCURSEL, selectedIndex, 0); UpdateText(); selectedIndex = -1; } } // NOTE: Setting SelectedIndex must be the last thing we do. See ASURT 73949. } /// /// /// We need to un-subclasses everything here. Inheriting classes should /// not forget to call base.OnHandleDestroyed() /// ///protected override void OnHandleDestroyed(EventArgs e) { dropDownHandle = IntPtr.Zero; if (Disposing) { itemsCollection = null; selectedIndex = -1; } else { selectedIndex = SelectedIndex; } if (stringSource != null) { stringSource.ReleaseAutoComplete(); stringSource = null; } base.OnHandleDestroyed(e); } /// /// /// This is the code that actually fires the drawItem event. Don't /// forget to call base.onDrawItem() to ensure that drawItem events /// are correctly fired at all other times. /// protected virtual void OnDrawItem(DrawItemEventArgs e) { DrawItemEventHandler handler = (DrawItemEventHandler)Events[EVENT_DRAWITEM]; if (handler != null) handler(this, e); } ////// /// This is the code that actually fires the dropDown event. Don't /// forget to call base.onDropDown() to ensure that dropDown events /// are correctly fired at all other times. /// protected virtual void OnDropDown(EventArgs e) { EventHandler handler = (EventHandler)Events[EVENT_DROPDOWN]; if (handler != null) handler(this, e); } ////// /// [EditorBrowsable(EditorBrowsableState.Advanced)] protected override void OnKeyDown(KeyEventArgs e) { // Do Return/ESC handling if (SystemAutoCompleteEnabled) { if (e.KeyCode == Keys.Return) { // Set SelectedIndex NotifyAutoComplete(true); } else if ((e.KeyCode == Keys.Escape) && autoCompleteDroppedDown) { // Fire TextChanged Only NotifyAutoComplete(false); } autoCompleteDroppedDown = false; } // Base Handling base.OnKeyDown(e); } ///Raises the ///event. /// /// Key press event handler. Overridden to close up the combo box when the /// user presses RETURN or ESCAPE. /// ///protected override void OnKeyPress(KeyPressEventArgs e) { base.OnKeyPress(e); //return when dropped down already fires commit. if (!e.Handled && (e.KeyChar == (char)(int)Keys.Return || e.KeyChar == (char)(int)Keys.Escape) && DroppedDown) { dropDown = false; if (FormattingEnabled) { //Set the Text which would Compare the WindowText with the TEXT and change SelectedIndex. Text = WindowText; SelectAll(); e.Handled = false; } else { DroppedDown = false; e.Handled = true; } } } /// /// /// This is the code that actually fires the measuereItem event. Don't /// forget to call base.onMeasureItem() to ensure that measureItem /// events are correctly fired at all other times. /// protected virtual void OnMeasureItem(MeasureItemEventArgs e) { MeasureItemEventHandler handler = (MeasureItemEventHandler)Events[EVENT_MEASUREITEM]; if (handler != null) handler(this, e); } ////// /// If we have the style set to popup show mouse over /// protected override void OnMouseEnter(EventArgs e) { base.OnMouseEnter(e); MouseIsOver = true; } ////// /// If we have the style set to popup show mouse over /// protected override void OnMouseLeave(EventArgs e) { base.OnMouseLeave(e); MouseIsOver = false; } ////// This internal helper allows us to call the committed function multiple times without worrying about double firing. /// private void OnSelectionChangeCommittedInternal(EventArgs e) { //There are cases where if we disable the combo while in this event handler, it sends the message again. //This is a recursion gaurd to ensure we only send one commit per user action. if (allowCommit) { try { allowCommit = false; OnSelectionChangeCommitted(e); } finally { allowCommit = true; } } } ////// /// This is the code that actually fires the SelectionChangeCommitted event. /// Don't forget to call base.OnSelectionChangeCommitted() to ensure /// that SelectionChangeCommitted events are correctly fired at all other times. /// protected virtual void OnSelectionChangeCommitted(EventArgs e) { EventHandler handler = (EventHandler)Events[EVENT_SELECTIONCHANGECOMMITTED]; if (handler != null) handler(this, e); } ////// /// This is the code that actually fires the selectedIndexChanged event. /// Don't forget to call base.onSelectedIndexChanged() to ensure /// that selectedIndexChanged events are correctly fired at all other times. /// protected override void OnSelectedIndexChanged(EventArgs e) { base.OnSelectedIndexChanged(e); EventHandler handler = (EventHandler)Events[EVENT_SELECTEDINDEXCHANGED]; if (handler != null) handler(this, e); // set the position in the dataSource, if there is any // we will only set the position in the currencyManager if it is different // from the SelectedIndex. Setting CurrencyManager::Position (even w/o changing it) // calls CurrencyManager::EndCurrentEdit, and that will pull the dataFrom the controls // into the backEnd. We do not need to do that. // // don't change the position if SelectedIndex is -1 because this indicates a selection not from the list. if (this.DataManager != null && DataManager.Position != SelectedIndex) { //read this as "if everett or (whidbey and selindex is valid)" if (!FormattingEnabled || SelectedIndex != -1) { this.DataManager.Position = this.SelectedIndex; } } } ///protected override void OnSelectedValueChanged(EventArgs e) { base.OnSelectedValueChanged(e); selectedValueChangedFired = true; } /// /// /// This is the code that actually fires the selectedItemChanged event. /// Don't forget to call base.onSelectedItemChanged() to ensure /// that selectedItemChanged events are correctly fired at all other times. /// protected virtual void OnSelectedItemChanged(EventArgs e) { EventHandler handler = (EventHandler)Events[EVENT_SELECTEDITEMCHANGED]; if (handler != null) handler(this, e); } ////// /// This is the code that actually fires the DropDownStyleChanged event. /// protected virtual void OnDropDownStyleChanged(EventArgs e) { EventHandler handler = (EventHandler)Events[EVENT_DROPDOWNSTYLE]; if (handler != null) handler(this, e); } ////// /// This method is called by the parent control when any property /// changes on the parent. This can be overriden by inheriting /// classes, however they must call base.OnParentPropertyChanged. /// protected override void OnParentBackColorChanged(EventArgs e) { base.OnParentBackColorChanged(e); if (DropDownStyle == ComboBoxStyle.Simple) Invalidate(); } ////// /// Indicates that a critical property, such as color or font has /// changed. /// ///protected override void OnFontChanged(EventArgs e) { base.OnFontChanged(e); ResetHeightCache(); if (this.AutoCompleteMode == AutoCompleteMode.None) { UpdateControl(true); } else { //we always will recreate the handle when autocomplete mode is on RecreateHandle(); } CommonProperties.xClearPreferredSizeCache(this); } private void OnAutoCompleteCustomSourceChanged(object sender, CollectionChangeEventArgs e) { if (AutoCompleteSource == AutoCompleteSource.CustomSource) { if (AutoCompleteCustomSource.Count == 0) { SetAutoComplete(true, true /*recreate handle*/); } else { SetAutoComplete(true, false); } } } /// /// /// Indicates that a critical property, such as color or font has /// changed. /// ///protected override void OnBackColorChanged(EventArgs e) { base.OnBackColorChanged(e); UpdateControl(false); } /// /// /// Indicates that a critical property, such as color or font has /// changed. /// ///protected override void OnForeColorChanged(EventArgs e) { base.OnForeColorChanged(e); UpdateControl(false); } /// /// [EditorBrowsable(EditorBrowsableState.Advanced)] protected override void OnGotFocus(EventArgs e) { if (!canFireLostFocus) { base.OnGotFocus(e); canFireLostFocus = true; } } /// /// [EditorBrowsable(EditorBrowsableState.Advanced)] protected override void OnLostFocus(EventArgs e) { if (canFireLostFocus) { if (this.AutoCompleteMode != AutoCompleteMode.None && this.AutoCompleteSource == AutoCompleteSource.ListItems && this.DropDownStyle == ComboBoxStyle.DropDownList) { this.MatchingText = ""; } base.OnLostFocus(e); canFireLostFocus = false; } } /// /// /// [EditorBrowsable(EditorBrowsableState.Advanced)] protected override void OnTextChanged(EventArgs e) { if (SystemAutoCompleteEnabled) { string text = this.Text; // Prevent multiple TextChanges... if (text != this.lastTextChangedValue) { // Need to still fire a TextChanged base.OnTextChanged(e); // Save the new value this.lastTextChangedValue = text; } } else { // Call the base base.OnTextChanged(e); } } ///[To be supplied.] ////// /// [EditorBrowsable(EditorBrowsableState.Advanced)] protected override void OnValidating(CancelEventArgs e) { if (SystemAutoCompleteEnabled) { // Handle AutoComplete notification NotifyAutoComplete(); } // Call base base.OnValidating(e); } private void UpdateControl(bool recreate) { //clear the pref height cache ResetHeightCache(); if (IsHandleCreated) { if (DropDownStyle == ComboBoxStyle.Simple && recreate) { // Control forgets to add a scrollbar. See ASURT 19113. RecreateHandle(); } else { UpdateItemHeight(); // Force everything to repaint. See ASURT 19113. InvalidateEverything(); } } } ///Raises the ////// event. /// /// protected override void OnResize(EventArgs e) { base.OnResize(e); if (DropDownStyle == ComboBoxStyle.Simple) { // simple style combo boxes have more painting problems than you can shake a stick at (#19113) InvalidateEverything(); } } ///Raises the ///event. protected override void OnDataSourceChanged(EventArgs e) { if (Sorted) { if (DataSource != null && Created) { // we will only throw the exception when the control is already on the form. Debug.Assert(DisplayMember.Equals(String.Empty), "if this list is sorted it means that dataSource was null when Sorted first became true. at that point DisplayMember had to be String.Empty"); DataSource = null; throw new InvalidOperationException(SR.GetString(SR.ComboBoxDataSourceWithSort)); } } if (DataSource == null) { BeginUpdate(); SelectedIndex = -1; Items.ClearInternal(); EndUpdate(); } if (!Sorted && Created) base.OnDataSourceChanged(e); RefreshItems(); } /// protected override void OnDisplayMemberChanged(EventArgs e) { base.OnDisplayMemberChanged(e); // bug 63005: when we change the displayMember, we need to refresh the // items collection // RefreshItems(); } /// /// /// This event is fired when the dropdown portion of the combobox is hidden. /// protected virtual void OnDropDownClosed(EventArgs e) { EventHandler handler = (EventHandler)Events[EVENT_DROPDOWNCLOSED]; if (handler != null) handler(this, e); } ////// /// This event is fired when the edit portion of a combobox is about to display altered text. /// This event is NOT fired if the TEXT property is programatically changed. /// protected virtual void OnTextUpdate(EventArgs e) { EventHandler handler = (EventHandler)Events[EVENT_TEXTUPDATE]; if (handler != null) handler(this, e); } ///[ SecurityPermission(SecurityAction.LinkDemand, Flags = SecurityPermissionFlag.UnmanagedCode), ] protected override bool ProcessKeyEventArgs(ref Message m) { if (this.AutoCompleteMode != AutoCompleteMode.None && this.AutoCompleteSource == AutoCompleteSource.ListItems && this.DropDownStyle == ComboBoxStyle.DropDownList && InterceptAutoCompleteKeystroke(m)) { return true; } else { return base.ProcessKeyEventArgs(ref m); } } private void ResetHeightCache() { prefHeightCache = -1; } /// /// /// Reparses the objects, getting new text strings for them. /// ///protected override void RefreshItems() { // Save off the selection and the current collection. // int selectedIndex = SelectedIndex; ObjectCollection savedItems = itemsCollection; itemsCollection = null; object[] newItems = null; // if we have a dataSource and a DisplayMember, then use it // to populate the Items collection // if (this.DataManager != null && this.DataManager.Count != -1) { newItems = new object[this.DataManager.Count]; for (int i = 0; i < newItems.Length; i++) { newItems[i] = this.DataManager[i]; } } else if (savedItems != null) { newItems = new object[savedItems.Count]; savedItems.CopyTo(newItems, 0); } BeginUpdate(); try { // Clear the items. // if (IsHandleCreated) { NativeClear(); } // Store the current list of items // if (newItems != null) { Items.AddRangeInternal(newItems); } if (this.DataManager != null) { // put the selectedIndex in [....] w/ the position in the dataManager this.SelectedIndex = this.DataManager.Position; } else { SelectedIndex = selectedIndex; } } finally { EndUpdate(); } } /// /// /// Reparses the object at the given index, getting new text string for it. /// ///protected override void RefreshItem(int index) { Items.SetItemInternal(index, Items[index]); } /// /// Release the ChildWindow object by un-subclassing the child edit and /// list controls and freeing the root of the ChildWindow object. /// private void ReleaseChildWindow() { if (childEdit != null) { childEdit.ReleaseHandle(); childEdit = null; } if (childListBox != null) { childListBox.ReleaseHandle(); childListBox = null; } } private void ResetAutoCompleteCustomSource() { AutoCompleteCustomSource = null; } private void ResetDropDownWidth() { Properties.RemoveInteger(PropDropDownWidth); } private void ResetItemHeight() { Properties.RemoveInteger(PropItemHeight); } public override void ResetText() { base.ResetText(); } ////// /// Enables the AutoComplete feature for combobox depending on the properties set. /// These properties are namely AutoCompleteMode, AutoCompleteSource and AutoCompleteCustomSource. /// [SuppressMessage("Microsoft.Security", "CA2122:DoNotIndirectlyExposeMethodsWithLinkDemands")] private void SetAutoComplete(bool reset, bool recreate) { if (!IsHandleCreated || childEdit == null) { return; } if (AutoCompleteMode != AutoCompleteMode.None) { if (!fromHandleCreate && recreate && IsHandleCreated) { //RecreateHandle to avoid Leak. // notice the use of member variable to avoid re-entrancy AutoCompleteMode backUpMode = this.AutoCompleteMode; autoCompleteMode = AutoCompleteMode.None; RecreateHandle(); autoCompleteMode = backUpMode; } if (AutoCompleteSource == AutoCompleteSource.CustomSource) { if (AutoCompleteCustomSource != null) { if (AutoCompleteCustomSource.Count == 0) { int mode = NativeMethods.AUTOSUGGEST_OFF | NativeMethods.AUTOAPPEND_OFF; SafeNativeMethods.SHAutoComplete(new HandleRef(this, childEdit.Handle), mode); } else { if (stringSource == null) { stringSource = new StringSource(GetStringsForAutoComplete(AutoCompleteCustomSource)); if (!stringSource.Bind(new HandleRef(this, childEdit.Handle), (int)AutoCompleteMode)) { throw new ArgumentException(SR.GetString(SR.AutoCompleteFailure)); } } else { stringSource.RefreshList(GetStringsForAutoComplete(AutoCompleteCustomSource)); } } } } else if (AutoCompleteSource == AutoCompleteSource.ListItems) { if (DropDownStyle != ComboBoxStyle.DropDownList) { if (itemsCollection != null) { if (itemsCollection.Count == 0) { int mode = NativeMethods.AUTOSUGGEST_OFF | NativeMethods.AUTOAPPEND_OFF; SafeNativeMethods.SHAutoComplete(new HandleRef(this, childEdit.Handle), mode); } else { if (stringSource == null) { stringSource = new StringSource(GetStringsForAutoComplete(Items)); if (!stringSource.Bind(new HandleRef(this, childEdit.Handle), (int)AutoCompleteMode)) { throw new ArgumentException(SR.GetString(SR.AutoCompleteFailureListItems)); } } else { stringSource.RefreshList(GetStringsForAutoComplete(Items)); } } } } else { // Drop Down List special handling Debug.Assert(DropDownStyle == ComboBoxStyle.DropDownList); int mode = NativeMethods.AUTOSUGGEST_OFF | NativeMethods.AUTOAPPEND_OFF; SafeNativeMethods.SHAutoComplete(new HandleRef(this, childEdit.Handle), mode); } } else { try { int mode = 0; if (AutoCompleteMode == AutoCompleteMode.Suggest) { mode |= NativeMethods.AUTOSUGGEST | NativeMethods.AUTOAPPEND_OFF; } if (AutoCompleteMode == AutoCompleteMode.Append) { mode |= NativeMethods.AUTOAPPEND | NativeMethods.AUTOSUGGEST_OFF; } if (AutoCompleteMode == AutoCompleteMode.SuggestAppend) { mode |= NativeMethods.AUTOSUGGEST; mode |= NativeMethods.AUTOAPPEND; } int ret = SafeNativeMethods.SHAutoComplete(new HandleRef(this, childEdit.Handle), (int)AutoCompleteSource | mode); } catch (SecurityException) { // If we don't have full trust, degrade gracefully. Allow the control to // function without auto-complete. Allow the app to continue running. } } } else if (reset) { int mode = NativeMethods.AUTOSUGGEST_OFF | NativeMethods.AUTOAPPEND_OFF; SafeNativeMethods.SHAutoComplete(new HandleRef(this, childEdit.Handle), mode); } } ////// /// Selects the text in the editable portion of the ComboBox at the /// from the given start index to the given end index. /// public void Select(int start, int length) { if (start < 0) { throw new ArgumentOutOfRangeException("start", SR.GetString(SR.InvalidArgument, "start", start.ToString(CultureInfo.CurrentCulture))); } // the Length can be negative to support Selecting in the "reverse" direction.. int end = start + length; // but end cannot be negative... this means Length is far negative... if (end < 0) { throw new ArgumentOutOfRangeException("length", SR.GetString(SR.InvalidArgument, "length", length.ToString(CultureInfo.CurrentCulture))); } SendMessage(NativeMethods.CB_SETEDITSEL, 0, NativeMethods.Util.MAKELPARAM(start, end)); } ////// /// Selects all the text in the editable portion of the ComboBox. /// public void SelectAll() { Select(0, Int32.MaxValue); } ///protected override void SetBoundsCore(int x, int y, int width, int height, BoundsSpecified specified) { // If we are changing height, store the requested height. // Requested height is used if the style is changed to simple. // (Bug fix #20966) if ((specified & BoundsSpecified.Height) != BoundsSpecified.None) { requestedHeight = height; } base.SetBoundsCore(x, y, width, height, specified); } /// /// /// Performs the work of setting the specified items to the combobox /// protected override void SetItemsCore(IList value) { BeginUpdate(); Items.ClearInternal(); Items.AddRangeInternal(value); // if the list changed, we want to keep the same selected index // CurrencyManager will provide the PositionChanged event // it will be provided before changing the list though... if (this.DataManager != null) { if (this.DataSource is ICurrencyManagerProvider) { // Everett ListControl's had a bug where they would not fire // OnSelectedValueChanged if their list of items were refreshed. // We fix this post-Everett. // However, for APPCOMPAT reasons, we only want to fix it when binding to // Whidbey components. // vsw 547279. this.selectedValueChangedFired = false; } if (IsHandleCreated) { SendMessage(NativeMethods.CB_SETCURSEL, DataManager.Position, 0); } else { selectedIndex = DataManager.Position; } // if set_SelectedIndexChanged did not fire OnSelectedValueChanged // then we have to fire it ourselves, cos the list changed anyway if (!selectedValueChangedFired) { OnSelectedValueChanged(EventArgs.Empty); selectedValueChangedFired = false; } } EndUpdate(); } ///protected override void SetItemCore(int index, object value) { Items.SetItemInternal(index, value); } private bool ShouldSerializeAutoCompleteCustomSource() { return autoCompleteCustomSource != null && autoCompleteCustomSource.Count > 0; } internal bool ShouldSerializeDropDownWidth() { return (Properties.ContainsInteger(PropDropDownWidth)); } /// /// Indicates whether the itemHeight property should be persisted. /// internal bool ShouldSerializeItemHeight() { return (Properties.ContainsInteger(PropItemHeight)); } ////// /// Determines if the Text property needs to be persisted. /// internal override bool ShouldSerializeText() { return SelectedIndex == -1 && base.ShouldSerializeText(); } ////// /// Provides some interesting info about this control in String form. /// ///public override string ToString() { string s = base.ToString(); return s + ", Items.Count: " + ((itemsCollection == null) ? (0).ToString(CultureInfo.CurrentCulture) : itemsCollection.Count.ToString(CultureInfo.CurrentCulture)); } /// /// ///private void UpdateDropDownHeight() { if (dropDownHandle != IntPtr.Zero) { //Now use the DropDownHeight property instead of calculating the Height... int height = DropDownHeight; if (height == DefaultDropDownHeight) { int itemCount = (itemsCollection == null) ? 0 : itemsCollection.Count; int count = Math.Min(Math.Max(itemCount, 1), maxDropDownItems); height = (ItemHeight * count + 2); } SafeNativeMethods.SetWindowPos(new HandleRef(this, dropDownHandle), NativeMethods.NullHandleRef, 0, 0, DropDownWidth, height, NativeMethods.SWP_NOMOVE | NativeMethods.SWP_NOZORDER); } } /// /// Manufactures a MeasureItemEventArgs for each item in the list to simulate /// the combobox requesting the info. This gives the effect of allowing the /// measureitem info to be updated at anytime. /// ///private void UpdateItemHeight() { if (!IsHandleCreated) { // VSWhidbey 156992: if we don't create control here we report item heights incorrectly later on. CreateControl(); } if (DrawMode == DrawMode.OwnerDrawFixed) { SendMessage(NativeMethods.CB_SETITEMHEIGHT, -1, ItemHeight); SendMessage(NativeMethods.CB_SETITEMHEIGHT, 0, ItemHeight); } else if (DrawMode == DrawMode.OwnerDrawVariable) { SendMessage(NativeMethods.CB_SETITEMHEIGHT, -1, ItemHeight); Graphics graphics = CreateGraphicsInternal(); for (int i = 0; i < Items.Count; i++) { int original = (int)SendMessage(NativeMethods.CB_GETITEMHEIGHT, i, 0); MeasureItemEventArgs mievent = new MeasureItemEventArgs(graphics, i, original); OnMeasureItem(mievent); if (mievent.ItemHeight != original) { SendMessage(NativeMethods.CB_SETITEMHEIGHT, i, mievent.ItemHeight); } } graphics.Dispose(); } } /// /// Forces the text to be updated based on the current selection. /// ///private void UpdateText() { // V#45724 - Fire text changed for dropdown combos when the selection // changes, since the text really does change. We've got // to do this asynchronously because the actual edit text // isn't updated until a bit later (V#51240). // // QFE 2471: // v1.0 - ComboBox::set_Text compared items w/ "value" and set the SelectedIndex accordingly // v1.0 - null values can't correspond to String.Empty // v1.0 - SelectedIndex == -1 corresponds to Text == String.Emtpy // // v1.1 - ComboBox::set_Text compares FilterItemOnProperty(item) w/ "value" and set the SelectedIndex accordingly // v1.1 - null values correspond to String.Empty // v1.1 - SelectedIndex == -1 corresponds to Text == null string s = null; if (SelectedIndex != -1) { object item = Items[SelectedIndex]; if (item != null) { s = GetItemText(item); } } Text = s; if (DropDownStyle == ComboBoxStyle.DropDown) { if (childEdit != null && childEdit.Handle != IntPtr.Zero) { UnsafeNativeMethods.SendMessage(new HandleRef(this, childEdit.Handle), NativeMethods.WM_SETTEXT, IntPtr.Zero, s); } } } /// /// ///private void WmEraseBkgnd(ref Message m) { if ((DropDownStyle == ComboBoxStyle.Simple) && ParentInternal != null) { NativeMethods.RECT rect = new NativeMethods.RECT(); SafeNativeMethods.GetClientRect(new HandleRef(this, Handle), ref rect); Control p = ParentInternal; Graphics graphics = Graphics.FromHdcInternal(m.WParam); if (p != null) { Brush brush = new SolidBrush(p.BackColor); graphics.FillRectangle(brush, rect.left, rect.top, rect.right - rect.left, rect.bottom - rect.top); brush.Dispose(); } else { graphics.FillRectangle(SystemBrushes.Control, rect.left, rect.top, rect.right - rect.left, rect.bottom - rect.top); } graphics.Dispose(); m.Result = (IntPtr)1; return; } base.WndProc(ref m); } /// /// ///private void WmParentNotify(ref Message m) { base.WndProc(ref m); if (unchecked((int)(long)m.WParam) == (NativeMethods.WM_CREATE | 1000 << 16)) { dropDownHandle = m.LParam; } } /// /// Text change behavior. /// Here are the window messages corresponding to each user event. /// /// DropDown (free text window): /// Type in Text Window: /// CBN_EDITUPDATE /// CBN_EDITCHANGE /// Down/Up Arrow /// CBN_SELENDOK (text not changed yet) /// CBN_SELCHANGE /// with text set to beginning of a valid item -- drop and close dropdown (selects that item) /// CBN_DROPDOWN /// CBN_CLOSEUP /// Drop List, up/down arrow to select item, click away /// CBN_DROPDOWN /// CBN_SELCHANGE /// CBN_CLOSEUP /// Drop List, click on item /// CBN_DROPDOWN /// CBN_SELENDOK /// CBN_CLOSEUP /// CBN_SELCHANGE (text changes here via selected item) /// /// DropDownList (limited text window): /// /// Type text and arrow up/down: /// CBN_SELENDOK (text already changed) /// CBN_SELCHANGE /// Drop List, up/down arrow to select item, click away /// CBN_DROPDOWN /// CBN_SELCHANGE /// CBN_CLOSEUP /// Drop List, click on item /// CBN_DROPDOWN /// CBN_SELENDOK /// CBN_CLOSEUP /// CBN_SELCHANGE /// /// Simple (listbox visible): /// Type in Text Window: /// CBN_EDITUPDATE /// CBN_EDITCHANGE /// Down/Up Arrow /// CBN_SELENDOK (text not changed yet) /// CBN_SELCHANGE /// Click on item /// CBN_SELENDOK (text not changed yet) /// CBN_SELCHANGE /// /// /// What we do is fire textchange events in these messages: /// CBN_SELCHANGE /// CBN_EDITCHANGE /// CBN_CLOSEUP /// /// and we only actually call the real event if the Text is different than currentText. /// currentText is never changed outside this method. /// This internal version can be called from anywhere we might suspect text has changed /// it's fairly safe to call anywhere. /// /// ///private void WmReflectCommand(ref Message m) { switch (NativeMethods.Util.HIWORD(m.WParam)) { case NativeMethods.CBN_DBLCLK: //OnDoubleClick(EventArgs.Empty); break; case NativeMethods.CBN_EDITUPDATE: OnTextUpdate(EventArgs.Empty); break; case NativeMethods.CBN_CLOSEUP: OnDropDownClosed(EventArgs.Empty); if (FormattingEnabled && Text != currentText && dropDown) { OnTextChanged(EventArgs.Empty); } dropDown = false; break; case NativeMethods.CBN_DROPDOWN: currentText = Text; dropDown = true; OnDropDown(EventArgs.Empty); UpdateDropDownHeight(); break; case NativeMethods.CBN_EDITCHANGE: OnTextChanged(EventArgs.Empty); break; case NativeMethods.CBN_SELCHANGE: UpdateText(); OnSelectedIndexChanged(EventArgs.Empty); break; case NativeMethods.CBN_SELENDOK: OnSelectionChangeCommittedInternal(EventArgs.Empty); break; } } /// /// ///private void WmReflectDrawItem(ref Message m) { NativeMethods.DRAWITEMSTRUCT dis = (NativeMethods.DRAWITEMSTRUCT)m.GetLParam(typeof(NativeMethods.DRAWITEMSTRUCT)); IntPtr oldPal = SetUpPalette(dis.hDC, false /*force*/, false /*realize*/); try { Graphics g = Graphics.FromHdcInternal(dis.hDC); try { OnDrawItem(new DrawItemEventArgs(g, Font, Rectangle.FromLTRB(dis.rcItem.left, dis.rcItem.top, dis.rcItem.right, dis.rcItem.bottom), dis.itemID, (DrawItemState)dis.itemState, ForeColor, BackColor)); } finally { g.Dispose(); } } finally { if (oldPal != IntPtr.Zero) { SafeNativeMethods.SelectPalette(new HandleRef(this, dis.hDC), new HandleRef(null, oldPal), 0); } } m.Result = (IntPtr)1; } /// /// ///private void WmReflectMeasureItem(ref Message m) { NativeMethods.MEASUREITEMSTRUCT mis = (NativeMethods.MEASUREITEMSTRUCT)m.GetLParam(typeof(NativeMethods.MEASUREITEMSTRUCT)); // Determine if message was sent by a combo item or the combo edit field if (DrawMode == DrawMode.OwnerDrawVariable && mis.itemID >= 0) { Graphics graphics = CreateGraphicsInternal(); MeasureItemEventArgs mie = new MeasureItemEventArgs(graphics, mis.itemID, ItemHeight); OnMeasureItem(mie); mis.itemHeight = mie.ItemHeight; graphics.Dispose(); } else { // Message was sent by the combo edit field mis.itemHeight = ItemHeight; } Marshal.StructureToPtr(mis, m.LParam, false); m.Result = (IntPtr)1; } /// /// /// The comboboxs window procedure. Inheritng classes can override this /// to add extra functionality, but should not forget to call /// base.wndProc(m); to ensure the combo continues to function properly. /// ///[SecurityPermission(SecurityAction.LinkDemand, Flags = SecurityPermissionFlag.UnmanagedCode)] protected override void WndProc(ref Message m) { switch (m.Msg) { // We don't want to fire the focus events twice - // once in the combobox and once in the ChildWndProc. case NativeMethods.WM_SETFOCUS: try { fireSetFocus = false; base.WndProc(ref m); } finally { fireSetFocus = true; } break; case NativeMethods.WM_KILLFOCUS: try { fireLostFocus = false; base.WndProc(ref m); // Nothing to see here... Just keep on walking... VSWhidbey 504477. // Turns out that with Theming off, we don't get quite the same messages as with theming on. // With theming on we get a WM_MOUSELEAVE after a WM_KILLFOCUS even if you use the Tab key // to move focus. Our response to WM_MOUSELEAVE causes us to repaint everything correctly. // With theming off, we do not get a WM_MOUSELEAVE after a WM_KILLFOCUS, and since we don't have a childwndproc // when we are a Flat DropDownList, we need to force a repaint. The easiest way to do this is to send a // WM_MOUSELEAVE to ourselves, since that also sets up the right state. Or... at least the state is the same // as with Theming on. // This is such a @#$(*@$ hack. if (!Application.RenderWithVisualStyles && GetStyle(ControlStyles.UserPaint) == false && this.DropDownStyle == ComboBoxStyle.DropDownList && (FlatStyle == FlatStyle.Flat || FlatStyle == FlatStyle.Popup)) { UnsafeNativeMethods.PostMessage(new HandleRef(this, Handle), NativeMethods.WM_MOUSELEAVE, 0, 0); } } finally { fireLostFocus = true; } break; case NativeMethods.WM_CTLCOLOREDIT: case NativeMethods.WM_CTLCOLORLISTBOX: m.Result = InitializeDCForWmCtlColor(m.WParam, m.Msg); break; case NativeMethods.WM_ERASEBKGND: WmEraseBkgnd(ref m); break; case NativeMethods.WM_PARENTNOTIFY: WmParentNotify(ref m); break; case NativeMethods.WM_REFLECT + NativeMethods.WM_COMMAND: WmReflectCommand(ref m); break; case NativeMethods.WM_REFLECT + NativeMethods.WM_DRAWITEM: WmReflectDrawItem(ref m); break; case NativeMethods.WM_REFLECT + NativeMethods.WM_MEASUREITEM: WmReflectMeasureItem(ref m); break; case NativeMethods.WM_LBUTTONDOWN: mouseEvents = true; base.WndProc(ref m); break; case NativeMethods.WM_LBUTTONUP: // Get the mouse location // NativeMethods.RECT r = new NativeMethods.RECT(); UnsafeNativeMethods.GetWindowRect(new HandleRef(this, Handle), ref r); Rectangle ClientRect = new Rectangle(r.left, r.top, r.right - r.left, r.bottom - r.top); int x = NativeMethods.Util.SignedLOWORD(m.LParam); int y = NativeMethods.Util.SignedHIWORD(m.LParam); Point pt = new Point(x, y); pt = PointToScreen(pt); //mouseEvents is used to keep the check that we get the WM_LBUTTONUP after //WM_LBUTTONDOWN or WM_LBUTTONDBLBCLK // combo box gets a WM_LBUTTONUP for focus change ... // if (mouseEvents && !ValidationCancelled) { mouseEvents = false; bool captured = Capture; if (captured && ClientRect.Contains(pt)) { OnClick(new MouseEventArgs(MouseButtons.Left, 1, NativeMethods.Util.SignedLOWORD(m.LParam), NativeMethods.Util.SignedHIWORD(m.LParam), 0)); OnMouseClick(new MouseEventArgs(MouseButtons.Left, 1, NativeMethods.Util.SignedLOWORD(m.LParam), NativeMethods.Util.SignedHIWORD(m.LParam), 0)); } base.WndProc(ref m); } else { CaptureInternal = false; DefWndProc(ref m); } break; case NativeMethods.WM_MOUSELEAVE: DefWndProc(ref m); OnMouseLeaveInternal(EventArgs.Empty); break; case NativeMethods.WM_PAINT: if (GetStyle(ControlStyles.UserPaint) == false && (FlatStyle == FlatStyle.Flat || FlatStyle == FlatStyle.Popup)) { using (WindowsRegion dr = new WindowsRegion(FlatComboBoxAdapter.dropDownRect)) { using (WindowsRegion wr = new WindowsRegion(this.Bounds)) { // Stash off the region we have to update (the base is going to clear this off in BeginPaint) NativeMethods.RegionFlags updateRegionFlags = (NativeMethods.RegionFlags)SafeNativeMethods.GetUpdateRgn(new HandleRef(this, this.Handle), new HandleRef(this, wr.HRegion), true); dr.CombineRegion(wr, dr, RegionCombineMode.DIFF); Rectangle updateRegionBoundingRect = wr.ToRectangle(); FlatComboBoxAdapter.ValidateOwnerDrawRegions(this, updateRegionBoundingRect); // Call the base class to do its painting (with a clipped DC). NativeMethods.PAINTSTRUCT ps = new NativeMethods.PAINTSTRUCT(); IntPtr dc; bool disposeDc = false; if (m.WParam == IntPtr.Zero) { dc = UnsafeNativeMethods.BeginPaint(new HandleRef(this, Handle), ref ps); disposeDc = true; } else { dc = m.WParam; } using (DeviceContext mDC = DeviceContext.FromHdc(dc)) { using (WindowsGraphics wg = new WindowsGraphics(mDC)) { if (updateRegionFlags != NativeMethods.RegionFlags.ERROR) { wg.DeviceContext.SetClip(dr); } m.WParam = dc; DefWndProc(ref m); if (updateRegionFlags != NativeMethods.RegionFlags.ERROR) { wg.DeviceContext.SetClip(wr); } using (Graphics g = Graphics.FromHdcInternal(dc)) { FlatComboBoxAdapter.DrawFlatCombo(this, g); } } } if (disposeDc) { UnsafeNativeMethods.EndPaint(new HandleRef(this, Handle), ref ps); } } return; } } base.WndProc(ref m); break; case NativeMethods.WM_PRINTCLIENT: // all the fancy stuff we do in OnPaint has to happen again in OnPrint. if (GetStyle(ControlStyles.UserPaint) == false && FlatStyle == FlatStyle.Flat || FlatStyle == FlatStyle.Popup) { DefWndProc(ref m); if (((int)m.LParam & NativeMethods.PRF_CLIENT) == NativeMethods.PRF_CLIENT) { if (GetStyle(ControlStyles.UserPaint) == false && FlatStyle == FlatStyle.Flat || FlatStyle == FlatStyle.Popup) { using (Graphics g = Graphics.FromHdcInternal(m.WParam)) { FlatComboBoxAdapter.DrawFlatCombo(this,g); } } return; } } base.WndProc(ref m); return; case NativeMethods.WM_SETCURSOR: base.WndProc(ref m); break; case NativeMethods.WM_SETFONT: //(bug 119265) if (Width == 0) { suppressNextWindosPos = true; } base.WndProc(ref m); break; case NativeMethods.WM_WINDOWPOSCHANGED: if (!suppressNextWindosPos) { base.WndProc(ref m); } suppressNextWindosPos = false; break; case NativeMethods.WM_NCDESTROY: base.WndProc(ref m); ReleaseChildWindow(); break; default: if (m.Msg == NativeMethods.WM_MOUSEENTER) { DefWndProc(ref m); OnMouseEnterInternal(EventArgs.Empty); break; } base.WndProc(ref m); break; } } /// /// ///private class ComboBoxChildNativeWindow : NativeWindow { private ComboBox _owner; private InternalAccessibleObject _accessibilityObject; internal ComboBoxChildNativeWindow(ComboBox comboBox) { _owner = comboBox; } protected override void WndProc(ref Message m) { switch (m.Msg) { case NativeMethods.WM_GETOBJECT: WmGetObject(ref m); return; default: _owner.ChildWndProc(ref m); break; } } private void WmGetObject(ref Message m) { // See "How to Handle WM_GETOBJECT" in MSDN // if (NativeMethods.OBJID_CLIENT == unchecked((int)(long)m.LParam)) { // Get the IAccessible GUID // Guid IID_IAccessible = new Guid(NativeMethods.uuid_IAccessible); // Get an Lresult for the accessibility Object for this control // IntPtr punkAcc; try { AccessibleObject wfAccessibleObject = null; UnsafeNativeMethods.IAccessibleInternal iacc = null; if (_accessibilityObject == null) { IntSecurity.UnmanagedCode.Assert(); try { wfAccessibleObject = new ChildAccessibleObject(_owner, Handle); _accessibilityObject = new InternalAccessibleObject(wfAccessibleObject); } finally { CodeAccessPermission.RevertAssert(); } } iacc = (UnsafeNativeMethods.IAccessibleInternal)_accessibilityObject; // Obtain the Lresult // punkAcc = Marshal.GetIUnknownForObject(iacc); IntSecurity.UnmanagedCode.Assert(); try { m.Result = UnsafeNativeMethods.LresultFromObject(ref IID_IAccessible, m.WParam, new HandleRef(this, punkAcc)); } finally { CodeAccessPermission.RevertAssert(); Marshal.Release(punkAcc); } } catch (Exception e) { throw new InvalidOperationException(SR.GetString(SR.RichControlLresult), e); } } else { // m.lparam != OBJID_CLIENT, so do default message processing DefWndProc(ref m); } } } private sealed class ItemComparer : System.Collections.IComparer { private ComboBox comboBox; public ItemComparer(ComboBox comboBox) { this.comboBox = comboBox; } public int Compare(object item1, object item2) { if (item1 == null) { if (item2 == null) return 0; //both null, then they are equal return -1; //item1 is null, but item2 is valid (greater) } if (item2 == null) return 1; //item2 is null, so item 1 is greater String itemName1 = comboBox.GetItemText(item1); String itemName2 = comboBox.GetItemText(item2); CompareInfo compInfo = (Application.CurrentCulture).CompareInfo; return compInfo.Compare(itemName1, itemName2, CompareOptions.StringSort); } } /// /// /// [ListBindable(false)] public class ObjectCollection : IList { private ComboBox owner; private ArrayList innerList; private IComparer comparer; ///[To be supplied.] ////// /// public ObjectCollection(ComboBox owner) { this.owner = owner; } private IComparer Comparer { get { if (comparer == null) { comparer = new ItemComparer(owner); } return comparer; } } private ArrayList InnerList { get { if (innerList == null) { innerList = new ArrayList(); } return innerList; } } ///[To be supplied.] ////// /// Retrieves the number of items. /// public int Count { get { return InnerList.Count; } } ////// object ICollection.SyncRoot { get { return this; } } /// /// bool ICollection.IsSynchronized { get { return false; } } /// /// bool IList.IsFixedSize { get { return false; } } /// /// /// public bool IsReadOnly { get { return false; } } ///[To be supplied.] ////// /// Adds an item to the combo box. For an unsorted combo box, the item is /// added to the end of the existing list of items. For a sorted combo box, /// the item is inserted into the list according to its sorted position. /// The item's toString() method is called to obtain the string that is /// displayed in the combo box. /// A SystemException occurs if there is insufficient space available to /// store the new item. /// public int Add(object item) { owner.CheckNoDataSource(); int index = AddInternal(item); if (owner.UpdateNeeded() && owner.AutoCompleteSource == AutoCompleteSource.ListItems) { owner.SetAutoComplete(false, false); } return index; } private int AddInternal(object item) { if (item == null) { throw new ArgumentNullException("item"); } int index = -1; if (!owner.sorted) { InnerList.Add(item); } else { index = InnerList.BinarySearch(item, Comparer); if (index < 0) { index = ~index; // getting the index of the first element that is larger than the search value } Debug.Assert(index>=0 && index <= InnerList.Count, "Wrong index for insert"); InnerList.Insert(index, item); } bool successful = false; try { if (owner.sorted) { if (owner.IsHandleCreated) { owner.NativeInsert(index, item); } } else { index = InnerList.Count-1; if (owner.IsHandleCreated) { owner.NativeAdd(item); } } successful = true; } finally { if (!successful) { InnerList.Remove(item); } } return index; } ////// int IList.Add(object item) { return Add(item); } /// /// /// public void AddRange(object[] items) { owner.CheckNoDataSource(); owner.BeginUpdate(); try { AddRangeInternal(items); } finally { owner.EndUpdate(); } } internal void AddRangeInternal(IList items) { if (items == null) { throw new ArgumentNullException("items"); } foreach (object item in items) { // adding items one-by-one for performance (especially for sorted combobox) // we can not rely on ArrayList.Sort since its worst case complexity is n*n // AddInternal is based on BinarySearch and ensures n*log(n) complexity AddInternal(item); } if (owner.AutoCompleteSource == AutoCompleteSource.ListItems) { owner.SetAutoComplete(false, false); } } ///[To be supplied.] ////// /// Retrieves the item with the specified index. /// [Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)] public virtual object this[int index] { get { if (index < 0 || index >= InnerList.Count) { throw new ArgumentOutOfRangeException("index", SR.GetString(SR.InvalidArgument, "index", (index).ToString(CultureInfo.CurrentCulture))); } return InnerList[index]; } set { owner.CheckNoDataSource(); SetItemInternal(index, value); } } ////// /// Removes all items from the ComboBox. /// public void Clear() { owner.CheckNoDataSource(); ClearInternal(); } internal void ClearInternal() { if (owner.IsHandleCreated) { owner.NativeClear(); } InnerList.Clear(); owner.selectedIndex = -1; if (owner.AutoCompleteSource == AutoCompleteSource.ListItems) { owner.SetAutoComplete(false, true /*recreateHandle*/); } } ////// /// public bool Contains(object value) { return IndexOf(value) != -1; } ///[To be supplied.] ////// /// Copies the ComboBox Items collection to a destination array. /// public void CopyTo(object[] destination, int arrayIndex) { InnerList.CopyTo(destination, arrayIndex); } ////// void ICollection.CopyTo(Array destination, int index) { InnerList.CopyTo(destination, index); } /// /// /// Returns an enumerator for the ComboBox Items collection. /// public IEnumerator GetEnumerator() { return InnerList.GetEnumerator(); } ////// /// public int IndexOf(object value) { if (value == null) { throw new ArgumentNullException("value"); } return InnerList.IndexOf(value); } ///[To be supplied.] ////// /// Adds an item to the combo box. For an unsorted combo box, the item is /// added to the end of the existing list of items. For a sorted combo box, /// the item is inserted into the list according to its sorted position. /// The item's toString() method is called to obtain the string that is /// displayed in the combo box. /// A SystemException occurs if there is insufficient space available to /// store the new item. /// public void Insert(int index, object item) { owner.CheckNoDataSource(); if (item == null) { throw new ArgumentNullException("item"); } if (index < 0 || index > InnerList.Count) { throw new ArgumentOutOfRangeException("index", SR.GetString(SR.InvalidArgument, "index", (index).ToString(CultureInfo.CurrentCulture))); } // If the combo box is sorted, then nust treat this like an add // because we are going to twiddle the index anyway. // if (owner.sorted) { Add(item); } else { InnerList.Insert(index, item); if (owner.IsHandleCreated) { bool successful = false; try { owner.NativeInsert(index, item); successful = true; } finally { if (successful) { if (owner.AutoCompleteSource == AutoCompleteSource.ListItems) { owner.SetAutoComplete(false, false); } } else { InnerList.RemoveAt(index); } } } } } ////// /// Removes an item from the ComboBox at the given index. /// public void RemoveAt(int index) { owner.CheckNoDataSource(); if (index < 0 || index >= InnerList.Count) { throw new ArgumentOutOfRangeException("index", SR.GetString(SR.InvalidArgument, "index", (index).ToString(CultureInfo.CurrentCulture))); } if (owner.IsHandleCreated) { owner.NativeRemoveAt(index); } InnerList.RemoveAt(index); if (!owner.IsHandleCreated && index < owner.selectedIndex) { owner.selectedIndex--; } if (owner.AutoCompleteSource == AutoCompleteSource.ListItems) { owner.SetAutoComplete(false, false); } } ////// /// Removes the given item from the ComboBox, provided that it is /// actually in the list. /// public void Remove(object value) { int index = InnerList.IndexOf(value); if (index != -1) { RemoveAt(index); } } internal void SetItemInternal(int index, object value) { if (value == null) { throw new ArgumentNullException("value"); } if (index < 0 || index >= InnerList.Count) { throw new ArgumentOutOfRangeException("index", SR.GetString(SR.InvalidArgument, "index", (index).ToString(CultureInfo.CurrentCulture))); } InnerList[index] = value; // If the native control has been created, and the display text of the new list item object // is different to the current text in the native list item, recreate the native list item... if (owner.IsHandleCreated) { bool selected = (index == owner.SelectedIndex); if (String.Compare(this.owner.GetItemText(value), this.owner.NativeGetItemText(index), true, CultureInfo.CurrentCulture) != 0) { owner.NativeRemoveAt(index); owner.NativeInsert(index, value); if (selected) { owner.SelectedIndex = index; owner.UpdateText(); } if (owner.AutoCompleteSource == AutoCompleteSource.ListItems) { owner.SetAutoComplete(false, false); } } else { // NEW - FOR COMPATIBILITY REASONS // Minimum compatibility fix for VSWhidbey 377287/444903 if (selected) { owner.OnSelectedItemChanged(EventArgs.Empty); //we do this because set_SelectedIndex does this. (for consistency) owner.OnSelectedIndexChanged(EventArgs.Empty); } } } } } // end ObjectCollection ////// [ComVisible(true)] public class ChildAccessibleObject : AccessibleObject { ComboBox owner; /// [SecurityPermission(SecurityAction.Demand, Flags = SecurityPermissionFlag.UnmanagedCode)] public ChildAccessibleObject(ComboBox owner, IntPtr handle) { Debug.Assert(owner != null && owner.Handle != IntPtr.Zero, "ComboBox's handle hasn't been created"); this.owner = owner; UseStdAccessibleObjects(handle); } /// public override string Name { get { return owner.AccessibilityObject.Name; } } } /// /// [System.Runtime.InteropServices.ComVisible(true)] internal class ComboBoxAccessibleObject : ControlAccessibleObject { private const int COMBOBOX_ACC_ITEM_INDEX = 1; public ComboBoxAccessibleObject(Control ownerControl) : base(ownerControl) { } internal override string get_accNameInternal(object childID) { this.ValidateChildID(ref childID); if (childID != null && ((int)childID) == COMBOBOX_ACC_ITEM_INDEX) { return this.Name; } else { return base.get_accNameInternal(childID); } } internal override string get_accKeyboardShortcutInternal(object childID) { this.ValidateChildID(ref childID); if (childID != null && ((int)childID) == COMBOBOX_ACC_ITEM_INDEX) { return this.KeyboardShortcut; } else { return base.get_accKeyboardShortcutInternal(childID); } } } ////// This subclasses an autocomplete window so that we can determine if control is inside the AC wndproc. /// private sealed class ACNativeWindow : NativeWindow { static internal int inWndProcCnt; //this hashtable can contain null for those ACWindows we find, but are sure are not ours. static private Hashtable ACWindows = new Hashtable(); [ SuppressMessage("Microsoft.Usage", "CA2214:DoNotCallOverridableMethodsInConstructors") ] internal ACNativeWindow(IntPtr acHandle) { Debug.Assert(!ACWindows.ContainsKey(acHandle)); this.AssignHandle(acHandle); ACWindows.Add(acHandle, this); UnsafeNativeMethods.EnumChildWindows(new HandleRef(this, acHandle), new NativeMethods.EnumChildrenCallback(ACNativeWindow.RegisterACWindowRecursive), NativeMethods.NullHandleRef); } [SuppressMessage("Microsoft.Performance", "CA1806:DoNotIgnoreMethodResults")] private static bool RegisterACWindowRecursive(IntPtr handle, IntPtr lparam) { if (!ACWindows.ContainsKey(handle)) { ACNativeWindow newAC = new ACNativeWindow(handle); } return true; } internal bool Visible { get { return SafeNativeMethods.IsWindowVisible(new HandleRef(this, Handle)); } } static internal bool AutoCompleteActive { get { if (inWndProcCnt>0) { return true; } foreach(object o in ACWindows.Values) { ACNativeWindow window = o as ACNativeWindow; if (window != null && window.Visible) { return true; } } return false; } } protected override void WndProc(ref Message m) { inWndProcCnt++; try { base.WndProc(ref m); } finally { inWndProcCnt--; } if (m.Msg == NativeMethods.WM_NCDESTROY) { Debug.Assert(ACWindows.ContainsKey(this.Handle)); ACWindows.Remove(this.Handle); //so we do not leak ac windows. } } [SuppressMessage("Microsoft.Performance", "CA1806:DoNotIgnoreMethodResults")] internal static void RegisterACWindow(IntPtr acHandle, bool subclass) { if (subclass && ACWindows.ContainsKey(acHandle)) { if (ACWindows[acHandle] == null) { ACWindows.Remove(acHandle); //if an external handle got destroyed, dont let it stop us. } } if (!ACWindows.ContainsKey(acHandle)) { if (subclass) { ACNativeWindow newAC = new ACNativeWindow(acHandle); } else { ACWindows.Add(acHandle, null); } } } ////// This method clears out null entries so we get a clean BEFORE and AFTER snapshot /// null entries are ACWindows that belong to someone else. /// [System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Performance", "CA1808:AvoidCallsThatBoxValueTypes")] // Perfectly acceptible when dealing with PropertyDescriptors internal static void ClearNullACWindows() { ArrayList nulllist = new ArrayList(); foreach (DictionaryEntry e in ACWindows) { if (e.Value == null) { nulllist.Add(e.Key); } } foreach (IntPtr handle in nulllist) { ACWindows.Remove(handle); } } } ////// This finds all autcomplete windows that belong to the active thread. /// private class AutoCompleteDropDownFinder { private const int MaxClassName = 256; private const string AutoCompleteClassName = "Auto-Suggest Dropdown"; bool shouldSubClass = false; //nonstatic internal void FindDropDowns() { FindDropDowns(true); } internal void FindDropDowns(bool subclass) { if (!subclass) { //generating a before snapshot -- lets lose the null handles ACNativeWindow.ClearNullACWindows(); } // Look for a popped up dropdown shouldSubClass = subclass; UnsafeNativeMethods.EnumThreadWindows(SafeNativeMethods.GetCurrentThreadId(), new NativeMethods.EnumThreadWindowsCallback(this.Callback), new HandleRef(null, IntPtr.Zero)); } private bool Callback(IntPtr hWnd, IntPtr lParam) { HandleRef hRef = new HandleRef(null, hWnd); // Check class name and see if it's visible if (GetClassName(hRef) == AutoCompleteClassName) { ACNativeWindow.RegisterACWindow(hRef.Handle, shouldSubClass); } return true; } static string GetClassName(HandleRef hRef) { StringBuilder sb = new StringBuilder(MaxClassName); UnsafeNativeMethods.GetClassName(hRef, sb, MaxClassName); return sb.ToString(); } } private FlatComboAdapter FlatComboBoxAdapter { get { FlatComboAdapter comboAdapter = Properties.GetObject(PropFlatComboAdapter) as FlatComboAdapter; if (comboAdapter == null || !comboAdapter.IsValid(this)) { comboAdapter = CreateFlatComboAdapterInstance(); Properties.SetObject(PropFlatComboAdapter, comboAdapter); } return comboAdapter; } } internal virtual FlatComboAdapter CreateFlatComboAdapterInstance() { return new FlatComboAdapter(this,/*smallButton=*/false); } internal class FlatComboAdapter { Rectangle outerBorder; Rectangle innerBorder; Rectangle innerInnerBorder; internal Rectangle dropDownRect; Rectangle whiteFillRect; Rectangle clientRect; RightToLeft origRightToLeft; // The combo box's RTL value when we were created private const int WhiteFillRectWidth = 5; // used for making the button look smaller than it is public FlatComboAdapter(ComboBox comboBox, bool smallButton) { clientRect = comboBox.ClientRectangle; int dropDownButtonWidth = System.Windows.Forms.SystemInformation.HorizontalScrollBarArrowWidth; outerBorder = new Rectangle(clientRect.Location, new Size(clientRect.Width - 1, clientRect.Height - 1)); innerBorder = new Rectangle(outerBorder.X + 1, outerBorder.Y + 1, outerBorder.Width - dropDownButtonWidth - 2, outerBorder.Height - 2); innerInnerBorder = new Rectangle(innerBorder.X + 1, innerBorder.Y + 1, innerBorder.Width - 2, innerBorder.Height - 2); dropDownRect = new Rectangle(innerBorder.Right + 1, innerBorder.Y, dropDownButtonWidth, innerBorder.Height+1); // fill in several pixels of the dropdown rect with white so that it looks like the combo button is thinner. if (smallButton) { whiteFillRect = dropDownRect; whiteFillRect.Width = WhiteFillRectWidth; dropDownRect.X += WhiteFillRectWidth; dropDownRect.Width -= WhiteFillRectWidth; } origRightToLeft = comboBox.RightToLeft; if (origRightToLeft == RightToLeft.Yes) { innerBorder.X = clientRect.Width - innerBorder.Right; innerInnerBorder.X = clientRect.Width - innerInnerBorder.Right; dropDownRect.X = clientRect.Width - dropDownRect.Right; whiteFillRect.X = clientRect.Width - whiteFillRect.Right + 1; // since we're filling, we need to move over to the next px. } } public bool IsValid(ComboBox combo) { return (combo.ClientRectangle == clientRect && combo.RightToLeft == origRightToLeft); } ////// /// Paints over the edges of the combo box to make it appear flat. /// public virtual void DrawFlatCombo(ComboBox comboBox, Graphics g) { if (comboBox.DropDownStyle == ComboBoxStyle.Simple){ return; } Color outerBorderColor = GetOuterBorderColor(comboBox); Color innerBorderColor = GetInnerBorderColor(comboBox); bool rightToLeft = comboBox.RightToLeft == RightToLeft.Yes; // draw the drop down DrawFlatComboDropDown(comboBox, g, dropDownRect); // when we are disabled there is one line of color that seems to eek through if backcolor is set // so lets erase it. if (!LayoutUtils.IsZeroWidthOrHeight(whiteFillRect)) { // fill in two more pixels with white so it looks smaller. using (Brush b = new SolidBrush(innerBorderColor)) { g.FillRectangle(b, whiteFillRect); } } // Draw the outer border if (outerBorderColor.IsSystemColor) { Pen outerBorderPen = SystemPens.FromSystemColor(outerBorderColor); g.DrawRectangle(outerBorderPen, outerBorder); if (rightToLeft) { g.DrawRectangle(outerBorderPen, new Rectangle(outerBorder.X, outerBorder.Y, dropDownRect.Width + 1, outerBorder.Height)); } else { g.DrawRectangle(outerBorderPen, new Rectangle(dropDownRect.X, outerBorder.Y, outerBorder.Right - dropDownRect.X, outerBorder.Height)); } } else { using (Pen outerBorderPen = new Pen(outerBorderColor)) { g.DrawRectangle(outerBorderPen, outerBorder); if (rightToLeft) { g.DrawRectangle(outerBorderPen, new Rectangle(outerBorder.X, outerBorder.Y, dropDownRect.Width + 1, outerBorder.Height)); } else { g.DrawRectangle(outerBorderPen, new Rectangle(dropDownRect.X, outerBorder.Y, outerBorder.Right - dropDownRect.X, outerBorder.Height)); } } } // Draw the inner border if (innerBorderColor.IsSystemColor) { Pen innerBorderPen = SystemPens.FromSystemColor(innerBorderColor); g.DrawRectangle(innerBorderPen, innerBorder); g.DrawRectangle(innerBorderPen, innerInnerBorder); } else { using (Pen innerBorderPen = new Pen(innerBorderColor)) { g.DrawRectangle(innerBorderPen, innerBorder); g.DrawRectangle(innerBorderPen, innerInnerBorder); } } // Draw a dark border around everything if we're in popup mode if ((!comboBox.Enabled) ||(comboBox.FlatStyle == FlatStyle.Popup)) { bool focused = comboBox.ContainsFocus || comboBox.MouseIsOver; Color borderPenColor = GetPopupOuterBorderColor(comboBox, focused); using (Pen borderPen = new Pen(borderPenColor)) { Pen innerPen = (comboBox.Enabled) ? borderPen : SystemPens.Control; // around the dropdown if (rightToLeft) { g.DrawRectangle(innerPen, new Rectangle(outerBorder.X, outerBorder.Y, dropDownRect.Width + 1, outerBorder.Height)); } else { g.DrawRectangle(innerPen, new Rectangle(dropDownRect.X, outerBorder.Y, outerBorder.Right - dropDownRect.X, outerBorder.Height)); } // around the whole combobox. g.DrawRectangle(borderPen, outerBorder); } } } ////// /// Paints over the edges of the combo box to make it appear flat. /// protected virtual void DrawFlatComboDropDown(ComboBox comboBox, Graphics g, Rectangle dropDownRect) { g.FillRectangle(SystemBrushes.Control, dropDownRect); Brush brush = (comboBox.Enabled) ? SystemBrushes.ControlText : SystemBrushes.ControlDark; Point middle = new Point(dropDownRect.Left + dropDownRect.Width / 2, dropDownRect.Top + dropDownRect.Height / 2); if (origRightToLeft == RightToLeft.Yes) { // if the width is odd - favor pushing it over one pixel left. middle.X -= (dropDownRect.Width % 2); } else { // if the width is odd - favor pushing it over one pixel right. middle.X += (dropDownRect.Width % 2); } g.FillPolygon(brush, new Point[] { new Point(middle.X - 2, middle.Y - 1), new Point(middle.X + 3, middle.Y - 1), new Point(middle.X, middle.Y + 2) }); } protected virtual Color GetOuterBorderColor(ComboBox comboBox) { return (comboBox.Enabled) ? SystemColors.Window : SystemColors.ControlDark; } protected virtual Color GetPopupOuterBorderColor(ComboBox comboBox, bool focused) { if (!comboBox.Enabled) { return SystemColors.ControlDark; } return (focused) ? SystemColors.ControlDark : SystemColors.Window; } protected virtual Color GetInnerBorderColor(ComboBox comboBox) { return (comboBox.Enabled) ? comboBox.BackColor : SystemColors.Control; } // this eliminates flicker by removing the pieces we're going to paint ourselves from // the update region. Note the UpdateRegionBox is the bounding box of the actual update region. // this is just here so we can quickly eliminate rectangles that arent in the update region. public void ValidateOwnerDrawRegions(ComboBox comboBox, Rectangle updateRegionBox) { NativeMethods.RECT validRect; if (comboBox != null) { return; } Rectangle topOwnerDrawArea = new Rectangle(0,0,comboBox.Width, innerBorder.Top); Rectangle bottomOwnerDrawArea = new Rectangle(0,innerBorder.Bottom,comboBox.Width, comboBox.Height-innerBorder.Bottom); Rectangle leftOwnerDrawArea = new Rectangle(0,0,innerBorder.Left, comboBox.Height); Rectangle rightOwnerDrawArea = new Rectangle(innerBorder.Right,0,comboBox.Width - innerBorder.Right,comboBox.Height); if (topOwnerDrawArea.IntersectsWith(updateRegionBox)) { validRect = new NativeMethods.RECT(topOwnerDrawArea); SafeNativeMethods.ValidateRect(new HandleRef(comboBox, comboBox.Handle), ref validRect); } if (bottomOwnerDrawArea.IntersectsWith(updateRegionBox)) { validRect = new NativeMethods.RECT(bottomOwnerDrawArea); SafeNativeMethods.ValidateRect(new HandleRef(comboBox, comboBox.Handle), ref validRect); } if (leftOwnerDrawArea.IntersectsWith(updateRegionBox)) { validRect = new NativeMethods.RECT(leftOwnerDrawArea); SafeNativeMethods.ValidateRect(new HandleRef(comboBox, comboBox.Handle), ref validRect); } if (rightOwnerDrawArea.IntersectsWith(updateRegionBox)) { validRect = new NativeMethods.RECT(rightOwnerDrawArea); SafeNativeMethods.ValidateRect(new HandleRef(comboBox, comboBox.Handle), ref validRect); } } } } } // File provided for Reference Use Only by Microsoft Corporation (c) 2007.
Link Menu
This book is available now!
Buy at Amazon US or
Buy at Amazon UK
- ParenExpr.cs
- ConstraintEnumerator.cs
- FastPropertyAccessor.cs
- ConfigXmlElement.cs
- ProxyWebPartManager.cs
- ping.cs
- CodeDOMUtility.cs
- CompositeDataBoundControl.cs
- SafeFindHandle.cs
- PixelShader.cs
- UpWmlMobileTextWriter.cs
- Parameter.cs
- CodeDomLocalizationProvider.cs
- MeshGeometry3D.cs
- HtmlAnchor.cs
- ToolboxBitmapAttribute.cs
- GroupItem.cs
- NativeRightsManagementAPIsStructures.cs
- DataViewSetting.cs
- HyperLinkField.cs
- EditingMode.cs
- WebPartTransformerAttribute.cs
- GradientBrush.cs
- AsymmetricKeyExchangeDeformatter.cs
- FormsAuthenticationModule.cs
- DataError.cs
- WindowsTreeView.cs
- HttpPostedFile.cs
- SqlAliaser.cs
- CachedPathData.cs
- Stroke2.cs
- ProfileSection.cs
- PageContentAsyncResult.cs
- XmlNodeComparer.cs
- ObjectToIdCache.cs
- LayoutEditorPart.cs
- FormViewDeleteEventArgs.cs
- CompoundFileStorageReference.cs
- TargetPerspective.cs
- VScrollProperties.cs
- Journaling.cs
- DataSourceComponent.cs
- Multiply.cs
- TreeNodeBindingCollection.cs
- LabelLiteral.cs
- CompositeDuplexBindingElement.cs
- PrinterSettings.cs
- HashCodeCombiner.cs
- NestPullup.cs
- WebPartConnectVerb.cs
- _SSPIWrapper.cs
- ChangeNode.cs
- XmlKeywords.cs
- TextSegment.cs
- SingleAnimationUsingKeyFrames.cs
- StringUtil.cs
- WindowsNonControl.cs
- PerfCounters.cs
- SymmetricAlgorithm.cs
- CompositeKey.cs
- ObjRef.cs
- MouseDevice.cs
- StringFunctions.cs
- IncrementalCompileAnalyzer.cs
- GetCardDetailsRequest.cs
- SmtpDateTime.cs
- DbResourceAllocator.cs
- HtmlTable.cs
- BoundColumn.cs
- ManualResetEvent.cs
- BamlStream.cs
- DuplexSecurityProtocolFactory.cs
- HttpsTransportBindingElement.cs
- UnsafeNativeMethods.cs
- BevelBitmapEffect.cs
- ResourceSetExpression.cs
- SmtpNegotiateAuthenticationModule.cs
- TextEndOfParagraph.cs
- Brush.cs
- ToolStripManager.cs
- XPathAncestorIterator.cs
- WebHttpBehavior.cs
- CompositeActivityTypeDescriptorProvider.cs
- DataObject.cs
- NodeFunctions.cs
- EastAsianLunisolarCalendar.cs
- AttributeTableBuilder.cs
- WebPartZoneBase.cs
- ContextInformation.cs
- DesignerAttribute.cs
- UncommonField.cs
- XNodeNavigator.cs
- VarInfo.cs
- ImageCodecInfo.cs
- ToggleButtonAutomationPeer.cs
- ForceCopyBuildProvider.cs
- SetterBase.cs
- Automation.cs
- HtmlContainerControl.cs
- DesigntimeLicenseContextSerializer.cs