File: System\Windows\Forms\Controls\ToolStrips\ToolStripItem.cs
Web Access
Project: src\src\System.Windows.Forms\src\System.Windows.Forms.csproj (System.Windows.Forms)
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
 
using System.Collections.Specialized;
using System.ComponentModel;
using System.Drawing;
using System.Drawing.Design;
using System.Drawing.Imaging;
using System.Windows.Forms.Layout;
using Windows.Win32.System.Ole;
using Com = Windows.Win32.System.Com;
using IComDataObject = System.Runtime.InteropServices.ComTypes.IDataObject;
 
namespace System.Windows.Forms;
 
[DesignTimeVisible(false)]
[Designer($"System.Windows.Forms.Design.ToolStripItemDesigner, {AssemblyRef.SystemDesign}")]
[DefaultEvent(nameof(Click))]
[ToolboxItem(false)]
[DefaultProperty(nameof(Text))]
public abstract partial class ToolStripItem :
    BindableComponent,
    ICommandBindingTargetProvider,
    IDropTarget,
    ISupportOleDropSource,
    IArrangedElement,
    IKeyboardToolTip
{
    private Rectangle _bounds = Rectangle.Empty;
    private PropertyStore? _propertyStore;
    private ToolStripItemAlignment _alignment = ToolStripItemAlignment.Left;
    private ToolStrip? _parent;
    private ToolStrip? _owner;
    private ToolStripItemOverflow _overflow = ToolStripItemOverflow.AsNeeded;
    private ToolStripItemPlacement _placement = ToolStripItemPlacement.None;
    private ContentAlignment _imageAlign = ContentAlignment.MiddleCenter;
    private ContentAlignment _textAlign = ContentAlignment.MiddleCenter;
    private TextImageRelation _textImageRelation = TextImageRelation.ImageBeforeText;
    private ToolStripItemImageIndexer? _imageIndexer;
    private ToolStripItemInternalLayout? _toolStripItemInternalLayout;
    private BitVector32 _state;
    private string? _toolTipText;
    private Color _imageTransparentColor = Color.Empty;
    private ToolStripItemImageScaling _imageScaling = ToolStripItemImageScaling.SizeToFit;
    private Size _cachedTextSize;
 
    private Padding _defaultMargin;
    private Padding _defaultStatusStripMargin;
 
    private ToolStripItemDisplayStyle _displayStyle = ToolStripItemDisplayStyle.ImageAndText;
 
    // Backing fields for the infrastructure to make ToolStripItem bindable and introduce (bindable) ICommand.
    private Input.ICommand? _command;
    private object? _commandParameter;
 
    private static readonly ArrangedElementCollection s_emptyChildCollection = new();
 
    internal static readonly object s_mouseDownEvent = new();
    internal static readonly object s_mouseEnterEvent = new();
    internal static readonly object s_mouseLeaveEvent = new();
    internal static readonly object s_mouseHoverEvent = new();
    internal static readonly object s_mouseMoveEvent = new();
    internal static readonly object s_mouseUpEvent = new();
    internal static readonly object s_clickEvent = new();
    internal static readonly object s_doubleClickEvent = new();
    internal static readonly object s_dragDropEvent = new();
    internal static readonly object s_dragEnterEvent = new();
    internal static readonly object s_dragLeaveEvent = new();
    internal static readonly object s_dragOverEvent = new();
    internal static readonly object s_displayStyleChangedEvent = new();
    internal static readonly object s_enabledChangedEvent = new();
    internal static readonly object s_internalEnabledChangedEvent = new();
    internal static readonly object s_fontChangedEvent = new();
    internal static readonly object s_foreColorChangedEvent = new();
    internal static readonly object s_backColorChangedEvent = new();
    internal static readonly object s_giveFeedbackEvent = new();
    internal static readonly object s_queryContinueDragEvent = new();
    internal static readonly object s_queryAccessibilityHelpEvent = new();
    internal static readonly object s_locationChangedEvent = new();
    internal static readonly object s_rightToLeftChangedEvent = new();
    internal static readonly object s_visibleChangedEvent = new();
    internal static readonly object s_availableChangedEvent = new();
    internal static readonly object s_ownerChangedEvent = new();
    internal static readonly object s_paintEvent = new();
    internal static readonly object s_textChangedEvent = new();
    internal static readonly object s_selectedChangedEvent = new();
 
    internal static readonly object s_commandChangedEvent = new();
    internal static readonly object s_commandParameterChangedEvent = new();
    internal static readonly object s_commandCanExecuteChangedEvent = new();
 
    // Property store keys for properties. The property store allocates most efficiently
    // in groups of four, so we try to lump properties in groups of four based on how
    // likely they are going to be used in a group.
 
    private static readonly int s_nameProperty = PropertyStore.CreateKey();
    private static readonly int s_textProperty = PropertyStore.CreateKey();
    private static readonly int s_backColorProperty = PropertyStore.CreateKey();
    private static readonly int s_foreColorProperty = PropertyStore.CreateKey();
 
    private static readonly int s_imageProperty = PropertyStore.CreateKey();
    private static readonly int s_fontProperty = PropertyStore.CreateKey();
    private static readonly int s_rightToLeftProperty = PropertyStore.CreateKey();
    private static readonly int s_tagProperty = PropertyStore.CreateKey();
 
    private static readonly int s_accessibilityProperty = PropertyStore.CreateKey();
    private static readonly int s_accessibleNameProperty = PropertyStore.CreateKey();
    private static readonly int s_accessibleRoleProperty = PropertyStore.CreateKey();
 
    private static readonly int s_accessibleDefaultActionDescriptionProperty = PropertyStore.CreateKey();
    private static readonly int s_accessibleDescriptionProperty = PropertyStore.CreateKey();
    private static readonly int s_textDirectionProperty = PropertyStore.CreateKey();
    private static readonly int s_mirroredImageProperty = PropertyStore.CreateKey();
 
    private static readonly int s_backgroundImageProperty = PropertyStore.CreateKey();
    private static readonly int s_backgroundImageLayoutProperty = PropertyStore.CreateKey();
 
    private static readonly int s_mergeActionProperty = PropertyStore.CreateKey();
    private static readonly int s_mergeIndexProperty = PropertyStore.CreateKey();
 
    private static readonly int s_stateAllowDrop = BitVector32.CreateMask();
    private static readonly int s_stateVisible = BitVector32.CreateMask(s_stateAllowDrop);
    private static readonly int s_stateEnabled = BitVector32.CreateMask(s_stateVisible);
    private static readonly int s_stateMouseDownAndNoDrag = BitVector32.CreateMask(s_stateEnabled);
    private static readonly int s_stateAutoSize = BitVector32.CreateMask(s_stateMouseDownAndNoDrag);
    private static readonly int s_statePressed = BitVector32.CreateMask(s_stateAutoSize);
    private static readonly int s_stateSelected = BitVector32.CreateMask(s_statePressed);
    private static readonly int s_stateConstructing = BitVector32.CreateMask(s_stateSelected);
    private static readonly int s_stateDisposed = BitVector32.CreateMask(s_stateConstructing);
    private static readonly int s_stateCurrentlyAnimatingImage = BitVector32.CreateMask(s_stateDisposed);
    private static readonly int s_stateDoubleClickEnabled = BitVector32.CreateMask(s_stateCurrentlyAnimatingImage);
    private static readonly int s_stateAutoToolTip = BitVector32.CreateMask(s_stateDoubleClickEnabled);
    private static readonly int s_stateSupportsRightClick = BitVector32.CreateMask(s_stateAutoToolTip);
    private static readonly int s_stateSupportsItemClick = BitVector32.CreateMask(s_stateSupportsRightClick);
    private static readonly int s_stateRightToLeftAutoMirrorImage = BitVector32.CreateMask(s_stateSupportsItemClick);
    private static readonly int s_stateInvalidMirroredImage = BitVector32.CreateMask(s_stateRightToLeftAutoMirrorImage);
    private static readonly int s_stateSupportsSpaceKey = BitVector32.CreateMask(s_stateInvalidMirroredImage);
    private static readonly int s_stateMouseDownAndUpMustBeInSameItem = BitVector32.CreateMask(s_stateSupportsSpaceKey);
    private static readonly int s_stateSupportsDisabledHotTracking = BitVector32.CreateMask(s_stateMouseDownAndUpMustBeInSameItem);
    private static readonly int s_stateUseAmbientMargin = BitVector32.CreateMask(s_stateSupportsDisabledHotTracking);
    private static readonly int s_stateDisposing = BitVector32.CreateMask(s_stateUseAmbientMargin);
 
    private long _lastClickTime;
    private int _deviceDpi = ScaleHelper.InitialSystemDpi;
    internal Font _defaultFont = ToolStripManager.DefaultFont;
 
    protected ToolStripItem()
    {
        ScaleConstants(ScaleHelper.InitialSystemDpi);
 
        _state[s_stateEnabled | s_stateAutoSize | s_stateVisible | s_stateConstructing | s_stateSupportsItemClick | s_stateInvalidMirroredImage | s_stateMouseDownAndUpMustBeInSameItem | s_stateUseAmbientMargin] = true;
        _state[s_stateAllowDrop | s_stateMouseDownAndNoDrag | s_stateSupportsRightClick | s_statePressed | s_stateSelected | s_stateDisposed | s_stateDoubleClickEnabled | s_stateRightToLeftAutoMirrorImage | s_stateSupportsSpaceKey] = false;
        SetAmbientMargin();
        Size = DefaultSize;
        DisplayStyle = DefaultDisplayStyle;
        CommonProperties.SetAutoSize(this, true);
        _state[s_stateConstructing] = false;
        AutoToolTip = DefaultAutoToolTip;
    }
 
    protected ToolStripItem(string? text, Image? image, EventHandler? onClick)
        : this(text, image, onClick, name: null)
    {
    }
 
    protected ToolStripItem(string? text, Image? image, EventHandler? onClick, string? name)
        : this()
    {
        Text = text;
        Image = image;
        if (onClick is not null)
        {
            Click += onClick;
        }
 
        Name = name;
    }
 
    /// <summary>
    ///  Indicates whether or not this control has an accessible object associated with it.
    /// </summary>
    internal bool IsAccessibilityObjectCreated => Properties.GetObject(s_accessibilityProperty) is AccessibleObject;
 
    /// <summary>
    ///  The Accessibility Object for this Control
    /// </summary>
    [Browsable(false)]
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
    [SRDescription(nameof(SR.ToolStripItemAccessibilityObjectDescr))]
    public AccessibleObject AccessibilityObject
    {
        get
        {
            AccessibleObject? accessibleObject = (AccessibleObject?)Properties.GetObject(s_accessibilityProperty);
            if (accessibleObject is null)
            {
                accessibleObject = CreateAccessibilityInstance();
                Properties.SetObject(s_accessibilityProperty, accessibleObject);
            }
 
            return accessibleObject;
        }
    }
 
    /// <summary>
    ///  The default action description of the control
    /// </summary>
    [SRCategory(nameof(SR.CatAccessibility))]
    [Browsable(false)]
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
    [SRDescription(nameof(SR.ToolStripItemAccessibleDefaultActionDescr))]
    public string? AccessibleDefaultActionDescription
    {
        get => (string?)Properties.GetObject(s_accessibleDefaultActionDescriptionProperty);
        set
        {
            Properties.SetObject(s_accessibleDefaultActionDescriptionProperty, value);
            OnAccessibleDefaultActionDescriptionChanged(EventArgs.Empty);
        }
    }
 
    /// <summary>
    ///  The accessible description of the control
    /// </summary>
    [SRCategory(nameof(SR.CatAccessibility))]
    [DefaultValue(null)]
    [Localizable(true)]
    [SRDescription(nameof(SR.ToolStripItemAccessibleDescriptionDescr))]
    public string? AccessibleDescription
    {
        get => (string?)Properties.GetObject(s_accessibleDescriptionProperty);
        set
        {
            Properties.SetObject(s_accessibleDescriptionProperty, value);
            OnAccessibleDescriptionChanged(EventArgs.Empty);
        }
    }
 
    /// <summary>
    ///  The accessible name of the control
    /// </summary>
    [SRCategory(nameof(SR.CatAccessibility))]
    [DefaultValue(null)]
    [Localizable(true)]
    [SRDescription(nameof(SR.ToolStripItemAccessibleNameDescr))]
    public string? AccessibleName
    {
        get => (string?)Properties.GetObject(s_accessibleNameProperty);
        set
        {
            Properties.SetObject(s_accessibleNameProperty, value);
            OnAccessibleNameChanged(EventArgs.Empty);
        }
    }
 
    /// <summary>
    ///  The accessible role of the control
    /// </summary>
    [SRCategory(nameof(SR.CatAccessibility))]
    [DefaultValue(AccessibleRole.Default)]
    [SRDescription(nameof(SR.ToolStripItemAccessibleRoleDescr))]
    public AccessibleRole AccessibleRole
    {
        get => Properties.GetValueOrDefault(s_accessibleRoleProperty, AccessibleRole.Default);
        set
        {
            SourceGenerated.EnumValidator.Validate(value);
            Properties.AddValue(s_accessibleRoleProperty, value);
            OnAccessibleRoleChanged(EventArgs.Empty);
        }
    }
 
    /// <summary>
    ///  Determines if the item aligns towards the beginning or end of the ToolStrip.
    /// </summary>
    [DefaultValue(ToolStripItemAlignment.Left)]
    [SRCategory(nameof(SR.CatLayout))]
    [SRDescription(nameof(SR.ToolStripItemAlignmentDescr))]
    public ToolStripItemAlignment Alignment
    {
        get => _alignment;
        set
        {
            SourceGenerated.EnumValidator.Validate(value);
 
            if (_alignment != value)
            {
                _alignment = value;
 
                if (ParentInternal is not null && ParentInternal.IsHandleCreated)
                {
                    ParentInternal.PerformLayout();
                }
            }
        }
    }
 
    /// <summary>
    ///  Determines if this item can be dragged.
    ///  This is EXACTLY like Control.AllowDrop - setting this to true WILL call
    ///  the dropTarget handlers. The ToolStripDropTargetManager is the one that
    ///  handles the routing of DropTarget events to the ToolStripItem's IDropTarget
    ///  methods.
    /// </summary>
    [SRCategory(nameof(SR.CatDragDrop))]
    [DefaultValue(false)]
    [SRDescription(nameof(SR.ToolStripItemAllowDropDescr))]
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    [Browsable(false)]
    public virtual bool AllowDrop
    {
        get => _state[s_stateAllowDrop];
        set
        {
            if (value != _state[s_stateAllowDrop])
            {
                EnsureParentDropTargetRegistered();
                _state[s_stateAllowDrop] = value;
            }
        }
    }
 
    /// <summary>
    ///  Determines whether we set the ToolStripItem to its preferred size
    /// </summary>
    [DefaultValue(true)]
    [SRCategory(nameof(SR.CatBehavior))]
    [RefreshProperties(RefreshProperties.All)]
    [Localizable(true)]
    [SRDescription(nameof(SR.ToolStripItemAutoSizeDescr))]
    [DesignerSerializationVisibility(DesignerSerializationVisibility.Visible)]
    public bool AutoSize
    {
        get => _state[s_stateAutoSize];
        set
        {
            if (_state[s_stateAutoSize] != value)
            {
                _state[s_stateAutoSize] = value;
                CommonProperties.SetAutoSize(this, value);
                InvalidateItemLayout(PropertyNames.AutoSize);
            }
        }
    }
 
    /// <summary>
    ///  !!!!This property ONLY works when toolStrip.ShowItemToolTips = true!!!!
    ///  if AutoToolTip is set to true we use the Text, if false, we use ToolTipText.
    /// </summary>
    [DefaultValue(false)]
    [SRDescription(nameof(SR.ToolStripItemAutoToolTipDescr))]
    [SRCategory(nameof(SR.CatBehavior))]
    public bool AutoToolTip
    {
        get => _state[s_stateAutoToolTip];
        set => _state[s_stateAutoToolTip] = value;
    }
 
    /// <summary>
    ///  as opposed to Visible, which returns whether or not the item and its parent are Visible
    ///  Available returns whether or not the item will be shown. Setting Available sets Visible and Vice/Versa
    /// </summary>
    [Browsable(false)]
    [SRDescription(nameof(SR.ToolStripItemAvailableDescr))]
    [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
    public bool Available
    {
        get => _state[s_stateVisible];
        set => SetVisibleCore(value);
    }
 
    [Browsable(false)]
    [SRCategory(nameof(SR.CatPropertyChanged))]
    [SRDescription(nameof(SR.ToolStripItemOnAvailableChangedDescr))]
    public event EventHandler? AvailableChanged
    {
        add => Events.AddHandler(s_availableChangedEvent, value);
        remove => Events.RemoveHandler(s_availableChangedEvent, value);
    }
 
    /// <summary>
    ///  Gets or sets the image that is displayed on a <see cref="Label"/>.
    /// </summary>
    [Localizable(true)]
    [SRCategory(nameof(SR.CatAppearance))]
    [SRDescription(nameof(SR.ToolStripItemImageDescr))]
    [DefaultValue(null)]
    public virtual Image? BackgroundImage
    {
        get => Properties.GetObject(s_backgroundImageProperty) as Image;
        set
        {
            if (BackgroundImage != value)
            {
                Properties.SetObject(s_backgroundImageProperty, value);
                Invalidate();
            }
        }
    }
 
    /// <summary>
    ///  Gets or sets the <see cref="Input.ICommand"/> whose <see cref="Input.ICommand.Execute(object?)"/>
    ///  method will be called when the ToolStripItem's <see cref="Click"/> event gets invoked.
    /// </summary>
    [Bindable(true)]
    [Browsable(false)]
    [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
    [SRCategory(nameof(SR.CatData))]
    [SRDescription(nameof(SR.CommandComponentCommandDescr))]
    public Input.ICommand? Command
    {
        get => _command;
        set => ICommandBindingTargetProvider.CommandSetter(this, value, ref _command);
    }
 
    /// <summary>
    ///  Occurs when the <see cref="Input.ICommand.CanExecute(object?)"/> status of the
    ///  <see cref="Input.ICommand"/> which is assigned to the <see cref="Command"/> property has changed.
    /// </summary>
    [SRCategory(nameof(SR.CatData))]
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    [SRDescription(nameof(SR.CommandCanExecuteChangedEventDescr))]
    public event EventHandler? CommandCanExecuteChanged
    {
        add => Events.AddHandler(s_commandCanExecuteChangedEvent, value);
        remove => Events.RemoveHandler(s_commandCanExecuteChangedEvent, value);
    }
 
    /// <summary>
    ///  Occurs when the assigned <see cref="Input.ICommand"/> of the <see cref="Command"/> property has changed.
    /// </summary>
    [SRCategory(nameof(SR.CatData))]
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    [SRDescription(nameof(SR.CommandChangedEventDescr))]
    public event EventHandler? CommandChanged
    {
        add => Events.AddHandler(s_commandChangedEvent, value);
        remove => Events.RemoveHandler(s_commandChangedEvent, value);
    }
 
    /// <summary>
    ///  Gets or sets the parameter that is passed to the <see cref="Input.ICommand"/>
    ///  which is assigned to the <see cref="Command"/> property.
    /// </summary>
    [Bindable(true)]
    [Browsable(false)]
    [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
    [SRCategory(nameof(SR.CatData))]
    [SRDescription(nameof(SR.CommandComponentCommandParameterDescr))]
    public object? CommandParameter
    {
        get => _commandParameter;
 
        set
        {
            if (!Equals(_commandParameter, value))
            {
                _commandParameter = value;
                OnCommandParameterChanged(EventArgs.Empty);
            }
        }
    }
 
    /// <summary>
    ///  Occurs when the value of the <see cref="CommandParameter"/> property has changed.
    /// </summary>
    [SRCategory(nameof(SR.CatData))]
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    [SRDescription(nameof(SR.CommandParameterChangedEventDescr))]
    public event EventHandler? CommandParameterChanged
    {
        add => Events.AddHandler(s_commandParameterChangedEvent, value);
        remove => Events.RemoveHandler(s_commandParameterChangedEvent, value);
    }
 
    // Every ToolStripItem needs to cache its last/current Parent's DeviceDpi
    // for PerMonitorV2 scaling purposes.
    internal virtual int DeviceDpi
    {
        get => _deviceDpi;
        set => _deviceDpi = value;
    }
 
    [SRCategory(nameof(SR.CatAppearance))]
    [DefaultValue(ImageLayout.Tile)]
    [Localizable(true)]
    [SRDescription(nameof(SR.ControlBackgroundImageLayoutDescr))]
    public virtual ImageLayout BackgroundImageLayout
    {
        get => Properties.TryGetObject(s_backgroundImageLayoutProperty, out ImageLayout imageLayout)
            ? imageLayout
            : ImageLayout.Tile;
        set
        {
            if (BackgroundImageLayout != value)
            {
                SourceGenerated.EnumValidator.Validate(value);
 
                Properties.SetObject(s_backgroundImageLayoutProperty, value);
                Invalidate();
            }
        }
    }
 
    /// <summary>
    ///  The BackColor of the item
    /// </summary>
    [SRCategory(nameof(SR.CatAppearance))]
    [SRDescription(nameof(SR.ToolStripItemBackColorDescr))]
    public virtual Color BackColor
    {
        get
        {
            Color c = RawBackColor;
            if (!c.IsEmpty)
            {
                return c;
            }
 
            Control? parent = ParentInternal;
            if (parent is not null)
            {
                return parent.BackColor;
            }
 
            return Control.DefaultBackColor;
        }
        set
        {
            Color previous = BackColor;
            Properties.AddOrRemoveValue(s_backColorProperty, value);
            if (!previous.Equals(BackColor))
            {
                OnBackColorChanged(EventArgs.Empty);
            }
        }
    }
 
    [SRCategory(nameof(SR.CatPropertyChanged))]
    [SRDescription(nameof(SR.ToolStripItemOnBackColorChangedDescr))]
    public event EventHandler? BackColorChanged
    {
        add => Events.AddHandler(s_backColorChangedEvent, value);
        remove => Events.RemoveHandler(s_backColorChangedEvent, value);
    }
 
    /// <summary>
    ///  The bounds of the item
    /// </summary>
    [Browsable(false)]
    public virtual Rectangle Bounds => _bounds;
 
    /// <summary>
    ///  Zero-based rectangle, same concept as ClientRect
    /// </summary>
    internal Rectangle ClientBounds
    {
        get
        {
            Rectangle client = _bounds;
            client.Location = Point.Empty;
            return client;
        }
    }
 
    [Browsable(false)]
    public Rectangle ContentRectangle
    {
        get
        {
            Rectangle content = LayoutUtils.InflateRect(InternalLayout.ContentRectangle, Padding);
            content.Size = LayoutUtils.UnionSizes(Size.Empty, content.Size);
            return content;
        }
    }
 
    /// <summary>
    ///  Determines whether or not the item can be selected.
    /// </summary>
    [Browsable(false)]
    public virtual bool CanSelect => Enabled;
 
    /// <remarks>
    ///  <para>Usually the same as can select, but things like the control box in an MDI window are exceptions</para>
    /// </remarks>
    internal virtual bool CanKeyboardSelect => CanSelect;
 
    /// <summary>
    ///  Occurs when the control is clicked.
    /// </summary>
    [SRCategory(nameof(SR.CatAction))]
    [SRDescription(nameof(SR.ToolStripItemOnClickDescr))]
    public event EventHandler? Click
    {
        add => Events.AddHandler(s_clickEvent, value);
        remove => Events.RemoveHandler(s_clickEvent, value);
    }
 
    [Browsable(false)]
    [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
    [DefaultValue(CommonProperties.DefaultAnchor)]
    public AnchorStyles Anchor
    {
        get
        {
            // since we don't support DefaultLayout go directly against the CommonProperties
            return CommonProperties.xGetAnchor(this);
        }
        set
        {
            if (value != Anchor)
            {
                // since we don't support DefaultLayout go directly against the CommonProperties
                CommonProperties.xSetAnchor(this, value);
                if (ParentInternal is not null)
                {
                    LayoutTransaction.DoLayout(this, ParentInternal, PropertyNames.Anchor);
                }
            }
        }
    }
 
    /// <summary>
    ///  This does not show up in the property grid because it only applies to flow and table layouts
    /// </summary>
    [Browsable(false)]
    [DefaultValue(CommonProperties.DefaultDock)]
    public DockStyle Dock
    {
        get
        {
            // since we don't support DefaultLayout go directly against the CommonProperties
            return CommonProperties.xGetDock(this);
        }
        set
        {
            SourceGenerated.EnumValidator.Validate(value);
 
            if (value != Dock)
            {
                // since we don't support DefaultLayout go directly against the CommonProperties
                CommonProperties.xSetDock(this, value);
                if (ParentInternal is not null)
                {
                    LayoutTransaction.DoLayout(this, ParentInternal, PropertyNames.Dock);
                }
            }
        }
    }
 
    /// <summary>
    ///  Default setting of auto tooltip when this object is created
    /// </summary>
    protected virtual bool DefaultAutoToolTip => false;
 
    /// <summary>
    ///  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.
    /// </summary>
    protected internal virtual Padding DefaultMargin
    {
        get
        {
            if (Owner is not null and StatusStrip)
            {
                return _defaultStatusStripMargin;
            }
 
            return _defaultMargin;
        }
    }
 
    /// <summary>
    ///  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.
    /// </summary>
    protected virtual Padding DefaultPadding => Padding.Empty;
 
    /// <summary>
    ///  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.
    /// </summary>
    protected virtual Size DefaultSize
    {
        get => ScaleHelper.IsThreadPerMonitorV2Aware ?
                ScaleHelper.ScaleToDpi(new Size(23, 23), DeviceDpi) :
                new Size(23, 23);
    }
 
    protected virtual ToolStripItemDisplayStyle DefaultDisplayStyle => ToolStripItemDisplayStyle.ImageAndText;
 
    /// <summary>
    ///  Specifies the default behavior of these items on ToolStripDropDowns when clicked.
    /// </summary>
    protected internal virtual bool DismissWhenClicked => true;
 
    /// <summary>
    ///  DisplayStyle specifies whether the image and text are rendered. This is not on the base
    ///  item class because different derived things will have different enumeration needs.
    /// </summary>
    [SRCategory(nameof(SR.CatAppearance))]
    [SRDescription(nameof(SR.ToolStripItemDisplayStyleDescr))]
    public virtual ToolStripItemDisplayStyle DisplayStyle
    {
        get => _displayStyle;
        set
        {
            if (_displayStyle != value)
            {
                SourceGenerated.EnumValidator.Validate(value);
 
                _displayStyle = value;
                if (!_state[s_stateConstructing])
                {
                    InvalidateItemLayout(PropertyNames.DisplayStyle);
                    OnDisplayStyleChanged(EventArgs.Empty);
                }
            }
        }
    }
 
    /// <summary>
    ///  Occurs when the display style has changed
    /// </summary>
    public event EventHandler? DisplayStyleChanged
    {
        add => Events.AddHandler(s_displayStyleChangedEvent, value);
        remove => Events.RemoveHandler(s_displayStyleChangedEvent, value);
    }
 
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    private static RightToLeft DefaultRightToLeft => RightToLeft.Inherit;
 
    /// <summary>
    ///  Occurs when the control is double clicked.
    /// </summary>
    [SRCategory(nameof(SR.CatAction))]
    [SRDescription(nameof(SR.ControlOnDoubleClickDescr))]
    public event EventHandler? DoubleClick
    {
        add => Events.AddHandler(s_doubleClickEvent, value);
        remove => Events.RemoveHandler(s_doubleClickEvent, value);
    }
 
    [DefaultValue(false)]
    [SRCategory(nameof(SR.CatBehavior))]
    [SRDescription(nameof(SR.ToolStripItemDoubleClickedEnabledDescr))]
    public bool DoubleClickEnabled
    {
        get => _state[s_stateDoubleClickEnabled];
        set => _state[s_stateDoubleClickEnabled] = value;
    }
 
    [SRCategory(nameof(SR.CatDragDrop))]
    [SRDescription(nameof(SR.ToolStripItemOnDragDropDescr))]
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    [Browsable(false)]
    public event DragEventHandler? DragDrop
    {
        add => Events.AddHandler(s_dragDropEvent, value);
        remove => Events.RemoveHandler(s_dragDropEvent, value);
    }
 
    [SRCategory(nameof(SR.CatDragDrop))]
    [SRDescription(nameof(SR.ToolStripItemOnDragEnterDescr))]
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    [Browsable(false)]
    public event DragEventHandler? DragEnter
    {
        add => Events.AddHandler(s_dragEnterEvent, value);
        remove => Events.RemoveHandler(s_dragEnterEvent, value);
    }
 
    [SRCategory(nameof(SR.CatDragDrop))]
    [SRDescription(nameof(SR.ToolStripItemOnDragOverDescr))]
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    [Browsable(false)]
    public event DragEventHandler? DragOver
    {
        add => Events.AddHandler(s_dragOverEvent, value);
        remove => Events.RemoveHandler(s_dragOverEvent, value);
    }
 
    [SRCategory(nameof(SR.CatDragDrop))]
    [SRDescription(nameof(SR.ToolStripItemOnDragLeaveDescr))]
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    [Browsable(false)]
    public event EventHandler? DragLeave
    {
        add => Events.AddHandler(s_dragLeaveEvent, value);
        remove => Events.RemoveHandler(s_dragLeaveEvent, value);
    }
 
    /// <summary>
    ///  Occurs when the control is clicked.
    /// </summary>
    [SRCategory(nameof(SR.CatBehavior))]
    [Localizable(true)]
    [SRDescription(nameof(SR.ToolStripItemEnabledDescr))]
    [DefaultValue(true)]
    public virtual bool Enabled
    {
        get
        {
            bool parentEnabled = true;
            if (Owner is not null)
            {
                parentEnabled = Owner.Enabled;
            }
 
            return _state[s_stateEnabled] && parentEnabled;
        }
        set
        {
            // flip disabled bit.
            if (_state[s_stateEnabled] != value)
            {
                _state[s_stateEnabled] = value;
                if (!_state[s_stateEnabled])
                {
                    bool wasSelected = _state[s_stateSelected];
                    // clear all the other states.
                    _state[s_stateSelected | s_statePressed] = false;
 
                    if (wasSelected)
                    {
                        KeyboardToolTipStateMachine.Instance.NotifyAboutLostFocus(this);
                        OnSelectedChanged(EventArgs.Empty);
                    }
                }
 
                OnEnabledChanged(EventArgs.Empty);
                Invalidate();
            }
 
            OnInternalEnabledChanged(EventArgs.Empty);
        }
    }
 
    [SRDescription(nameof(SR.ToolStripItemEnabledChangedDescr))]
    public event EventHandler? EnabledChanged
    {
        add => Events.AddHandler(s_enabledChangedEvent, value);
        remove => Events.RemoveHandler(s_enabledChangedEvent, value);
    }
 
    internal event EventHandler? InternalEnabledChanged
    {
        add => Events.AddHandler(s_internalEnabledChangedEvent, value);
        remove => Events.RemoveHandler(s_internalEnabledChangedEvent, value);
    }
 
    private void EnsureParentDropTargetRegistered()
    {
        ParentInternal?.DropTargetManager.EnsureRegistered();
    }
 
    /// <summary>
    ///  Retrieves the current font for this item. This will be the font used
    ///  by default for painting and text in the control.
    /// </summary>
    [SRCategory(nameof(SR.CatAppearance))]
    [SRDescription(nameof(SR.ToolStripItemForeColorDescr))]
    public virtual Color ForeColor
    {
        get
        {
            Color foreColor = Properties.GetValueOrDefault<Color>(s_foreColorProperty);
            if (!foreColor.IsEmpty)
            {
                return foreColor;
            }
 
            Control? parent = ParentInternal;
            if (parent is not null)
            {
                return parent.ForeColor;
            }
 
            return Control.DefaultForeColor;
        }
        set
        {
            Color previous = ForeColor;
            Properties.AddOrRemoveValue(s_foreColorProperty, value);
            if (!previous.Equals(ForeColor))
            {
                OnForeColorChanged(EventArgs.Empty);
            }
        }
    }
 
    [SRCategory(nameof(SR.CatPropertyChanged))]
    [SRDescription(nameof(SR.ToolStripItemOnForeColorChangedDescr))]
    public event EventHandler? ForeColorChanged
    {
        add => Events.AddHandler(s_foreColorChangedEvent, value);
        remove => Events.RemoveHandler(s_foreColorChangedEvent, value);
    }
 
    /// <summary>
    ///  Retrieves the current font for this control. This will be the font used
    ///  by default for painting and text in the control.
    /// </summary>
    [SRCategory(nameof(SR.CatAppearance))]
    [Localizable(true)]
    [SRDescription(nameof(SR.ToolStripItemFontDescr))]
    [AllowNull]
    public virtual Font Font
    {
        get
        {
            if (TryGetExplicitlySetFont(out Font? font))
            {
                return font;
            }
 
            font = GetOwnerFont();
            if (font is not null)
            {
                return font;
            }
 
            return ScaleHelper.IsThreadPerMonitorV2Aware ? _defaultFont : ToolStripManager.DefaultFont;
        }
        set
        {
            var local = (Font?)Properties.GetObject(s_fontProperty);
            if (local != value)
            {
                Properties.SetObject(s_fontProperty, value);
                OnFontChanged(EventArgs.Empty);
            }
        }
    }
 
    [SRCategory(nameof(SR.CatDragDrop))]
    [SRDescription(nameof(SR.ToolStripItemOnGiveFeedbackDescr))]
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    [Browsable(false)]
    public event GiveFeedbackEventHandler? GiveFeedback
    {
        add => Events.AddHandler(s_giveFeedbackEvent, value);
        remove => Events.RemoveHandler(s_giveFeedbackEvent, value);
    }
 
    /// <summary>
    ///  The height of this control
    /// </summary>
    [SRCategory(nameof(SR.CatLayout))]
    [Browsable(false)]
    [EditorBrowsable(EditorBrowsableState.Always)]
    [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
    public int Height
    {
        get => Bounds.Height;
        set
        {
            Rectangle currentBounds = Bounds;
            SetBounds(currentBounds.X, currentBounds.Y, currentBounds.Width, value);
        }
    }
 
    /// <summary>
    ///  ToolStripItems do not have children. For perf reasons always return a static empty collection.
    ///  Consider creating readonly collection.
    /// </summary>
    ArrangedElementCollection IArrangedElement.Children => s_emptyChildCollection;
 
    /// <summary>
    ///  Should not be exposed as this returns an unexposed type.
    /// </summary>
    IArrangedElement? IArrangedElement.Container => ParentInternal ?? Owner;
 
    Rectangle IArrangedElement.DisplayRectangle => Bounds;
 
    bool IArrangedElement.ParticipatesInLayout
    {
        get
        {
            // this can be different than "Visible" property as "Visible" takes into account whether or not you
            // are parented and your parent is visible.
            return _state[s_stateVisible];
        }
    }
 
    PropertyStore IArrangedElement.Properties => Properties;
 
    void IArrangedElement.SetBounds(Rectangle bounds, BoundsSpecified specified)
    {
        // in this case the parent is telling us to refresh our bounds - don't
        // call PerformLayout
        SetBounds(bounds);
    }
 
    void IArrangedElement.PerformLayout(IArrangedElement container, string? propertyName)
    {
    }
 
    /// <summary>
    ///  Gets or sets the alignment of the image on the label control.
    /// </summary>
    [DefaultValue(ContentAlignment.MiddleCenter)]
    [Localizable(true)]
    [SRCategory(nameof(SR.CatAppearance))]
    [SRDescription(nameof(SR.ToolStripItemImageAlignDescr))]
    public ContentAlignment ImageAlign
    {
        get => _imageAlign;
        set
        {
            SourceGenerated.EnumValidator.Validate(value);
 
            if (_imageAlign != value)
            {
                _imageAlign = value;
                InvalidateItemLayout(PropertyNames.ImageAlign);
            }
        }
    }
 
    /// <summary>
    ///  Gets or sets the image that is displayed on a <see cref="Label"/>.
    /// </summary>
    [Localizable(true)]
    [SRCategory(nameof(SR.CatAppearance))]
    [SRDescription(nameof(SR.ToolStripItemImageDescr))]
    public virtual Image? Image
    {
        get
        {
            Image? image = (Image?)Properties.GetObject(s_imageProperty);
            if (image is null && Owner?.ImageList is not null && ImageIndexer.ActualIndex >= 0)
            {
                bool disposing = _state[s_stateDisposing];
                if (!disposing && ImageIndexer.ActualIndex < Owner.ImageList.Images.Count)
                {
                    // CACHE (by design). If we fetched out of the image list every time it would dramatically hit perf.
                    image = Owner.ImageList.Images[ImageIndexer.ActualIndex];
                    _state[s_stateInvalidMirroredImage] = true;
                    Properties.SetObject(s_imageProperty, image);
                    return image;
                }
 
                return null;
            }
 
            return image;
        }
        set
        {
            if (Image == value)
            {
                return;
            }
 
            StopAnimate();
 
            if (value is Bitmap bmp && ImageTransparentColor != Color.Empty)
            {
                if (bmp.RawFormat.Guid != ImageFormat.Icon.Guid && !ImageAnimator.CanAnimate(bmp))
                {
                    bmp.MakeTransparent(ImageTransparentColor);
                }
 
                value = bmp;
            }
 
            if (value is not null)
            {
                ImageIndex = ImageList.Indexer.DefaultIndex;
            }
 
            Properties.SetObject(s_imageProperty, value);
            _state[s_stateInvalidMirroredImage] = true;
 
            Animate();
            InvalidateItemLayout(PropertyNames.Image);
        }
    }
 
    [Localizable(true)]
    [SRCategory(nameof(SR.CatAppearance))]
    [SRDescription(nameof(SR.ToolStripItemImageTransparentColorDescr))]
    public Color ImageTransparentColor
    {
        get => _imageTransparentColor;
        set
        {
            if (_imageTransparentColor != value)
            {
                _imageTransparentColor = value;
                if (Image is Bitmap currentImage && value != Color.Empty)
                {
                    if (currentImage.RawFormat.Guid != ImageFormat.Icon.Guid && !ImageAnimator.CanAnimate(currentImage))
                    {
                        currentImage.MakeTransparent(_imageTransparentColor);
                    }
                }
 
                Invalidate();
            }
        }
    }
 
    /// <summary>
    ///  Returns the ToolStripItem's currently set image index
    ///  Here for compat only - this is NOT to be visible at DT.
    /// </summary>
    [SRDescription(nameof(SR.ToolStripItemImageIndexDescr))]
    [Localizable(true)]
    [SRCategory(nameof(SR.CatBehavior))]
    [RefreshProperties(RefreshProperties.Repaint)]
    [TypeConverter(typeof(NoneExcludedImageIndexConverter))]
    [Editor($"System.Windows.Forms.Design.ImageIndexEditor, {AssemblyRef.SystemDesign}", typeof(UITypeEditor))]
    [Browsable(false)]
    [RelatedImageList("Owner.ImageList")]
    public int ImageIndex
    {
        get
        {
            if ((Owner is not null) && ImageIndexer.Index != ImageList.Indexer.DefaultIndex
                && Owner.ImageList is not null && ImageIndexer.Index >= Owner.ImageList.Images.Count)
            {
                return Owner.ImageList.Images.Count - 1;
            }
 
            return ImageIndexer.Index;
        }
        set
        {
            ArgumentOutOfRangeException.ThrowIfLessThan(value, ImageList.Indexer.DefaultIndex);
 
            ImageIndexer.Index = value;
            _state[s_stateInvalidMirroredImage] = true;
 
            // Set the Image Property to null
            Properties.SetObject(s_imageProperty, null);
 
            InvalidateItemLayout(PropertyNames.ImageIndex);
        }
    }
 
    internal ToolStripItemImageIndexer ImageIndexer
        => _imageIndexer ??= new ToolStripItemImageIndexer(this);
 
    /// <summary>
    ///  Returns the ToolStripItem's currently set image index
    ///  Here for compat only - this is NOT to be visible at DT.
    /// </summary>
    [SRDescription(nameof(SR.ToolStripItemImageKeyDescr))]
    [SRCategory(nameof(SR.CatBehavior))]
    [Localizable(true)]
    [TypeConverter(typeof(ImageKeyConverter))]
    [RefreshProperties(RefreshProperties.Repaint)]
    [Editor($"System.Windows.Forms.Design.ToolStripImageIndexEditor, {AssemblyRef.SystemDesign}", typeof(UITypeEditor))]
    [Browsable(false)]
    [RelatedImageList("Owner.ImageList")]
    [AllowNull]
    public string ImageKey
    {
        get => ImageIndexer.Key;
        set
        {
            ImageIndexer.Key = value;
            _state[s_stateInvalidMirroredImage] = true;
            Properties.SetObject(s_imageProperty, null);
 
            InvalidateItemLayout(PropertyNames.ImageKey);
        }
    }
 
    [SRCategory(nameof(SR.CatAppearance))]
    [DefaultValue(ToolStripItemImageScaling.SizeToFit)]
    [Localizable(true)]
    [SRDescription(nameof(SR.ToolStripItemImageScalingDescr))]
    public ToolStripItemImageScaling ImageScaling
    {
        get => _imageScaling;
        set
        {
            SourceGenerated.EnumValidator.Validate(value);
 
            if (_imageScaling != value)
            {
                _imageScaling = value;
                InvalidateItemLayout(PropertyNames.ImageScaling);
            }
        }
    }
 
    /// <summary>
    ///  This object helps determine where the image and text should be drawn.
    /// </summary>
    internal ToolStripItemInternalLayout InternalLayout
        => _toolStripItemInternalLayout ??= CreateInternalLayout();
 
    internal bool IsForeColorSet
    {
        get
        {
            Color color = Properties.GetValueOrDefault<Color>(s_foreColorProperty);
            if (!color.IsEmpty)
            {
                return true;
            }
 
            ToolStrip? parent = ParentInternal;
            if (parent is not null)
            {
                return parent.ShouldSerializeForeColor();
            }
 
            return false;
        }
    }
 
    /// <summary>
    ///  This is used by ToolStrip to pass on the mouseMessages for ActiveDropDown.
    /// </summary>
    internal bool IsInDesignMode => DesignMode;
 
    [Browsable(false)]
    public bool IsDisposed => _state[s_stateDisposed];
 
    [Browsable(false)]
    public bool IsOnDropDown
    {
        get
        {
            if (ParentInternal is not null)
            {
                return ParentInternal.IsDropDown;
            }
            else if (Owner is not null && Owner.IsDropDown)
            {
                return true;
            }
 
            return false;
        }
    }
 
    [Browsable(false)]
    public bool IsOnOverflow => Placement == ToolStripItemPlacement.Overflow;
 
    /// <summary>
    ///  Specifies whether the control is willing to process mnemonics when hosted in an container ActiveX (Ax Sourcing).
    /// </summary>
    internal virtual bool IsMnemonicsListenerAxSourced => true;
 
    /// <summary>
    ///  Occurs when the location of the ToolStripItem has been updated -- usually by layout by its
    ///  owner of ToolStrips
    /// </summary>
    [SRCategory(nameof(SR.CatLayout))]
    [SRDescription(nameof(SR.ToolStripItemOnLocationChangedDescr))]
    public event EventHandler? LocationChanged
    {
        add => Events.AddHandler(s_locationChangedEvent, value);
        remove => Events.RemoveHandler(s_locationChangedEvent, value);
    }
 
    /// <summary>
    ///  Specifies the external spacing between this item and any other item or the ToolStrip.
    /// </summary>
    [SRDescription(nameof(SR.ToolStripItemMarginDescr))]
    [SRCategory(nameof(SR.CatLayout))]
    public Padding Margin
    {
        get => CommonProperties.GetMargin(this);
        set
        {
            if (Margin != value)
            {
                _state[s_stateUseAmbientMargin] = false;
                CommonProperties.SetMargin(this, value);
            }
        }
    }
 
    /// <summary>
    ///  Specifies the merge action when merging two ToolStrip.
    /// </summary>
    [SRDescription(nameof(SR.ToolStripMergeActionDescr))]
    [DefaultValue(MergeAction.Append)]
    [SRCategory(nameof(SR.CatLayout))]
    public MergeAction MergeAction
    {
        get => Properties.GetValueOrDefault(s_mergeActionProperty, MergeAction.Append);
        set
        {
            SourceGenerated.EnumValidator.Validate(value);
            Properties.AddValue(s_mergeActionProperty, value);
        }
    }
 
    /// <summary>
    ///  Specifies the merge action when merging two ToolStrip.
    /// </summary>
    [SRDescription(nameof(SR.ToolStripMergeIndexDescr))]
    [DefaultValue(-1)]
    [SRCategory(nameof(SR.CatLayout))]
    public int MergeIndex
    {
        get => Properties.GetValueOrDefault(s_mergeIndexProperty, -1);
        set => Properties.AddValue(s_mergeIndexProperty, value);
    }
 
    internal bool MouseDownAndUpMustBeInSameItem
    {
        get => _state[s_stateMouseDownAndUpMustBeInSameItem];
        set => _state[s_stateMouseDownAndUpMustBeInSameItem] = value;
    }
 
    /// <summary>
    ///  Occurs when the mouse pointer is over the control and a mouse button is
    ///  pressed.
    /// </summary>
    [SRCategory(nameof(SR.CatMouse))]
    [SRDescription(nameof(SR.ToolStripItemOnMouseDownDescr))]
    public event MouseEventHandler? MouseDown
    {
        add => Events.AddHandler(s_mouseDownEvent, value);
        remove => Events.RemoveHandler(s_mouseDownEvent, value);
    }
 
    /// <summary>
    ///  Occurs when the mouse pointer enters the control.
    /// </summary>
    [SRCategory(nameof(SR.CatMouse))]
    [SRDescription(nameof(SR.ToolStripItemOnMouseEnterDescr))]
    public event EventHandler? MouseEnter
    {
        add => Events.AddHandler(s_mouseEnterEvent, value);
        remove => Events.RemoveHandler(s_mouseEnterEvent, value);
    }
 
    /// <summary>
    ///  Occurs when the mouse pointer leaves the control.
    /// </summary>
    [SRCategory(nameof(SR.CatMouse))]
    [SRDescription(nameof(SR.ToolStripItemOnMouseLeaveDescr))]
    public event EventHandler? MouseLeave
    {
        add => Events.AddHandler(s_mouseLeaveEvent, value);
        remove => Events.RemoveHandler(s_mouseLeaveEvent, value);
    }
 
    /// <summary>
    ///  Occurs when the mouse pointer hovers over the control.
    /// </summary>
    [SRCategory(nameof(SR.CatMouse))]
    [SRDescription(nameof(SR.ToolStripItemOnMouseHoverDescr))]
    public event EventHandler? MouseHover
    {
        add => Events.AddHandler(s_mouseHoverEvent, value);
        remove => Events.RemoveHandler(s_mouseHoverEvent, value);
    }
 
    /// <summary>
    ///  Occurs when the mouse pointer is moved over the control.
    /// </summary>
    [SRCategory(nameof(SR.CatMouse))]
    [SRDescription(nameof(SR.ToolStripItemOnMouseMoveDescr))]
    public event MouseEventHandler? MouseMove
    {
        add => Events.AddHandler(s_mouseMoveEvent, value);
        remove => Events.RemoveHandler(s_mouseMoveEvent, value);
    }
 
    /// <summary>
    ///  Occurs when the mouse pointer is over the control and a mouse button is released.
    /// </summary>
    [SRCategory(nameof(SR.CatMouse))]
    [SRDescription(nameof(SR.ToolStripItemOnMouseUpDescr))]
    public event MouseEventHandler? MouseUp
    {
        add => Events.AddHandler(s_mouseUpEvent, value);
        remove => Events.RemoveHandler(s_mouseUpEvent, value);
    }
 
    /// <summary>
    ///  Name of this control. The designer will set this to the same
    ///  as the programmatic Id "(name)" of the control. The name can be
    ///  used as a key into the ControlCollection.
    /// </summary>
    [Browsable(false)]
    [DefaultValue(null)]
    public string? Name
    {
        get => WindowsFormsUtils.GetComponentName(this, (string?)Properties.GetObject(s_nameProperty));
        set
        {
            if (DesignMode)
            {
                return;
            }
 
            Properties.SetObject(s_nameProperty, value);
        }
    }
 
    /// <summary>
    ///  The owner of this ToolStripItem. The owner is essentially a backPointer to
    ///  the ToolStrip who contains this item in it's item collection. Handy for getting
    ///  to things such as the ImageList, which would be defined on the ToolStrip.
    /// </summary>
    [Browsable(false)]
    [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
    public ToolStrip? Owner
    {
        get => _owner;
        set
        {
            if (_owner != value)
            {
                _owner?.Items.Remove(this);
 
                value?.Items.Add(this);
            }
        }
    }
 
    /// <summary>
    ///  Returns the "parent" item on the preceding menu which has spawned this item.
    ///  e.g. File->Open  the OwnerItem of Open is File.
    /// </summary>
    [Browsable(false)]
    [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
    public ToolStripItem? OwnerItem
    {
        get
        {
            ToolStripDropDown? currentParent = null;
            if (ParentInternal is not null)
            {
                currentParent = ParentInternal as ToolStripDropDown;
            }
            else if (Owner is not null)
            {
                // parent may be null, but we may be "owned" by a collection.
                currentParent = Owner as ToolStripDropDown;
            }
 
            if (currentParent is not null)
            {
                return currentParent.OwnerItem;
            }
 
            return null;
        }
    }
 
    [SRCategory(nameof(SR.CatBehavior))]
    [SRDescription(nameof(SR.ToolStripItemOwnerChangedDescr))]
    public event EventHandler? OwnerChanged
    {
        add => Events.AddHandler(s_ownerChangedEvent, value);
        remove => Events.RemoveHandler(s_ownerChangedEvent, value);
    }
 
    [SRCategory(nameof(SR.CatAppearance))]
    [SRDescription(nameof(SR.ToolStripItemOnPaintDescr))]
    public event PaintEventHandler? Paint
    {
        add => Events.AddHandler(s_paintEvent, value);
        remove => Events.RemoveHandler(s_paintEvent, value);
    }
 
    /// <summary>
    ///  The parent of this ToolStripItem. This can be distinct from the owner because
    ///  the item can fall onto another window (overflow). In this case the overflow
    ///  would be the parent but the original ToolStrip would be the Owner. The "parent"
    ///  ToolStrip will be firing things like paint events - where as the "owner" ToolStrip
    ///  will be containing shared data like image lists. Typically the only one who should
    ///  set the parent property is the layout manager on the ToolStrip.
    /// </summary>
    [Browsable(false)]
    [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
    protected internal ToolStrip? Parent
    {
        get => ParentInternal;
        set => ParentInternal = value;
    }
 
    /// <summary>
    ///  Specifies whether or not the item is glued to the ToolStrip or overflow or
    ///  can float between the two.
    /// </summary>
    [DefaultValue(ToolStripItemOverflow.AsNeeded)]
    [SRDescription(nameof(SR.ToolStripItemOverflowDescr))]
    [SRCategory(nameof(SR.CatLayout))]
    public ToolStripItemOverflow Overflow
    {
        get => _overflow;
        set
        {
            SourceGenerated.EnumValidator.Validate(value);
 
            if (_overflow != value)
            {
                _overflow = value;
                if (Owner is not null)
                {
                    LayoutTransaction.DoLayout(Owner, Owner, "Overflow");
                }
            }
        }
    }
 
    /// <summary>
    ///  Specifies the internal spacing between the contents and the edges of the item
    /// </summary>
    [SRDescription(nameof(SR.ToolStripItemPaddingDescr))]
    [SRCategory(nameof(SR.CatLayout))]
    public virtual Padding Padding
    {
        get => CommonProperties.GetPadding(this, DefaultPadding);
        set
        {
            if (Padding != value)
            {
                CommonProperties.SetPadding(this, value);
                InvalidateItemLayout(PropertyNames.Padding);
            }
        }
    }
 
    /// <summary>
    ///  This is explicitly a ToolStrip, because only ToolStrips know how to manage ToolStripItems
    /// </summary>
    internal ToolStrip? ParentInternal
    {
        get => _parent;
        set
        {
            if (_parent != value)
            {
                ToolStrip? oldParent = _parent;
                _parent = value;
                OnParentChanged(oldParent, value);
            }
        }
    }
 
    /// <summary>
    ///  Where the item actually ended up.
    /// </summary>
    [Browsable(false)]
    public ToolStripItemPlacement Placement => _placement;
 
    internal Size PreferredImageSize
    {
        get
        {
            if ((DisplayStyle & ToolStripItemDisplayStyle.Image) != ToolStripItemDisplayStyle.Image)
            {
                return Size.Empty;
            }
 
            Image? image = (Image?)Properties.GetObject(s_imageProperty);
            bool usingImageList = ((Owner is not null) && (Owner.ImageList is not null) && (ImageIndexer.ActualIndex >= 0));
 
            if (ImageScaling == ToolStripItemImageScaling.SizeToFit)
            {
                ToolStrip? ownerToolStrip = Owner;
                if (ownerToolStrip is not null && (image is not null || usingImageList))
                {
                    return ownerToolStrip.ImageScalingSize;
                }
            }
 
            Size imageSize = Size.Empty;
            if (usingImageList)
            {
                imageSize = Owner?.ImageList?.ImageSize ?? Size.Empty;
            }
            else
            {
                imageSize = (image is null) ? Size.Empty : image.Size;
            }
 
            return imageSize;
        }
    }
 
    /// <summary>
    ///  Retrieves our internal property storage object. If you have a property
    ///  whose value is not always set, you should store it in here to save
    ///  space.
    /// </summary>
    internal PropertyStore Properties => _propertyStore ??= new PropertyStore();
 
    /// <summary>
    ///  Returns true if the state of the item is pushed
    /// </summary>
    [Browsable(false)]
    [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
    public virtual bool Pressed => CanSelect && _state[s_statePressed];
 
    [SRCategory(nameof(SR.CatDragDrop))]
    [SRDescription(nameof(SR.ToolStripItemOnQueryContinueDragDescr))]
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    [Browsable(false)]
    public event QueryContinueDragEventHandler? QueryContinueDrag
    {
        add => Events.AddHandler(s_queryContinueDragEvent, value);
        remove => Events.RemoveHandler(s_queryContinueDragEvent, value);
    }
 
    [SRCategory(nameof(SR.CatBehavior))]
    [SRDescription(nameof(SR.ToolStripItemOnQueryAccessibilityHelpDescr))]
    public event QueryAccessibilityHelpEventHandler? QueryAccessibilityHelp
    {
        add => Events.AddHandler(s_queryAccessibilityHelpEvent, value);
        remove => Events.RemoveHandler(s_queryAccessibilityHelpEvent, value);
    }
 
    /// <summary>
    ///  Returns the value of the backColor field -- no asking the parent with its color is, etc.
    /// </summary>
    internal Color RawBackColor => Properties.GetValueOrDefault<Color>(s_backColorProperty);
 
    /// <summary>
    ///  Returns the parent <see cref="ToolStrip"/>'s renderer
    /// </summary>
    protected internal virtual ToolStripRenderer? Renderer
    {
        get
        {
            if (Owner is not null)
            {
                return Owner.Renderer;
            }
 
            return ParentInternal?.Renderer;
        }
    }
 
    /// <summary>
    ///  This is used for international applications where the language
    ///  is written from RightToLeft. When this property is true,
    ///  control placement and text will be from right to left.
    /// </summary>
    [SRCategory(nameof(SR.CatAppearance))]
    [Localizable(true)]
    [SRDescription(nameof(SR.ToolStripItemRightToLeftDescr))]
    public virtual RightToLeft RightToLeft
    {
        get
        {
            if (!Properties.TryGetValue(s_rightToLeftProperty, out RightToLeft rightToLeft))
            {
                rightToLeft = RightToLeft.Inherit;
            }
 
            if (rightToLeft == RightToLeft.Inherit)
            {
                if (Owner is not null)
                {
                    rightToLeft = Owner.RightToLeft;
                }
                else if (ParentInternal is not null)
                {
                    // case for Overflow & Grip
                    rightToLeft = ParentInternal.RightToLeft;
                }
                else
                {
                    rightToLeft = DefaultRightToLeft;
                }
            }
 
            return rightToLeft;
        }
 
        set
        {
            SourceGenerated.EnumValidator.Validate(value);
 
            RightToLeft oldValue = RightToLeft;
 
            if (Properties.ContainsKey(s_rightToLeftProperty) || value != RightToLeft.Inherit)
            {
                Properties.AddValue(s_rightToLeftProperty, value);
            }
 
            if (oldValue != RightToLeft)
            {
                OnRightToLeftChanged(EventArgs.Empty);
            }
        }
    }
 
    /// <summary>
    ///  Mirrors the image when RTL.Yes.
    ///  Note we do not change what is returned back from the Image property as this would cause problems with serialization.
    ///  Instead we only change what is painted - there's an internal MirroredImage property which fills in as
    ///  e.Image in the ToolStripItemImageRenderEventArgs if the item is RTL.Yes and AutoMirrorImage is turned on.
    /// </summary>
    [DefaultValue(false)]
    [SRCategory(nameof(SR.CatAppearance))]
    [Localizable(true)]
    [SRDescription(nameof(SR.ToolStripItemRightToLeftAutoMirrorImageDescr))]
    public bool RightToLeftAutoMirrorImage
    {
        get => _state[s_stateRightToLeftAutoMirrorImage];
        set
        {
            if (_state[s_stateRightToLeftAutoMirrorImage] != value)
            {
                _state[s_stateRightToLeftAutoMirrorImage] = value;
                Invalidate();
            }
        }
    }
 
    internal Image? MirroredImage
    {
        get
        {
            if (_state[s_stateInvalidMirroredImage])
            {
                Image? image = Image;
                if (image is null)
                {
                    return null;
                }
 
                Image mirroredImage = (Image)image.Clone();
                mirroredImage.RotateFlip(RotateFlipType.RotateNoneFlipX);
 
                Properties.SetObject(s_mirroredImageProperty, mirroredImage);
                _state[s_stateInvalidMirroredImage] = false;
                return mirroredImage;
            }
 
            return Properties.GetObject(s_mirroredImageProperty) as Image;
        }
    }
 
    bool? ICommandBindingTargetProvider.PreviousEnabledStatus { get; set; }
 
    [SRCategory(nameof(SR.CatPropertyChanged))]
    [SRDescription(nameof(SR.ToolStripItemOnRightToLeftChangedDescr))]
    public event EventHandler? RightToLeftChanged
    {
        add => Events.AddHandler(s_rightToLeftChangedEvent, value);
        remove => Events.RemoveHandler(s_rightToLeftChangedEvent, value);
    }
 
    /// <summary>
    ///  if the item is selected we return true.
    ///
    ///  FAQ: Why don't we have a Hot or MouseIsOver property?
    ///  After going through the scenarios, we've decided NOT to add a separate MouseIsOver or Hot flag to ToolStripItem.
    ///  The thing to use is 'Selected'.
    ///  Why?  While the selected thing can be different than the moused over item, the selected item is ALWAYS the
    ///  one you want to paint differently
    ///
    ///  Scenario 1:  Keyboard select an item then select a different item with the mouse.
    ///  -          Do Alt+F to expand your File menu, keyboard down several items.
    ///  -          Mouse over a different item
    ///  -          Notice how two things are never painted hot at the same time, and how the selection changes
    ///  -          from the keyboard selected item to the one selected with the mouse. In  this case the selection
    ///  -          should move with the mouse selection.
    ///  -          Notice how if you hit enter when the mouse is over it, it executes the item. That's selection.
    ///  Scenario 2: Put focus into a combo box, then mouse over a different item
    ///  -          Notice how all the other items you mouse over do not change the way they are painted, if you
    ///  -          hit enter, that goes to the ComboBox, rather than executing the current item.
    ///
    ///  At first look "MouseIsOver" or "Hot" seems to be the thing people want, but its almost never the desired
    ///  behavior. A unified selection model is simpler and seems to meet the scenarios.
    /// </summary>
    [Browsable(false)]
    [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
    public virtual bool Selected
        => CanSelect && !DesignMode && (_state[s_stateSelected] ||
            (ParentInternal is not null && ParentInternal.IsSelectionSuspended &&
             ParentInternal.LastMouseDownedItem == this));
 
    /// <summary>
    ///  Occurs when value of the <see cref="Selected" /> property changes.
    /// </summary>
    /// <remarks>
    ///  <para>
    ///   This event is raised when item is selected by mouse, keyboard or programmatically.
    ///   .NET Core 3.1 had removed MainMenu and MenuItem controls. ToolStripMenuItem.SelectedChanged event
    ///   is recommended as a replacement for MenuItem.Select event.
    ///  </para>
    /// </remarks>
    [SRDescription(nameof(SR.ToolStripItemSelectedChangedDescr))]
    public event EventHandler? SelectedChanged
    {
        add => Events.AddHandler(s_selectedChangedEvent, value);
        remove => Events.RemoveHandler(s_selectedChangedEvent, value);
    }
 
    /// <summary>
    ///  Raises the <see cref="SelectedChanged" /> event. This method will be called when selected
    ///  <see cref="ToolStripItem" /> changes. Call base.OnSelectedChanged to send this event to any registered
    ///  event listeners.
    /// </summary>
    protected virtual void OnSelectedChanged(EventArgs e) => RaiseEvent(s_selectedChangedEvent, e);
 
    protected internal virtual bool ShowKeyboardCues
        => DesignMode || ToolStripManager.ShowMenuFocusCues;
 
    /// <summary>
    ///  The size of the item
    /// </summary>
    [SRCategory(nameof(SR.CatLayout))]
    [Localizable(true)]
    [SRDescription(nameof(SR.ToolStripItemSizeDescr))]
    public virtual Size Size
    {
        get => Bounds.Size;
        set
        {
            Rectangle currentBounds = Bounds;
            currentBounds.Size = value;
            SetBounds(currentBounds);
        }
    }
 
    internal bool SupportsRightClick
    {
        get => _state[s_stateSupportsRightClick];
        set => _state[s_stateSupportsRightClick] = value;
    }
 
    internal bool SupportsItemClick
    {
        get => _state[s_stateSupportsItemClick];
        set => _state[s_stateSupportsItemClick] = value;
    }
 
    internal bool SupportsSpaceKey
    {
        get => _state[s_stateSupportsSpaceKey];
        set => _state[s_stateSupportsSpaceKey] = value;
    }
 
    internal bool SupportsDisabledHotTracking
    {
        get => _state[s_stateSupportsDisabledHotTracking];
        set => _state[s_stateSupportsDisabledHotTracking] = value;
    }
 
    [DefaultValue(null)]
    [SRCategory(nameof(SR.CatData))]
    [Localizable(false)]
    [Bindable(true)]
    [SRDescription(nameof(SR.ToolStripItemTagDescr))]
    [TypeConverter(typeof(StringConverter))]
    public object? Tag
    {
        get => Properties.TryGetObject(s_tagProperty, out object? tag) ? tag : null;
        set => Properties.SetObject(s_tagProperty, value);
    }
 
    /// <summary>
    ///  The text of the item
    /// </summary>
    [DefaultValue("")]
    [SRCategory(nameof(SR.CatAppearance))]
    [Localizable(true)]
    [SRDescription(nameof(SR.ToolStripItemTextDescr))]
    public virtual string? Text
    {
        get => Properties.TryGetObject(s_textProperty, out string? text)
            ? text
            : string.Empty;
        set
        {
            if (value != Text)
            {
                Properties.SetObject(s_textProperty, value);
                OnTextChanged(EventArgs.Empty);
            }
        }
    }
 
    /// <summary>
    ///  Gets or sets the alignment of the text on the label control.
    /// </summary>
    [DefaultValue(ContentAlignment.MiddleCenter)]
    [Localizable(true)]
    [SRCategory(nameof(SR.CatAppearance))]
    [SRDescription(nameof(SR.ToolStripItemTextAlignDescr))]
    public virtual ContentAlignment TextAlign
    {
        get => _textAlign;
        set
        {
            SourceGenerated.EnumValidator.Validate(value);
 
            if (_textAlign != value)
            {
                _textAlign = value;
                InvalidateItemLayout(PropertyNames.TextAlign);
            }
        }
    }
 
    [SRCategory(nameof(SR.CatPropertyChanged))]
    [SRDescription(nameof(SR.ToolStripItemOnTextChangedDescr))]
    public event EventHandler? TextChanged
    {
        add => Events.AddHandler(s_textChangedEvent, value);
        remove => Events.RemoveHandler(s_textChangedEvent, value);
    }
 
    [SRDescription(nameof(SR.ToolStripTextDirectionDescr))]
    [SRCategory(nameof(SR.CatAppearance))]
    public virtual ToolStripTextDirection TextDirection
    {
        get
        {
            ToolStripTextDirection textDirection = ToolStripTextDirection.Inherit;
            if (Properties.TryGetObject(s_textDirectionProperty, out ToolStripTextDirection direction))
            {
                textDirection = direction;
            }
 
            if (textDirection == ToolStripTextDirection.Inherit)
            {
                if (ParentInternal is not null)
                {
                    // in the case we're on a ToolStripOverflow
                    textDirection = ParentInternal.TextDirection;
                }
                else
                {
                    textDirection = (Owner is null) ? ToolStripTextDirection.Horizontal : Owner.TextDirection;
                }
            }
 
            return textDirection;
        }
        set
        {
            SourceGenerated.EnumValidator.Validate(value);
 
            Properties.SetObject(s_textDirectionProperty, value);
            InvalidateItemLayout("TextDirection");
        }
    }
 
    [DefaultValue(TextImageRelation.ImageBeforeText)]
    [Localizable(true)]
    [SRDescription(nameof(SR.ToolStripItemTextImageRelationDescr))]
    [SRCategory(nameof(SR.CatAppearance))]
    public TextImageRelation TextImageRelation
    {
        get => _textImageRelation;
        set
        {
            SourceGenerated.EnumValidator.Validate(value);
 
            if (value != TextImageRelation)
            {
                _textImageRelation = value;
                InvalidateItemLayout(PropertyNames.TextImageRelation);
            }
        }
    }
 
    /// <summary>
    ///  !!!!This property ONLY works when toolStrip.ShowItemToolTips = true!!!!
    ///  if AutoToolTip is set to true we return the Text as the ToolTipText.
    /// </summary>
    [SRDescription(nameof(SR.ToolStripItemToolTipTextDescr))]
    [SRCategory(nameof(SR.CatBehavior))]
    [Editor($"System.ComponentModel.Design.MultilineStringEditor, {AssemblyRef.SystemDesign}", typeof(UITypeEditor))]
    [Localizable(true)]
    public string? ToolTipText
    {
        get
        {
            if (AutoToolTip && string.IsNullOrEmpty(_toolTipText))
            {
                string? toolText = Text;
                if (WindowsFormsUtils.ContainsMnemonic(toolText))
                {
                    // this shouldn't be called a lot so we can take the perf hit here.
                    toolText = string.Join(string.Empty, toolText.Split('&'));
                }
 
                return toolText;
            }
 
            return _toolTipText;
        }
        set => _toolTipText = value;
    }
 
    /// <summary>
    ///  Whether or not the item is visible
    /// </summary>
    [SRCategory(nameof(SR.CatBehavior))]
    [Localizable(true)]
    [SRDescription(nameof(SR.ToolStripItemVisibleDescr))]
    public bool Visible
    {
        get => (ParentInternal is not null) && (ParentInternal.Visible) && Available;
        set => SetVisibleCore(value);
    }
 
    [SRCategory(nameof(SR.CatPropertyChanged))]
    [SRDescription(nameof(SR.ToolStripItemOnVisibleChangedDescr))]
    public event EventHandler? VisibleChanged
    {
        add => Events.AddHandler(s_visibleChangedEvent, value);
        remove => Events.RemoveHandler(s_visibleChangedEvent, value);
    }
 
    /// <summary>
    ///  The width of this ToolStripItem.
    /// </summary>
    [SRCategory(nameof(SR.CatLayout))]
    [Browsable(false)]
    [EditorBrowsable(EditorBrowsableState.Always)]
    [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
    public int Width
    {
        get => Bounds.Width;
        set
        {
            Rectangle currentBounds = Bounds;
            SetBounds(currentBounds.X, currentBounds.Y, value, currentBounds.Height);
        }
    }
 
    internal void AccessibilityNotifyClients(AccessibleEvents accEvent)
    {
        if (ParentInternal is not null)
        {
            int index = ParentInternal.DisplayedItems.IndexOf(this);
            ParentInternal.AccessibilityNotifyClients(accEvent, index);
        }
    }
 
    private void Animate()
        => Animate(!DesignMode && Visible && Enabled && ParentInternal is not null);
 
    private void StopAnimate() => Animate(false);
 
    private void Animate(bool animate)
    {
        if (animate == _state[s_stateCurrentlyAnimatingImage])
        {
            return;
        }
 
        Image? image = Image;
        if (image is null)
        {
            return;
        }
 
        if (animate)
        {
            ImageAnimator.Animate(image, OnAnimationFrameChanged);
        }
        else
        {
            ImageAnimator.StopAnimate(image, OnAnimationFrameChanged);
        }
 
        _state[s_stateCurrentlyAnimatingImage] = animate;
    }
 
    internal bool BeginDragForItemReorder()
    {
        if (Control.ModifierKeys == Keys.Alt)
        {
            if (ParentInternal is not null && ParentInternal.Items.Contains(this) && ParentInternal.AllowItemReorder)
            {
                // we only drag
                ToolStripItem item = this;
                DoDragDrop(item, DragDropEffects.Move);
                return true;
            }
        }
 
        return false;
    }
 
    /// <summary>
    ///  Constructs the new instance of the accessibility object for this ToolStripItem. Subclasses
    ///  should not call base.CreateAccessibilityObject.
    /// </summary>
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    protected virtual AccessibleObject CreateAccessibilityInstance()
        => new ToolStripItemAccessibleObject(this);
 
    /// <summary>
    ///  Creates an instance of the object that defines how image and text
    ///  gets laid out in the ToolStripItem
    /// </summary>
    private protected virtual ToolStripItemInternalLayout CreateInternalLayout()
        => new(this);
 
    /// <summary>
    ///  Disposes this ToolStrip item.
    /// </summary>
    protected override void Dispose(bool disposing)
    {
        if (disposing)
        {
            _state[s_stateDisposing] = true;
 
            if (Owner is not null)
            {
                StopAnimate();
                Debug.Assert(Owner.Items.Contains(this), "How can there be a owner and not be in the collection?");
                Owner.Items.Remove(this);
                _toolStripItemInternalLayout = null;
                _state[s_stateDisposed] = true;
            }
        }
 
        base.Dispose(disposing);
 
        if (disposing)
        {
            // need to call base() first since the Component.Dispose(_) is listened to by the ComponentChangeService
            // which Serializes the object in Undo-Redo transactions.
            Properties.SetObject(s_mirroredImageProperty, null);
            Properties.SetObject(s_imageProperty, null);
            _state[s_stateDisposing] = false;
        }
    }
 
    internal static long DoubleClickTicks
    {
        // (DoubleClickTime in ms) * 1,000,000 ns/1ms * 1 Tick / 100ns = XXX in Ticks.
        // Therefore: (DoubleClickTime) * 1,000,000/100 = xxx in Ticks.
        get => SystemInformation.DoubleClickTime * 10000;
    }
 
    /// <summary>
    ///  Begins a drag operation. The allowedEffects determine which
    ///  drag operations can occur. If the drag operation needs to interop
    ///  with applications in another process, data should either be
    ///  a base managed class (String, Bitmap, or Metafile) or some Object
    ///  that implements System.Runtime.Serialization.ISerializable. data can also be any Object that
    ///  implements System.Windows.Forms.IDataObject.
    /// </summary>
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    public DragDropEffects DoDragDrop(object data, DragDropEffects allowedEffects)
    {
        return DoDragDrop(data, allowedEffects, dragImage: null, cursorOffset: default, useDefaultDragImage: false);
    }
 
    /// <summary>
    ///  Begins a drag operation. The <paramref name="allowedEffects"/> determine which drag operations can occur.
    ///  If the drag operation needs to interop with applications in another process, <paramref name="data"/> should
    ///  either be a base managed class (<see cref="string"/>, <see cref="Bitmap"/>, or <see cref="Metafile"/>)
    ///  or some <see cref="object"/> that implements <see cref="Runtime.Serialization.ISerializable"/>.
    ///  <paramref name="data"/> can also be any <see cref="object"/> that implements <see cref="IDataObject"/>.
    ///  <paramref name="dragImage"/> is the bitmap that will be displayed during the  drag operation and
    ///  <paramref name="cursorOffset"/> specifies the location of the cursor within <paramref name="dragImage"/>,
    ///  which is an offset from the upper-left corner. Specify <see langword="true"/> for
    ///  <paramref name="useDefaultDragImage"/> to use a layered window drag image with a
    ///  size of 96x96; otherwise <see langword="false"/>. Note the outer edges of <paramref name="dragImage"/>
    ///  are blended out if the image width or height exceeds 300 pixels.
    /// </summary>
    /// <returns>
    ///  A value from the <see cref="DragDropEffects"/> enumeration that represents the final effect that was performed
    ///  during the drag-and-drop operation.
    /// </returns>
    /// <remarks>
    ///  <para>
    ///   Because <see cref="DoDragDrop(object, DragDropEffects, Bitmap, Point, bool)"/> always performs the
    ///   RGB multiplication step in calculating the alpha value, you should always pass a <see cref="Bitmap"/>
    ///   without pre-multiplied alpha blending. Note that no error will result from passing a <see cref="Bitmap"/>
    ///   with pre-multiplied alpha blending, but this method will multiply it again, doubling the resulting alpha value.
    ///  </para>
    /// </remarks>
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    public unsafe DragDropEffects DoDragDrop(object data, DragDropEffects allowedEffects, Bitmap? dragImage, Point cursorOffset, bool useDefaultDragImage)
    {
        if (data is not IComDataObject dataObject)
        {
            DataObject? iwdata = null;
            if (data is IDataObject idataObject)
            {
                iwdata = new DataObject(idataObject);
            }
            else if (data is ToolStripItem)
            {
                // it seems that the DataObject does string comparison
                // on the type, so you can't ask for GetDataPresent of
                // a base type (e.g. ToolStripItem) when you are really
                // looking at a ToolStripButton. The alternative is
                // to set the format string expressly to a string matching
                // the type of ToolStripItem
                iwdata = new DataObject();
                iwdata.SetData(typeof(ToolStripItem).ToString(), data);
            }
            else
            {
                iwdata = new DataObject();
                iwdata.SetData(data);
            }
 
            dataObject = iwdata;
        }
 
        DROPEFFECT finalEffect;
 
        try
        {
            using var dropSource = ComHelpers.GetComScope<IDropSource>(CreateDropSource(dataObject, dragImage, cursorOffset, useDefaultDragImage));
            using var dataObjectScope = ComHelpers.GetComScope<Com.IDataObject>(dataObject);
            if (PInvoke.DoDragDrop(dataObjectScope, dropSource, (DROPEFFECT)(uint)allowedEffects, out finalEffect).Failed)
            {
                return DragDropEffects.None;
            }
        }
        finally
        {
            if (DragDropHelper.IsInDragLoop(dataObject))
            {
                DragDropHelper.SetInDragLoop(dataObject, inDragLoop: false);
            }
        }
 
        return (DragDropEffects)finalEffect;
    }
 
    /// <summary>
    ///  This represents what we're actually going to drag. If the parent has set AllowItemReorder to true,
    ///  then the item should call back on the private OnQueryContinueDrag/OnGiveFeedback that is implemented
    ///  in the parent ToolStrip.
    ///
    ///  Else if the parent does not support reordering of items (Parent.AllowItemReorder = false) -
    ///  then call back on the ToolStripItem's OnQueryContinueDrag/OnGiveFeedback methods.
    /// </summary>
    internal IDropSource.Interface CreateDropSource(IComDataObject dataObject, Bitmap? dragImage, Point cursorOffset, bool useDefaultDragImage)
        => ParentInternal is not null && ParentInternal.AllowItemReorder && ParentInternal.ItemReorderDropSource is not null
            ? new DropSource(ParentInternal.ItemReorderDropSource, dataObject, dragImage, cursorOffset, useDefaultDragImage)
            : new DropSource(this, dataObject, dragImage, cursorOffset, useDefaultDragImage);
 
    internal void FireEvent(ToolStripItemEventType met)
        => FireEvent(EventArgs.Empty, met);
 
    internal void FireEvent(EventArgs e, ToolStripItemEventType met)
    {
        switch (met)
        {
            case ToolStripItemEventType.LocationChanged:
                OnLocationChanged(e);
                break;
            case ToolStripItemEventType.Paint:
                HandlePaint((PaintEventArgs)e);
                break;
            case ToolStripItemEventType.MouseHover:
                // disabled toolstrip items should show tooltips.
                // we won't raise mouse events though.
                if (!Enabled && ParentInternal is not null && !string.IsNullOrEmpty(ToolTipText))
                {
                    ParentInternal.UpdateToolTip(this);
                }
                else
                {
                    FireEventInteractive(e, met);
                }
 
                break;
            case ToolStripItemEventType.MouseEnter:
                HandleMouseEnter(e);
                break;
            case ToolStripItemEventType.MouseLeave:
                // disabled toolstrip items should also clear tooltips.
                // disabled toolstrip items should also handle leave.
                // we won't raise mouse events though.
                if (!Enabled && ParentInternal is not null)
                {
                    ParentInternal.UpdateToolTip(item: null);
                    HandleLeave();
                }
                else
                {
                    HandleMouseLeave(e);
                }
 
                break;
            case ToolStripItemEventType.MouseMove:
                // Disabled items typically don't get mouse move
                // but they should be allowed to re-order if the ALT key is pressed
                if (!Enabled && ParentInternal is not null)
                {
                    BeginDragForItemReorder();
                }
                else
                {
                    FireEventInteractive(e, met);
                }
 
                break;
            default:
                FireEventInteractive(e, met);
                break;
        }
    }
 
    internal void FireEventInteractive(EventArgs e, ToolStripItemEventType met)
    {
        if (Enabled)
        {
            switch (met)
            {
                case ToolStripItemEventType.MouseMove:
                    HandleMouseMove((MouseEventArgs)e);
                    break;
                case ToolStripItemEventType.MouseHover:
                    HandleMouseHover(e);
                    break;
                case ToolStripItemEventType.MouseUp:
                    HandleMouseUp((MouseEventArgs)e);
                    break;
                case ToolStripItemEventType.MouseDown:
                    HandleMouseDown((MouseEventArgs)e);
                    break;
                case ToolStripItemEventType.Click:
                    HandleClick(e);
                    break;
                case ToolStripItemEventType.DoubleClick:
                    HandleDoubleClick(e);
                    break;
                default:
                    Debug.Assert(false, "Invalid event type.");
                    break;
            }
        }
    }
 
    private Font? GetOwnerFont() => Owner?.Font;
 
    /// <summary>
    ///  We don't want a public settable property and usually owner will work
    ///  except for things like the overflow button
    /// </summary>
    public ToolStrip? GetCurrentParent() => Parent;
 
    internal ToolStripDropDown? GetCurrentParentDropDown()
    {
        if (ParentInternal is not null)
        {
            return ParentInternal as ToolStripDropDown;
        }
 
        return Owner as ToolStripDropDown;
    }
 
    public virtual Size GetPreferredSize(Size constrainingSize)
    {
        // Switch Size.Empty to maximum possible values
        constrainingSize = LayoutUtils.ConvertZeroToUnbounded(constrainingSize);
        return InternalLayout.GetPreferredSize(constrainingSize - Padding.Size) + Padding.Size;
    }
 
    internal Size GetTextSize()
    {
        if (string.IsNullOrEmpty(Text))
        {
            return Size.Empty;
        }
        else if (_cachedTextSize == Size.Empty)
        {
            _cachedTextSize = TextRenderer.MeasureText(Text, Font);
        }
 
        return _cachedTextSize;
    }
 
    /// <summary>
    ///  Invalidates the ToolStripItem
    /// </summary>
    public void Invalidate() => ParentInternal?.Invalidate(Bounds, true);
 
    /// <summary>
    ///  invalidates a rectangle within the ToolStripItem's bounds
    /// </summary>
    public void Invalidate(Rectangle r)
    {
        // the only value add to this over calling invalidate on the ToolStrip is that
        // you can specify the rectangle with coordinates relative to the upper left hand
        // corner of the ToolStripItem.
        Point rectangleLocation = TranslatePoint(r.Location, ToolStripPointType.ToolStripItemCoords, ToolStripPointType.ToolStripCoords);
        ParentInternal?.Invalidate(new Rectangle(rectangleLocation, r.Size), true);
    }
 
    internal void InvalidateItemLayout(string affectedProperty)
    {
        _toolStripItemInternalLayout = null;
 
        if (Owner is not null)
        {
            LayoutTransaction.DoLayout(Owner, this, affectedProperty);
            // DoLayout may cause the ToolStrip size to change. If the ToolStrip is an MdiControlStrip, the
            // active Mdi child is maximized, and DoLayout causes the size to change then Form.MdiControlStrip
            // (Owner) will be disposed and replaced with a new one. This means the current ToolStripItem will
            // also be disposed and have no owner on the next line. See https://github.com/dotnet/winforms/issues/6535
            Owner?.Invalidate();
        }
    }
 
    internal void InvalidateImageListImage()
    {
        // invalidate the cache.
        if (ImageIndexer.ActualIndex >= 0)
        {
            Properties.SetObject(s_imageProperty, null);
            InvalidateItemLayout(PropertyNames.Image);
        }
    }
 
    internal void InvokePaint() => ParentInternal?.InvokePaintItem(this);
 
    protected internal virtual bool IsInputKey(Keys keyData) => false;
 
    protected internal virtual bool IsInputChar(char charCode) => false;
 
    private void HandleClick(EventArgs e)
    {
        try
        {
            if (!DesignMode)
            {
                _state[s_statePressed] = true;
            }
 
            // Force painting w/o using message loop here because it may be quite a long
            // time before it gets pumped again.
            InvokePaint();
 
            if (SupportsItemClick && Owner is not null)
            {
                Owner.HandleItemClick(this);
            }
 
            OnClick(e);
 
            if (SupportsItemClick && Owner is not null)
            {
                Owner.HandleItemClicked(this);
            }
        }
        finally
        {
            _state[s_statePressed] = false;
        }
 
        // when we get around to it, paint unpressed.
        Invalidate();
    }
 
    private void HandleDoubleClick(EventArgs e) => OnDoubleClick(e);
 
    private void HandlePaint(PaintEventArgs e)
    {
        Animate();
        ImageAnimator.UpdateFrames(Image);
 
        OnPaint(e);
        RaisePaintEvent(s_paintEvent, e);
    }
 
    private void HandleMouseEnter(EventArgs e)
    {
        if (!DesignMode)
        {
            if (ParentInternal is not null
                 && ParentInternal.CanHotTrack
                 && ParentInternal.ShouldSelectItem())
            {
                if (Enabled)
                {
                    // calling select can dismiss a child dropdown which would break auto-expansion.
                    // save off auto expand and restore it.
                    bool autoExpand = ParentInternal.MenuAutoExpand;
 
                    if (ParentInternal.LastMouseDownedItem == this)
                    {
                        // Same as Control.MouseButtons == MouseButtons.Left, but slightly more efficient.
                        if (PInvoke.GetKeyState((int)Keys.LButton) < 0)
                        {
                            Push(true);
                        }
                    }
 
                    Select();
                    ParentInternal.MenuAutoExpand = autoExpand;
                }
                else if (SupportsDisabledHotTracking)
                {
                    Select();
                }
            }
        }
 
        KeyboardToolTipStateMachine.Instance.NotifyAboutMouseEnter(this);
 
        if (Enabled)
        {
            OnMouseEnter(e);
            RaiseEvent(s_mouseEnterEvent, e);
        }
    }
 
    private void HandleMouseMove(MouseEventArgs mea)
    {
        if (Enabled && CanSelect && !Selected)
        {
            if (ParentInternal is not null
                 && ParentInternal.CanHotTrack
                 && ParentInternal.ShouldSelectItem())
            {
                // this is the case where we got a mouse enter, but ShouldSelectItem
                // returned false.
                // typically occurs when a window first opens - we get a mouse enter on the item
                // the cursor is hovering over - but we don't actually want to change selection to it.
                Select();
            }
        }
 
        OnMouseMove(mea);
        RaiseMouseEvent(s_mouseMoveEvent, mea);
    }
 
    private void HandleMouseHover(EventArgs e)
    {
        OnMouseHover(e);
        RaiseEvent(s_mouseHoverEvent, e);
    }
 
    private void HandleLeave()
    {
        if (_state[s_stateMouseDownAndNoDrag] || _state[s_statePressed] || _state[s_stateSelected])
        {
            _state[s_stateMouseDownAndNoDrag | s_statePressed | s_stateSelected] = false;
 
            KeyboardToolTipStateMachine.Instance.NotifyAboutLostFocus(this);
 
            Invalidate();
        }
    }
 
    private void HandleMouseLeave(EventArgs e)
    {
        HandleLeave();
        if (Enabled)
        {
            OnMouseLeave(e);
            RaiseEvent(s_mouseLeaveEvent, e);
        }
    }
 
    private void HandleMouseDown(MouseEventArgs e)
    {
        _state[s_stateMouseDownAndNoDrag] = !BeginDragForItemReorder();
        if (_state[s_stateMouseDownAndNoDrag])
        {
            if (e.Button == MouseButtons.Left)
            {
                Push(true);
            }
 
            OnMouseDown(e);
            RaiseMouseEvent(s_mouseDownEvent, e);
        }
    }
 
    private void HandleMouseUp(MouseEventArgs e)
    {
        bool fireMouseUp = (ParentInternal?.LastMouseDownedItem == this);
 
        if (!fireMouseUp && !MouseDownAndUpMustBeInSameItem)
        {
            // in the case of menus, you can mouse down on one item and mouse up
            // on another. We do need to be careful
            // that the mouse has actually moved from when a dropdown has been opened -
            // otherwise we may accidentally click what's underneath the mouse at the time
            // the dropdown is opened.
            fireMouseUp = ParentInternal is not null && ParentInternal.ShouldSelectItem();
        }
 
        if (_state[s_stateMouseDownAndNoDrag] || fireMouseUp)
        {
            Push(false);
 
            if (e.Button == MouseButtons.Left || (e.Button == MouseButtons.Right && _state[s_stateSupportsRightClick]))
            {
                bool shouldFireDoubleClick = false;
                if (DoubleClickEnabled)
                {
                    long newTime = DateTime.Now.Ticks;
                    long deltaTicks = newTime - _lastClickTime;
                    _lastClickTime = newTime;
                    // use >= for cases where the delta is so fast DateTime cannot pick up the delta.
                    Debug.Assert(deltaTicks >= 0, "why are deltaticks less than zero? thats some mighty fast clicking");
                    // if we've seen a mouse up less than the double click time ago, we should fire.
                    if (deltaTicks >= 0 && deltaTicks < DoubleClickTicks)
                    {
                        shouldFireDoubleClick = true;
                    }
                }
 
                if (shouldFireDoubleClick)
                {
                    HandleDoubleClick(EventArgs.Empty);
                    // If we actually fired DoubleClick - reset the lastClickTime.
                    _lastClickTime = 0;
                }
                else
                {
                    HandleClick(EventArgs.Empty);
                }
            }
 
            OnMouseUp(e);
            RaiseMouseEvent(s_mouseUpEvent, e);
        }
    }
 
    internal virtual void OnAccessibleDescriptionChanged(EventArgs e)
    {
    }
 
    internal virtual void OnAccessibleNameChanged(EventArgs e)
    {
    }
 
    internal virtual void OnAccessibleDefaultActionDescriptionChanged(EventArgs e)
    {
    }
 
    internal virtual void OnAccessibleRoleChanged(EventArgs e)
    {
    }
 
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    protected virtual void OnBackColorChanged(EventArgs e)
    {
        Invalidate();
        RaiseEvent(s_backColorChangedEvent, e);
    }
 
    protected virtual void OnBoundsChanged()
    {
        LayoutTransaction.DoLayout(ParentInternal, this, PropertyNames.Bounds);
        InternalLayout.PerformLayout();
    }
 
    protected virtual void OnClick(EventArgs e)
    {
        RaiseEvent(s_clickEvent, e);
        OnRequestCommandExecute(e);
    }
 
    protected internal virtual void OnLayout(LayoutEventArgs e)
    {
    }
 
    void IDropTarget.OnDragEnter(DragEventArgs dragEvent) => OnDragEnter(dragEvent);
 
    void IDropTarget.OnDragOver(DragEventArgs dragEvent) => OnDragOver(dragEvent);
 
    void IDropTarget.OnDragLeave(EventArgs e) => OnDragLeave(e);
 
    void IDropTarget.OnDragDrop(DragEventArgs dragEvent) => OnDragDrop(dragEvent);
 
    void ISupportOleDropSource.OnGiveFeedback(GiveFeedbackEventArgs giveFeedbackEventArgs)
        => OnGiveFeedback(giveFeedbackEventArgs);
 
    void ISupportOleDropSource.OnQueryContinueDrag(QueryContinueDragEventArgs queryContinueDragEventArgs)
        => OnQueryContinueDrag(queryContinueDragEventArgs);
 
    private void OnAnimationFrameChanged(object? o, EventArgs e)
    {
        ToolStrip? parent = ParentInternal;
        if (parent is not null)
        {
            if (parent.Disposing || parent.IsDisposed)
            {
                return;
            }
 
            if (parent.IsHandleCreated && parent.InvokeRequired)
            {
                parent.BeginInvoke(new EventHandler(OnAnimationFrameChanged), [o, e]);
                return;
            }
 
            Invalidate();
        }
    }
 
    protected virtual void OnAvailableChanged(EventArgs e) => RaiseEvent(s_availableChangedEvent, e);
 
    /// <summary>
    ///  Raises the <see cref="CommandChanged"/> event.
    /// </summary>
    /// <param name="e">An empty <see cref="EventArgs"/> instance.</param>
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    protected virtual void OnCommandChanged(EventArgs e)
        => RaiseEvent(s_commandChangedEvent, e);
 
    /// <summary>
    ///  Raises the <see cref="CommandCanExecuteChanged"/> event.
    /// </summary>
    /// <param name="e">An empty <see cref="EventArgs"/> instance.</param>
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    protected virtual void OnCommandCanExecuteChanged(EventArgs e)
        => ((EventHandler?)Events[s_commandCanExecuteChangedEvent])?.Invoke(this, e);
 
    /// <summary>
    ///  Raises the <see cref="CommandParameterChanged"/> event.
    /// </summary>
    /// <param name="e">An empty <see cref="EventArgs"/> instance.</param>
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    protected virtual void OnCommandParameterChanged(EventArgs e) => RaiseEvent(s_commandParameterChangedEvent, e);
 
    /// <summary>
    ///  Called in the context of <see cref="OnClick(EventArgs)"/> to invoke
    ///  <see cref="Input.ICommand.Execute(object?)"/> if the context allows.
    /// </summary>
    /// <param name="e">An empty <see cref="EventArgs"/> instance.</param>
    protected virtual void OnRequestCommandExecute(EventArgs e)
        => ICommandBindingTargetProvider.RequestCommandExecute(this);
 
    // Called by the CommandProviderManager's command handling logic.
    void ICommandBindingTargetProvider.RaiseCommandChanged(EventArgs e)
        => OnCommandChanged(e);
 
    // Called by the CommandProviderManager's command handling logic.
    void ICommandBindingTargetProvider.RaiseCommandCanExecuteChanged(EventArgs e)
        => OnCommandCanExecuteChanged(e);
 
    /// <summary>
    ///  Raises the <see cref="DragEnter"/> event.
    ///  Inheriting classes should override this method to handle this event.
    ///  Call base.OnEnter to send this event to any registered event listeners.
    /// </summary>
    /// <summary>
    ///  Inheriting classes should override this method to handle this event.
    ///  Call base.OnDragEnter to send this event to any registered event listeners.
    /// </summary>
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    protected virtual void OnDragEnter(DragEventArgs dragEvent)
        => RaiseDragEvent(s_dragEnterEvent, dragEvent);
 
    /// <summary>
    ///  Inheriting classes should override this method to handle this event.
    ///  Call base.OnDragOver to send this event to any registered event listeners.
    /// </summary>
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    protected virtual void OnDragOver(DragEventArgs dragEvent)
        => RaiseDragEvent(s_dragOverEvent, dragEvent);
 
    /// <summary>
    ///  Inheriting classes should override this method to handle this event.
    ///  Call base.OnDragLeave to send this event to any registered event listeners.
    /// </summary>
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    protected virtual void OnDragLeave(EventArgs e)
        => RaiseEvent(s_dragLeaveEvent, e);
 
    /// <summary>
    ///  Inheriting classes should override this method to handle this event.
    ///  Call base.OnDragDrop to send this event to any registered event listeners.
    /// </summary>
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    protected virtual void OnDragDrop(DragEventArgs dragEvent)
        => RaiseDragEvent(s_dragDropEvent, dragEvent);
 
    /// <summary>
    ///  Inheriting classes should override this method to handle this event.
    /// </summary>
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    protected virtual void OnDisplayStyleChanged(EventArgs e)
        => RaiseEvent(s_displayStyleChangedEvent, e);
 
    /// <summary>
    ///  Inheriting classes should override this method to handle this event.
    ///  Call base.OnGiveFeedback to send this event to any registered event listeners.
    /// </summary>
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    protected virtual void OnGiveFeedback(GiveFeedbackEventArgs giveFeedbackEvent)
        => ((GiveFeedbackEventHandler?)Events[s_giveFeedbackEvent])?.Invoke(this, giveFeedbackEvent);
 
    internal virtual void OnImageScalingSizeChanged(EventArgs e)
    {
    }
 
    /// <summary>
    ///  Inheriting classes should override this method to handle this event.
    ///  Call base.OnQueryContinueDrag to send this event to any registered event listeners.
    /// </summary>
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    protected virtual void OnQueryContinueDrag(QueryContinueDragEventArgs queryContinueDragEvent)
        => RaiseQueryContinueDragEvent(s_queryContinueDragEvent, queryContinueDragEvent);
 
    /// <summary>
    ///  Inheriting classes should override this method to handle this event.
    /// </summary>
    protected virtual void OnDoubleClick(EventArgs e) => RaiseEvent(s_doubleClickEvent, e);
 
    /// <summary>
    ///  Inheriting classes should override this method to handle this event.
    /// </summary>
    protected virtual void OnEnabledChanged(EventArgs e)
    {
        RaiseEvent(s_enabledChangedEvent, e);
        Animate();
    }
 
    internal void OnInternalEnabledChanged(EventArgs e) => RaiseEvent(s_internalEnabledChangedEvent, e);
 
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    protected virtual void OnForeColorChanged(EventArgs e)
    {
        Invalidate();
        RaiseEvent(s_foreColorChangedEvent, e);
    }
 
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    protected virtual void OnFontChanged(EventArgs e)
    {
        _cachedTextSize = Size.Empty;
        // PERF - only invalidate if we actually care about the font
        if ((DisplayStyle & ToolStripItemDisplayStyle.Text) == ToolStripItemDisplayStyle.Text)
        {
            InvalidateItemLayout(PropertyNames.Font);
        }
        else
        {
            _toolStripItemInternalLayout = null;
        }
 
        RaiseEvent(s_fontChangedEvent, e);
    }
 
    protected virtual void OnLocationChanged(EventArgs e) => RaiseEvent(s_locationChangedEvent, e);
 
    /// <summary>
    ///  Inheriting classes should override this method to handle this event.
    /// </summary>
    protected virtual void OnMouseEnter(EventArgs e)
    {
    }
 
    /// <summary>
    ///  Inheriting classes should override this method to handle this event.
    /// </summary>
    protected virtual void OnMouseMove(MouseEventArgs mea)
    {
    }
 
    /// <summary>
    ///  Inheriting classes should override this method to handle this event.
    /// </summary>
    protected virtual void OnMouseHover(EventArgs e)
    {
        if (ParentInternal is not null && !string.IsNullOrEmpty(ToolTipText))
        {
            ParentInternal.UpdateToolTip(this);
        }
    }
 
    /// <summary>
    ///  Inheriting classes should override this method to handle this event.
    /// </summary>
    protected virtual void OnMouseLeave(EventArgs e) => ParentInternal?.UpdateToolTip(item: null);
 
    /// <summary>
    ///  Inheriting classes should override this method to handle this event.
    /// </summary>
    protected virtual void OnMouseDown(MouseEventArgs e)
    {
    }
 
    /// <summary>
    ///  Inheriting classes should override this method to handle this event.
    /// </summary>
    protected virtual void OnMouseUp(MouseEventArgs e)
    {
    }
 
    /// <summary>
    ///  Inheriting classes should override this method to handle this event.
    /// </summary>
    protected virtual void OnPaint(PaintEventArgs e)
    {
    }
 
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    protected virtual void OnParentBackColorChanged(EventArgs e)
    {
        Color backColor = Properties.GetValueOrDefault<Color>(s_backColorProperty);
        if (backColor.IsEmpty)
        {
            OnBackColorChanged(e);
        }
    }
 
    /// <summary>
    ///  Inheriting classes should override this method to handle this event.
    /// </summary>
    protected virtual void OnParentChanged(ToolStrip? oldParent, ToolStrip? newParent)
    {
        SetAmbientMargin();
        if ((oldParent is not null) && (oldParent.DropTargetManager is not null))
        {
            oldParent.DropTargetManager.EnsureUnRegistered();
        }
 
        if (AllowDrop && (newParent is not null))
        {
            EnsureParentDropTargetRegistered();
        }
 
        Animate();
    }
 
    /// <summary>
    ///  Occurs when this.Parent.Enabled changes.
    /// </summary>
    protected internal virtual void OnParentEnabledChanged(EventArgs e)
        => OnEnabledChanged(EventArgs.Empty);
 
    /// <summary>
    ///  Occurs when the font property has changed on the parent - used to notify inheritors of the font property that
    ///  the font has changed
    /// </summary>
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    protected internal virtual void OnOwnerFontChanged(EventArgs e)
    {
        if (!TryGetExplicitlySetFont(out _))
        {
            OnFontChanged(e);
        }
    }
 
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    protected virtual void OnParentForeColorChanged(EventArgs e)
    {
        Color foreColor = Properties.GetValueOrDefault<Color>(s_foreColorProperty);
        if (foreColor.IsEmpty)
        {
            OnForeColorChanged(e);
        }
    }
 
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    protected internal virtual void OnParentRightToLeftChanged(EventArgs e)
    {
        if (!Properties.TryGetValue(s_rightToLeftProperty, out RightToLeft rightToLeft) || rightToLeft == RightToLeft.Inherit)
        {
            OnRightToLeftChanged(e);
        }
    }
 
    /// <summary>
    ///  Occurs when the owner of an item changes.
    /// </summary>
    protected virtual void OnOwnerChanged(EventArgs e)
    {
        RaiseEvent(s_ownerChangedEvent, e);
        SetAmbientMargin();
        if (Owner is not null)
        {
            // check if we need to fire OnRightToLeftChanged
            if (!Properties.TryGetValue(s_rightToLeftProperty, out RightToLeft rightToLeft))
            {
                rightToLeft = RightToLeft.Inherit;
            }
 
            if (rightToLeft == RightToLeft.Inherit && RightToLeft != DefaultRightToLeft)
            {
                OnRightToLeftChanged(EventArgs.Empty);
            }
        }
    }
 
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    internal void OnOwnerTextDirectionChanged()
    {
        ToolStripTextDirection textDirection = ToolStripTextDirection.Inherit;
        if (Properties.TryGetObject(s_textDirectionProperty, out ToolStripTextDirection direction))
        {
            textDirection = direction;
        }
 
        if (textDirection == ToolStripTextDirection.Inherit)
        {
            InvalidateItemLayout("TextDirection");
        }
    }
 
    /// <summary>
    ///  Inheriting classes should override this method to handle this event.
    /// </summary>
    protected virtual void OnRightToLeftChanged(EventArgs e)
    {
        InvalidateItemLayout(PropertyNames.RightToLeft);
        RaiseEvent(s_rightToLeftChangedEvent, e);
    }
 
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    protected virtual void OnTextChanged(EventArgs e)
    {
        _cachedTextSize = Size.Empty;
        // Make sure we clear the cache before we perform the layout.
        InvalidateItemLayout(PropertyNames.Text);
        RaiseEvent(s_textChangedEvent, e);
    }
 
    /// <summary>
    ///  Inheriting classes should override this method to handle this event.
    /// </summary>
    protected virtual void OnVisibleChanged(EventArgs e)
    {
        if (Owner is not null && !(Owner.IsDisposed || Owner.Disposing))
        {
            Owner.OnItemVisibleChanged(new ToolStripItemEventArgs(this), performLayout: true);
        }
 
        RaiseEvent(s_visibleChangedEvent, e);
        Animate();
    }
 
    public void PerformClick()
    {
        if (Enabled && Available)
        {
            FireEvent(ToolStripItemEventType.Click);
        }
    }
 
    /// <summary>
    ///  Pushes the button.
    /// </summary>
    internal void Push(bool push)
    {
        if (!CanSelect || !Enabled || DesignMode)
        {
            return;
        }
 
        if (_state[s_statePressed] != push)
        {
            _state[s_statePressed] = push;
            if (Available)
            {
                Invalidate();
            }
        }
    }
 
    /// <summary>
    ///  See Control.ProcessDialogKey for more info.
    /// </summary>
    protected internal virtual bool ProcessDialogKey(Keys keyData)
    {
        if (keyData == Keys.Enter || (_state[s_stateSupportsSpaceKey] && keyData == Keys.Space))
        {
            FireEvent(ToolStripItemEventType.Click);
            if (ParentInternal is not null && !ParentInternal.IsDropDown && Enabled)
            {
                ParentInternal.RestoreFocusInternal();
            }
 
            return true;
        }
 
        return false;
    }
 
    /// <summary>
    ///  See Control.ProcessCmdKey for more info.
    /// </summary>
    protected internal virtual bool ProcessCmdKey(ref Message m, Keys keyData) => false;
 
    protected internal virtual bool ProcessMnemonic(char charCode)
    {
        // checking IsMnemonic is not necessary - control does this for us.
        FireEvent(ToolStripItemEventType.Click);
        return true;
    }
 
    internal void RaiseCancelEvent(object key, CancelEventArgs e)
        => ((CancelEventHandler?)Events[key])?.Invoke(this, e);
 
    internal void RaiseDragEvent(object key, DragEventArgs e)
        => ((DragEventHandler?)Events[key])?.Invoke(this, e);
 
    internal void RaiseKeyEvent(object key, KeyEventArgs e)
        => ((KeyEventHandler?)Events[key])?.Invoke(this, e);
 
    internal void RaiseKeyPressEvent(object key, KeyPressEventArgs e)
        => ((KeyPressEventHandler?)Events[key])?.Invoke(this, e);
 
    internal void RaiseMouseEvent(object key, MouseEventArgs e)
        => ((MouseEventHandler?)Events[key])?.Invoke(this, e);
 
    internal void RaisePaintEvent(object key, PaintEventArgs e)
        => ((PaintEventHandler?)Events[key])?.Invoke(this, e);
 
    internal void RaiseQueryContinueDragEvent(object key, QueryContinueDragEventArgs e)
        => ((QueryContinueDragEventHandler?)Events[key])?.Invoke(this, e);
 
    internal virtual void ReleaseUiaProvider()
    {
        if (TryGetAccessibilityObject(out AccessibleObject? accessibleObject))
        {
            PInvoke.UiaDisconnectProvider(accessibleObject, skipOSCheck: true);
            Properties.SetObject(s_accessibilityProperty, null);
        }
 
        bool TryGetAccessibilityObject(out AccessibleObject? accessibleObject)
        {
            accessibleObject = Properties.GetObject(s_accessibilityProperty) as AccessibleObject;
            return accessibleObject is not null;
        }
    }
 
    private void ResetToolTipText() => _toolTipText = null;
 
    // This will only be called in PerMonitorV2 scenarios.
    internal virtual void ToolStrip_RescaleConstants(int oldDpi, int newDpi)
    {
        DeviceDpi = newDpi;
        RescaleConstantsInternal(newDpi);
        OnFontChanged(EventArgs.Empty);
    }
 
    internal void RescaleConstantsInternal(int newDpi)
    {
        ToolStripManager.CurrentDpi = newDpi;
        _defaultFont = ToolStripManager.DefaultFont;
        ScaleConstants(newDpi);
    }
 
    private void ScaleConstants(int dpi)
    {
        _defaultMargin = ScaleHelper.ScaleToDpi(new Padding(0, 1, 0, 2), dpi);
        _defaultStatusStripMargin = ScaleHelper.ScaleToDpi(new Padding(0, 2, 0, 0), dpi);
    }
 
    public void Select() => Select(forceRaiseAccessibilityFocusChanged: false);
 
    internal void Select(bool forceRaiseAccessibilityFocusChanged)
    {
        if (!CanSelect)
        {
            return;
        }
 
        if (Owner is not null && Owner.IsCurrentlyDragging)
        {
            // Make sure we don't select during a drag operation.
            return;
        }
 
        if (ParentInternal is not null && ParentInternal.IsSelectionSuspended)
        {
            return;
        }
 
        if (!Selected)
        {
            _state[s_stateSelected] = true;
            ParentInternal?.NotifySelectionChange(this);
 
            if (IsOnDropDown)
            {
                if (OwnerItem is not null && OwnerItem.IsOnDropDown)
                {
                    // Ensure the selection is moved back to our owner item.
                    OwnerItem.Select();
                }
            }
 
            KeyboardToolTipStateMachine.Instance.NotifyAboutGotFocus(this);
 
            OnSelectedChanged(EventArgs.Empty);
 
            forceRaiseAccessibilityFocusChanged = true;
        }
 
        if (forceRaiseAccessibilityFocusChanged)
        {
            bool accessibilityIsOn = IsAccessibilityObjectCreated ||
                // When ToolStripItem is selected automatically for the first time
                // (for example, when menu bar gets focus or a sub menu is opened, its first item is selected automatically),
                // ToolStripItem's and parent sub menu's AccessibilityObjects might not be created yet.
                // AO tree is going to be constructed just after this first selection, if any accessibility client is on.
                // In this case, to be able to notify Accessibility of focus event right now
                // we determine Accessibility status by checking if parent AO instance is created.
                // If so, then we can force child AO creation.
                (IsOnDropDown
                    ? OwnerItem?.IsAccessibilityObjectCreated ?? false
                    : IsParentAccessibilityObjectCreated);
 
            if (accessibilityIsOn && AccessibilityObject is ToolStripItemAccessibleObject accessibleObject)
            {
                accessibleObject.RaiseFocusChanged();
            }
        }
    }
 
    internal void SetOwner(ToolStrip? newOwner)
    {
        if (_owner != newOwner)
        {
            Font f = Font;
 
            if (_owner is not null)
            {
                _owner._rescaleConstsCallbackDelegate -= ToolStrip_RescaleConstants;
            }
 
            _owner = newOwner;
 
            if (_owner is not null)
            {
                _owner._rescaleConstsCallbackDelegate += ToolStrip_RescaleConstants;
            }
 
            // clear the parent if the owner is null.
            if (newOwner is null)
            {
                ParentInternal = null;
            }
 
            if (!_state[s_stateDisposing] && !IsDisposed)
            {
                OnOwnerChanged(EventArgs.Empty);
                if (f != Font)
                {
                    OnFontChanged(EventArgs.Empty);
                }
            }
        }
    }
 
    protected virtual void SetVisibleCore(bool visible)
    {
        if (_state[s_stateVisible] != visible)
        {
            _state[s_stateVisible] = visible;
            Unselect();
            Push(false);
 
            OnAvailableChanged(EventArgs.Empty);
            OnVisibleChanged(EventArgs.Empty);
        }
    }
 
    /// <summary>
    ///  Sets the bounds of the item
    /// </summary>
    protected internal virtual void SetBounds(Rectangle bounds)
    {
        Rectangle oldBounds = _bounds;
        _bounds = bounds;
 
        if (!_state[s_stateConstructing])
        {
            // Don't fire while we're in the base constructor as the inherited
            // class may not have had a chance to initialize yet.
 
            if (_bounds != oldBounds)
            {
                OnBoundsChanged();
            }
 
            if (_bounds.Location != oldBounds.Location)
            {
                OnLocationChanged(EventArgs.Empty);
            }
        }
    }
 
    /// <summary>
    ///  Sets the bounds of the item
    /// </summary>
    internal void SetBounds(int x, int y, int width, int height)
        => SetBounds(new Rectangle(x, y, width, height));
 
    /// <summary>
    ///  Sets the placement of the item
    /// </summary>
    internal void SetPlacement(ToolStripItemPlacement placement) => _placement = placement;
 
    /// <remarks>
    ///  <para>
    ///   Some implementations of DefaultMargin check which container they
    ///   are on. They need to be re-evaluated when the containership changes.
    ///   DefaultMargin will stop being honored the moment someone sets the Margin property.
    ///  </para>
    /// </remarks>
    internal void SetAmbientMargin()
    {
        if (_state[s_stateUseAmbientMargin] && Margin != DefaultMargin)
        {
            CommonProperties.SetMargin(this, DefaultMargin);
        }
    }
 
    [EditorBrowsable(EditorBrowsableState.Never)]
    private bool ShouldSerializeImageTransparentColor() => ImageTransparentColor != Color.Empty;
 
    /// <summary>
    ///  Returns true if the backColor should be persisted in code gen.
    /// </summary>
    [EditorBrowsable(EditorBrowsableState.Never)]
    internal virtual bool ShouldSerializeBackColor()
    {
        Color backColor = Properties.GetValueOrDefault<Color>(s_backColorProperty);
        return !backColor.IsEmpty;
    }
 
    private bool ShouldSerializeDisplayStyle() => DisplayStyle != DefaultDisplayStyle;
 
    private bool ShouldSerializeToolTipText() => !string.IsNullOrEmpty(_toolTipText);
 
    /// <summary>
    ///  Returns true if the foreColor should be persisted in code gen.
    /// </summary>
    [EditorBrowsable(EditorBrowsableState.Never)]
    internal virtual bool ShouldSerializeForeColor()
    {
        Color foreColor = Properties.GetValueOrDefault<Color>(s_foreColorProperty);
        return !foreColor.IsEmpty;
    }
 
    /// <summary>
    ///  Returns true if the font should be persisted in code gen.
    /// </summary>
    [EditorBrowsable(EditorBrowsableState.Never)]
    internal virtual bool ShouldSerializeFont() => TryGetExplicitlySetFont(out _);
 
    /// <summary>
    ///  Determines if the <see cref="Padding"/> property needs to be persisted.
    /// </summary>
    [EditorBrowsable(EditorBrowsableState.Never)]
    private bool ShouldSerializePadding() => Padding != DefaultPadding;
 
    /// <summary>
    ///  Determines if the <see cref="Margin"/> property needs to be persisted.
    /// </summary>
    [EditorBrowsable(EditorBrowsableState.Never)]
    private bool ShouldSerializeMargin() => Margin != DefaultMargin;
 
    /// <summary>
    ///  Determines if the <see cref="Visible"/> property needs to be persisted.
    /// </summary>
    [EditorBrowsable(EditorBrowsableState.Never)]
    private bool ShouldSerializeVisible() => !_state[s_stateVisible];
 
    /// <summary>
    ///  Determines if the <see cref="Image"/> property needs to be persisted.
    /// </summary>
    [EditorBrowsable(EditorBrowsableState.Never)]
    private bool ShouldSerializeImage() => Image is not null && ImageIndexer.ActualIndex < 0;
 
    /// <summary>
    ///  Determines if the <see cref="Image"/> property needs to be persisted.
    /// </summary>
    [EditorBrowsable(EditorBrowsableState.Never)]
    private bool ShouldSerializeImageKey()
        => Image is not null && ImageIndexer.ActualIndex >= 0 && !string.IsNullOrEmpty(ImageIndexer.Key);
 
    /// <summary>
    ///  Determines if the <see cref="Image"/> property needs to be persisted.
    /// </summary>
    [EditorBrowsable(EditorBrowsableState.Never)]
    private bool ShouldSerializeImageIndex()
        => Image is not null && ImageIndexer.ActualIndex >= 0 && ImageIndexer.Index != ImageList.Indexer.DefaultIndex;
 
    /// <summary>
    ///  Determines if the <see cref="RightToLeft"/> property needs to be persisted.
    /// </summary>
    [EditorBrowsable(EditorBrowsableState.Never)]
    internal virtual bool ShouldSerializeRightToLeft()
    {
        if (!Properties.TryGetValue(s_rightToLeftProperty, out RightToLeft rightToLeft))
        {
            return false;
        }
 
        return rightToLeft != DefaultRightToLeft;
    }
 
    private bool ShouldSerializeTextDirection()
    {
        ToolStripTextDirection textDirection = ToolStripTextDirection.Inherit;
        if (Properties.TryGetObject(s_textDirectionProperty, out ToolStripTextDirection direction))
        {
            textDirection = direction;
        }
 
        return textDirection != ToolStripTextDirection.Inherit;
    }
 
    /// <summary>
    ///  Resets the back color to be based on the parent's back color.
    /// </summary>
    [EditorBrowsable(EditorBrowsableState.Never)]
    public virtual void ResetBackColor() => BackColor = Color.Empty;
 
    [EditorBrowsable(EditorBrowsableState.Never)]
    public virtual void ResetDisplayStyle() => DisplayStyle = DefaultDisplayStyle;
 
    /// <summary>
    ///  Resets the fore color to be based on the parent's fore color.
    /// </summary>
    [EditorBrowsable(EditorBrowsableState.Never)]
    public virtual void ResetForeColor() => ForeColor = Color.Empty;
 
    /// <summary>
    ///  Resets the Font to be based on the parent's Font.
    /// </summary>
    [EditorBrowsable(EditorBrowsableState.Never)]
    public virtual void ResetFont() => Font = null;
 
    /// <summary>
    ///  Resets the back color to be based on the parent's back color.
    /// </summary>
    [EditorBrowsable(EditorBrowsableState.Never)]
    public virtual void ResetImage() => Image = null;
 
    [EditorBrowsable(EditorBrowsableState.Never)]
    private void ResetImageTransparentColor() => ImageTransparentColor = Color.Empty;
 
    [EditorBrowsable(EditorBrowsableState.Never)]
    public void ResetMargin()
    {
        _state[s_stateUseAmbientMargin] = true;
        SetAmbientMargin();
    }
 
    [EditorBrowsable(EditorBrowsableState.Never)]
    public void ResetPadding()
    {
        CommonProperties.ResetPadding(this);
    }
 
    /// <summary>
    ///  Resets the RightToLeft to be the default.
    /// </summary>
    [EditorBrowsable(EditorBrowsableState.Never)]
    public virtual void ResetRightToLeft() => RightToLeft = RightToLeft.Inherit;
 
    /// <summary>
    ///  Resets the TextDirection to be the default.
    /// </summary>
    [EditorBrowsable(EditorBrowsableState.Never)]
    public virtual void ResetTextDirection() => TextDirection = ToolStripTextDirection.Inherit;
 
    /// <summary>
    ///  Translates a point from one coordinate system to another
    /// </summary>
    internal Point TranslatePoint(Point fromPoint, ToolStripPointType fromPointType, ToolStripPointType toPointType)
    {
        ToolStrip? parent = ParentInternal;
 
        parent ??= (IsOnOverflow && Owner is not null) ? Owner.OverflowButton.DropDown : Owner;
 
        if (parent is null)
        {
            // should not throw here as it's an internal function call.
            return fromPoint;
        }
 
        if (fromPointType == toPointType)
        {
            return fromPoint;
        }
 
        Point toPoint = Point.Empty;
        Point currentToolStripItemLocation = Bounds.Location;
 
        // From: Screen
        // To:      ToolStrip or ToolStripItem
        if (fromPointType == ToolStripPointType.ScreenCoords)
        {
            // Convert ScreenCoords --> ToolStripCoords
            toPoint = parent.PointToClient(fromPoint);
 
            // Convert ToolStripCoords --> ToolStripItemCoords
            if (toPointType == ToolStripPointType.ToolStripItemCoords)
            {
                toPoint.X += currentToolStripItemLocation.X;
                toPoint.Y += currentToolStripItemLocation.Y;
            }
        }
 
        // From: ToolStrip or ToolStripItem
        // To:      Screen or ToolStripItem
        else
        {
            // Convert "fromPoint" ToolStripItemCoords --> ToolStripCoords
            if (fromPointType == ToolStripPointType.ToolStripItemCoords)
            {
                fromPoint.X += currentToolStripItemLocation.X;
                fromPoint.Y += currentToolStripItemLocation.Y;
            }
 
            // At this point, fromPoint is now in ToolStrip coordinates.
 
            // Convert ToolStripCoords --> ScreenCoords
            if (toPointType == ToolStripPointType.ScreenCoords)
            {
                toPoint = parent.PointToScreen(fromPoint);
            }
 
            // Convert ToolStripCoords --> ToolStripItemCoords
            else if (toPointType == ToolStripPointType.ToolStripItemCoords)
            {
                fromPoint.X -= currentToolStripItemLocation.X;
                fromPoint.Y -= currentToolStripItemLocation.Y;
                toPoint = fromPoint;
            }
            else
            {
                Debug.Assert((toPointType == ToolStripPointType.ToolStripCoords), "why are we here! - investigate");
                toPoint = fromPoint;
            }
        }
 
        return toPoint;
    }
 
    internal static bool GetIsOffscreenPropertyValue(ToolStripItemPlacement? toolStripItemPlacement, Rectangle bounds)
    {
        return toolStripItemPlacement != ToolStripItemPlacement.Main || bounds.Height <= 0 || bounds.Width <= 0;
    }
 
    internal ToolStrip? RootToolStrip
    {
        get
        {
            ToolStripItem item = this;
            while (item.OwnerItem is not null)
            {
                item = item.OwnerItem;
            }
 
            return item.ParentInternal;
        }
    }
 
    public override string ToString()
    {
        if (!string.IsNullOrEmpty(Text))
        {
            return Text;
        }
 
        return base.ToString();
    }
 
    /// <summary>
    ///  Removes selection bits from item state
    /// </summary>
    internal void Unselect()
    {
        if (_state[s_stateSelected])
        {
            _state[s_stateSelected] = false;
            if (Available)
            {
                Invalidate();
                ParentInternal?.NotifySelectionChange(this);
 
                KeyboardToolTipStateMachine.Instance.NotifyAboutLostFocus(this);
            }
        }
    }
 
    bool IKeyboardToolTip.CanShowToolTipsNow()
        => Visible && _parent is not null && ((IKeyboardToolTip)_parent).AllowsChildrenToShowToolTips();
 
    Rectangle IKeyboardToolTip.GetNativeScreenRectangle() => AccessibilityObject.Bounds;
 
    IList<Rectangle> IKeyboardToolTip.GetNeighboringToolsRectangles()
    {
        List<Rectangle> neighbors = new(3);
        if (_parent is not null)
        {
            ToolStripItemCollection items = _parent.DisplayedItems;
            int i = 0, count = items.Count;
            bool found = false;
            while (!found && i < count)
            {
                found = ReferenceEquals(items[i], this);
                if (found)
                {
                    int previousIndex = i - 1;
                    if (previousIndex >= 0)
                    {
                        neighbors.Add(((IKeyboardToolTip)items[previousIndex]).GetNativeScreenRectangle());
                    }
 
                    int nextIndex = i + 1;
                    if (nextIndex < count)
                    {
                        neighbors.Add(((IKeyboardToolTip)items[nextIndex]).GetNativeScreenRectangle());
                    }
                }
                else
                {
                    i++;
                }
            }
 
            Debug.Assert(i < count, "Item has a parent set but the parent doesn't own the item");
        }
 
        if (_parent is ToolStripDropDown dropDown && dropDown.OwnerItem is not null)
        {
            neighbors.Add(((IKeyboardToolTip)dropDown.OwnerItem).GetNativeScreenRectangle());
        }
 
        return neighbors;
    }
 
    bool IKeyboardToolTip.IsHoveredWithMouse()
        => ((IKeyboardToolTip)this).GetNativeScreenRectangle().Contains(Control.MousePosition);
 
    bool IKeyboardToolTip.HasRtlModeEnabled()
        => _parent is not null && ((IKeyboardToolTip)_parent).HasRtlModeEnabled();
 
    bool IKeyboardToolTip.AllowsToolTip() => true;
 
    IWin32Window IKeyboardToolTip.GetOwnerWindow()
    {
        Debug.Assert(ParentInternal is not null, "Tool Strip Item Parent is null");
        return ParentInternal;
    }
 
    void IKeyboardToolTip.OnHooked(ToolTip toolTip) => OnKeyboardToolTipHook(toolTip);
 
    void IKeyboardToolTip.OnUnhooked(ToolTip toolTip) => OnKeyboardToolTipUnhook(toolTip);
 
    string? IKeyboardToolTip.GetCaptionForTool(ToolTip toolTip) => ToolTipText;
 
    bool IKeyboardToolTip.ShowsOwnToolTip() => true;
 
    bool IKeyboardToolTip.IsBeingTabbedTo() => IsBeingTabbedTo();
 
    bool IKeyboardToolTip.AllowsChildrenToShowToolTips() => true;
 
    internal virtual void OnKeyboardToolTipHook(ToolTip toolTip)
    {
    }
 
    internal virtual void OnKeyboardToolTipUnhook(ToolTip toolTip)
    {
    }
 
    /// <summary>
    ///  Indicates whether or not the parent of this item has an accessible object associated with it.
    /// </summary>
    internal bool IsParentAccessibilityObjectCreated => ParentInternal is not null && ParentInternal.IsAccessibilityObjectCreated;
 
    internal virtual bool IsBeingTabbedTo() => Control.AreCommonNavigationalKeysDown();
 
    /// <summary>
    ///  Query font from property bag.
    /// </summary>
    internal bool TryGetExplicitlySetFont([NotNullWhen(true)] out Font? local)
    {
        local = (Font?)Properties.GetObject(s_fontProperty);
 
        return local is not null;
    }
}