File: System\Windows\Forms\Controls\TabControl\TabControl.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.Text;
using System.Windows.Forms.Layout;
using Windows.Win32.UI.Accessibility;
 
namespace System.Windows.Forms;
 
/// <summary>
///  The TabControl. This control has a lot of the functionality of a TabStrip
///  but manages a list of TabPages which are the 'pages' that appear on each tab.
/// </summary>
[DefaultProperty(nameof(TabPages))]
[DefaultEvent(nameof(SelectedIndexChanged))]
[Designer($"System.Windows.Forms.Design.TabControlDesigner, {AssemblyRef.SystemDesign}")]
[SRDescription(nameof(SR.DescriptionTabControl))]
public partial class TabControl : Control
{
    private static readonly Size s_defaultItemSize = Size.Empty;
    private static readonly Point s_defaultPaddingPoint = new(6, 3);
 
    // Properties
    private readonly TabPageCollection _tabCollection;
    private TabAlignment _alignment = TabAlignment.Top;
    private TabDrawMode _drawMode = TabDrawMode.Normal;
    private ImageList? _imageList;
    private Size _itemSize = s_defaultItemSize;
    private Point _padding = s_defaultPaddingPoint;
    private TabSizeMode _sizeMode = TabSizeMode.Normal;
    private TabAppearance _appearance = TabAppearance.Normal;
    private Rectangle _cachedDisplayRect;
    private bool _currentlyScaling;
    private int _selectedIndex = -1;
    private string? _controlTipText = string.Empty;
    private bool _handleInTable;
 
    // Events
    private EventHandler? _onSelectedIndexChanged;
    private DrawItemEventHandler? _onDrawItem;
 
    private static readonly object s_deselectingEvent = new();
    private static readonly object s_deselectedEvent = new();
    private static readonly object s_selectingEvent = new();
    private static readonly object s_selectedEvent = new();
    private static readonly object s_rightToLeftLayoutChangedEvent = new();
 
    // Perf: take all the bools and put them into a state variable: see TabControlState constants above
    private BitVector32 _tabControlState;
 
    private const string TabBaseReLayoutMessageName = "_TabBaseReLayout";
 
    /// <summary>
    ///  This message is posted by the control to itself after a TabPage is
    ///  added to it. On certain occasions, after items are added to a
    ///  TabControl in quick succession, TCM_ADJUSTRECT calls return the wrong
    ///  display rectangle. When the message is received, the control calls
    ///  updateTabSelection() to layout the TabPages correctly.
    /// </summary>
    private readonly MessageId _tabBaseReLayoutMessage = PInvoke.RegisterWindowMessage($"{Application.WindowMessagesVersion}{TabBaseReLayoutMessageName}");
 
    // State
    private readonly List<TabPage> _tabPages = [];
    private int _lastSelection;
    private short _windowId;
 
    private bool _rightToLeftLayout;
    private bool _skipUpdateSize;
 
    private ToolTipBuffer _toolTipBuffer;
 
    /// <summary>
    ///  Constructs a TabBase object, usually as the base class for a TabStrip or TabControl.
    /// </summary>
    public TabControl()
        : base()
    {
        _tabControlState = new BitVector32(0x00000000);
 
        _tabCollection = new TabPageCollection(this);
        SetStyle(ControlStyles.UserPaint, false);
#pragma warning disable WFO5001 // Type is for evaluation purposes only and is subject to change or removal in future updates. Suppress this diagnostic to proceed.
        SetStyle(ControlStyles.ApplyThemingImplicitly, true);
#pragma warning restore WFO5001
    }
 
    /// <summary>
    ///  Returns on what area of the control the tabs reside on (A TabAlignment value).
    ///  The possibilities are Top (the default), Bottom, Left, and Right. When alignment
    ///  is left or right, the Multiline property is ignored and Multiline is implicitly on.
    ///  If the alignment is anything other than top, TabAppearance.FlatButtons degenerates
    ///  to TabAppearance.Buttons.
    /// </summary>
    [SRCategory(nameof(SR.CatBehavior))]
    [Localizable(true)]
    [DefaultValue(TabAlignment.Top)]
    [RefreshProperties(RefreshProperties.All)]
    [SRDescription(nameof(SR.TabBaseAlignmentDescr))]
    public TabAlignment Alignment
    {
        get
        {
            return _alignment;
        }
 
        set
        {
            if (_alignment != value)
            {
                // valid values are 0x0 to 0x3
                SourceGenerated.EnumValidator.Validate(value);
 
                _alignment = value;
                if (_alignment is TabAlignment.Left or TabAlignment.Right)
                {
                    SetState(State.Multiline, true);
                }
 
                RecreateHandle();
            }
        }
    }
 
    /// <summary>
    ///  Indicates whether the tabs in the TabStrip look like regular tabs, or if they look
    ///  like buttons as seen in the Windows 95 taskbar.
    ///  If the alignment is anything other than top, TabAppearance.FlatButtons degenerates
    ///  to TabAppearance.Buttons.
    /// </summary>
    [SRCategory(nameof(SR.CatBehavior))]
    [Localizable(true)]
    [DefaultValue(TabAppearance.Normal)]
    [SRDescription(nameof(SR.TabBaseAppearanceDescr))]
    public TabAppearance Appearance
    {
        get
        {
            if (_appearance == TabAppearance.FlatButtons && _alignment != TabAlignment.Top)
            {
                return TabAppearance.Buttons;
            }
            else
            {
                return _appearance;
            }
        }
 
        set
        {
            if (_appearance != value)
            {
                // valid values are 0x0 to 0x2
                SourceGenerated.EnumValidator.Validate(value);
 
                _appearance = value;
                RecreateHandle();
 
                // Fire OnStyleChanged(EventArgs.Empty) here since we are no longer calling UpdateStyles( ) but always reCreating the Handle.
                OnStyleChanged(EventArgs.Empty);
            }
        }
    }
 
    [Browsable(false)]
    [EditorBrowsable(EditorBrowsableState.Never)]
    public override Color BackColor
    {
        get
        {
            // The tab control can only be rendered in 1 color: System's Control color.
            // So, always return this value... otherwise, we're inheriting the forms BackColor
            // and passing it on to the pab pages.
            return SystemColors.Control;
        }
        set
        {
        }
    }
 
    [Browsable(false)]
    [EditorBrowsable(EditorBrowsableState.Never)]
    public new event EventHandler? BackColorChanged
    {
        add => base.BackColorChanged += value;
        remove => base.BackColorChanged -= value;
    }
 
    [Browsable(false)]
    [EditorBrowsable(EditorBrowsableState.Never)]
    public override Image? BackgroundImage
    {
        get => base.BackgroundImage;
        set => base.BackgroundImage = value;
    }
 
    [Browsable(false)]
    [EditorBrowsable(EditorBrowsableState.Never)]
    public new event EventHandler? BackgroundImageChanged
    {
        add => base.BackgroundImageChanged += value;
        remove => base.BackgroundImageChanged -= value;
    }
 
    [Browsable(false)]
    [EditorBrowsable(EditorBrowsableState.Never)]
    public override ImageLayout BackgroundImageLayout
    {
        get => base.BackgroundImageLayout;
        set => base.BackgroundImageLayout = value;
    }
 
    [Browsable(false)]
    [EditorBrowsable(EditorBrowsableState.Never)]
    public new event EventHandler? BackgroundImageLayoutChanged
    {
        add => base.BackgroundImageLayoutChanged += value;
        remove => base.BackgroundImageLayoutChanged -= value;
    }
 
    /// <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 override Size DefaultSize
    {
        get
        {
            return new Size(200, 100);
        }
    }
 
    /// <summary>
    ///  This property is overridden and hidden from statement completion
    ///  on controls that are based on Win32 Native Controls.
    /// </summary>
    [EditorBrowsable(EditorBrowsableState.Never)]
    protected override bool DoubleBuffered
    {
        get => base.DoubleBuffered;
        set => base.DoubleBuffered = value;
    }
 
    [Browsable(false)]
    [EditorBrowsable(EditorBrowsableState.Never)]
    public override Color ForeColor
    {
        get => base.ForeColor;
        set => base.ForeColor = value;
    }
 
    [Browsable(false)]
    [EditorBrowsable(EditorBrowsableState.Never)]
    public new event EventHandler? ForeColorChanged
    {
        add => base.ForeColorChanged += value;
        remove => base.ForeColorChanged -= value;
    }
 
    /// <summary>
    ///  Returns the parameters needed to create the handle. Inheriting classes
    ///  can override this to provide extra functionality. They should not,
    ///  however, forget to call base.getCreateParams() first to get the struct
    ///  filled up with the basic info.
    /// </summary>
    protected override CreateParams CreateParams
    {
        get
        {
            CreateParams cp = base.CreateParams;
            cp.ClassName = PInvoke.WC_TABCONTROL;
 
            // set up window styles
            //
            if (Multiline)
            {
                cp.Style |= (int)PInvoke.TCS_MULTILINE;
            }
 
            if (_drawMode == TabDrawMode.OwnerDrawFixed)
            {
                cp.Style |= (int)PInvoke.TCS_OWNERDRAWFIXED;
            }
 
            if (ShowToolTips && !DesignMode)
            {
                cp.Style |= (int)PInvoke.TCS_TOOLTIPS;
            }
 
            if (_alignment is TabAlignment.Bottom or
                TabAlignment.Right)
            {
                cp.Style |= (int)PInvoke.TCS_BOTTOM;
            }
 
            if (_alignment is TabAlignment.Left or
                TabAlignment.Right)
            {
                cp.Style |= (int)PInvoke.TCS_VERTICAL | (int)PInvoke.TCS_MULTILINE;
            }
 
            if (GetState(State.HotTrack))
            {
                cp.Style |= (int)PInvoke.TCS_HOTTRACK;
            }
 
            if (_appearance == TabAppearance.Normal)
            {
                cp.Style |= (int)PInvoke.TCS_TABS;
            }
            else
            {
                cp.Style |= (int)PInvoke.TCS_BUTTONS;
                if (_appearance == TabAppearance.FlatButtons && _alignment == TabAlignment.Top)
                {
                    cp.Style |= (int)PInvoke.TCS_FLATBUTTONS;
                }
            }
 
            switch (_sizeMode)
            {
                case TabSizeMode.Normal:
                    cp.Style |= (int)PInvoke.TCS_RAGGEDRIGHT;
                    break;
                case TabSizeMode.FillToRight:
                    cp.Style |= (int)PInvoke.TCS_RIGHTJUSTIFY;
                    break;
                case TabSizeMode.Fixed:
                    cp.Style |= (int)PInvoke.TCS_FIXEDWIDTH;
                    break;
            }
 
            if (RightToLeft == RightToLeft.Yes && RightToLeftLayout)
            {
                // We want to turn on mirroring for Form explicitly.
                cp.ExStyle |= (int)(WINDOW_EX_STYLE.WS_EX_LAYOUTRTL | WINDOW_EX_STYLE.WS_EX_NOINHERITLAYOUT);
                // Don't need these styles when mirroring is turned on.
                cp.ExStyle &= ~(int)(WINDOW_EX_STYLE.WS_EX_RTLREADING | WINDOW_EX_STYLE.WS_EX_RIGHT | WINDOW_EX_STYLE.WS_EX_LEFTSCROLLBAR);
            }
 
            return cp;
        }
    }
 
    /// <summary>
    ///  The rectangle that represents the Area of the tab strip not
    ///  taken up by the tabs, borders, or anything else owned by the Tab. This
    ///  is typically the rectangle you want to use to place the individual
    ///  children of the tab strip.
    /// </summary>
    public override Rectangle DisplayRectangle
    {
        get
        {
            // Set the cached display rect to Rectangle.Empty whenever we do anything to change it.
            if (!_cachedDisplayRect.IsEmpty)
            {
                return _cachedDisplayRect;
            }
 
            RECT rect = Bounds;
 
            // We force a handle creation here, because otherwise the DisplayRectangle will be wildly inaccurate
            if (!IsDisposed)
            {
                // Since this is called thru the OnResize (and Layout) which is triggered by SetExtent if the TabControl is hosted as
                // a ActiveX control, so check if this is ActiveX and don't force Handle Creation here as the native code breaks in this case.
                if (!IsActiveX)
                {
                    if (!IsHandleCreated)
                    {
                        CreateHandle();
                    }
                }
 
                if (IsHandleCreated)
                {
                    PInvokeCore.SendMessage(this, PInvoke.TCM_ADJUSTRECT, 0, ref rect);
                }
            }
 
            Rectangle r = rect;
 
            Point p = Location;
            r.X -= p.X;
            r.Y -= p.Y;
 
            _cachedDisplayRect = r;
            return r;
        }
    }
 
    /// <summary>
    ///  The drawing mode of the tabs in the tab strip. This will indicate
    /// </summary>
    [SRCategory(nameof(SR.CatBehavior))]
    [DefaultValue(TabDrawMode.Normal)]
    [SRDescription(nameof(SR.TabBaseDrawModeDescr))]
    public TabDrawMode DrawMode
    {
        get
        {
            return _drawMode;
        }
 
        set
        {
            // valid values are 0x0 to 0x1
            SourceGenerated.EnumValidator.Validate(value);
 
            if (_drawMode != value)
            {
                _drawMode = value;
                RecreateHandle();
            }
        }
    }
 
    /// <summary>
    ///  Indicates whether the tabs visually change when the mouse passes over them.
    /// </summary>
    [SRCategory(nameof(SR.CatBehavior))]
    [DefaultValue(false)]
    [SRDescription(nameof(SR.TabBaseHotTrackDescr))]
    public bool HotTrack
    {
        get => GetState(State.HotTrack);
        set
        {
            if (HotTrack != value)
            {
                SetState(State.HotTrack, value);
                if (IsHandleCreated)
                {
                    RecreateHandle();
                }
            }
        }
    }
 
    /// <summary>
    ///  Returns the imageList the control points at. This is where tabs that have imageIndex
    ///  set will get there images from.
    /// </summary>
    [SRCategory(nameof(SR.CatAppearance))]
    [RefreshProperties(RefreshProperties.Repaint)]
    [DefaultValue(null)]
    [SRDescription(nameof(SR.TabBaseImageListDescr))]
    public ImageList? ImageList
    {
        get
        {
            return _imageList;
        }
        set
        {
            if (_imageList != value)
            {
                EventHandler recreateHandler = new(ImageListRecreateHandle);
                EventHandler disposedHandler = new(DetachImageList);
 
                if (_imageList is not null)
                {
                    _imageList.RecreateHandle -= recreateHandler;
                    _imageList.Disposed -= disposedHandler;
                }
 
                _imageList = value;
                IntPtr handle = (value is not null) ? value.Handle : IntPtr.Zero;
                if (IsHandleCreated)
                {
                    PInvokeCore.SendMessage(this, PInvoke.TCM_SETIMAGELIST, 0, handle);
                }
 
                // Update the image list in the tab pages.
                foreach (TabPage tabPage in TabPages)
                {
                    tabPage.ImageIndexer.ImageList = value;
                }
 
                if (value is not null)
                {
                    value.RecreateHandle += recreateHandler;
                    value.Disposed += disposedHandler;
                }
            }
        }
    }
 
    /// <summary>
    ///  By default, tabs will automatically size themselves to fit their icon, if any, and their label.
    ///  However, the tab size can be explicitly set by setting this property.
    /// </summary>
    [SRCategory(nameof(SR.CatBehavior))]
    [Localizable(true)]
    [SRDescription(nameof(SR.TabBaseItemSizeDescr))]
    public Size ItemSize
    {
        get
        {
            if (_itemSize.IsEmpty)
            {
                // Obtain the current ItemSize of the first tab from the winctl control
                if (IsHandleCreated)
                {
                    SetState(State.GetTabRectfromItemSize, true);
                    return GetTabRect(0).Size;
                }
 
                return s_defaultItemSize;
            }
 
            return _itemSize;
        }
        set
        {
            if (value.Width < 0 || value.Height < 0)
            {
                throw new ArgumentOutOfRangeException(nameof(value), value, string.Format(SR.InvalidArgument, nameof(ItemSize), value));
            }
 
            _itemSize = value;
            ApplyItemSize();
            UpdateSize();
            Invalidate();
        }
    }
 
    /// <summary>
    ///  This private property is set by the TabPageCollection when the user calls "InsertItem".
    ///  The problem is when InsertItem is called then we add this item to the ControlsCollection
    ///  (in addition to the TabPageCollection) to keep both the collections is sync. But the controlCollection.
    ///  Add is overridden to again ADD the item to the TabPageCollection.
    ///  So we keep this flag in order to avoid repeated addition (only during insert)
    ///  When the Add ends ... we reset this flag.
    /// </summary>
    private bool InsertingItem
    {
        get => GetState(State.InsertingItem);
        set => SetState(State.InsertingItem, value);
    }
 
    /// <summary>
    ///  Indicates if there can be more than one row of tabs. By default [when
    ///  this property is false], if there are more tabs than available display
    ///  space, arrows are shown to let the user navigate between the extra
    ///  tabs, but only one row is shown. If this property is set to true, then
    ///  Windows spills extra tabs over on to second rows.
    /// </summary>
    [SRCategory(nameof(SR.CatBehavior))]
    [DefaultValue(false)]
    [SRDescription(nameof(SR.TabBaseMultilineDescr))]
    public bool Multiline
    {
        get => GetState(State.Multiline);
        set
        {
            if (Multiline != value)
            {
                SetState(State.Multiline, value);
                if (!Multiline && (_alignment == TabAlignment.Left || _alignment == TabAlignment.Right))
                {
                    _alignment = TabAlignment.Top;
                }
 
                RecreateHandle();
            }
        }
    }
 
    /// <summary>
    ///  The amount of padding around the items in the individual tabs.
    ///  You can specify both horizontal and vertical padding.
    /// </summary>
    [SRCategory(nameof(SR.CatBehavior))]
    [Localizable(true)]
    [SRDescription(nameof(SR.TabBasePaddingDescr))]
    public new Point Padding
    {
        get
        {
            return _padding;
        }
        set
        {
            if (value.X < 0 || value.Y < 0)
            {
                throw new ArgumentOutOfRangeException(nameof(value), value, string.Format(SR.InvalidArgument, nameof(Padding), value));
            }
 
            if (_padding != value)
            {
                _padding = value;
                if (IsHandleCreated)
                {
                    RecreateHandle();
                }
            }
        }
    }
 
    /// <summary>
    ///  This is used for international applications where the language is written from RightToLeft.
    ///  When this property is true, and the RightToLeft is true, mirroring will be turned on on
    ///  the form, and control placement and text will be from right to left.
    /// </summary>
    [SRCategory(nameof(SR.CatAppearance))]
    [Localizable(true)]
    [DefaultValue(false)]
    [SRDescription(nameof(SR.ControlRightToLeftLayoutDescr))]
    public virtual bool RightToLeftLayout
    {
        get
        {
            return _rightToLeftLayout;
        }
 
        set
        {
            if (value != _rightToLeftLayout)
            {
                _rightToLeftLayout = value;
                using (new LayoutTransaction(this, this, PropertyNames.RightToLeftLayout))
                {
                    OnRightToLeftLayoutChanged(EventArgs.Empty);
                }
            }
        }
    }
 
    /// <summary>
    ///  The number of rows currently being displayed in
    ///  the tab strip. This is most commonly used when the Multiline property
    ///  is 'true' and you want to know how many rows the tabs are currently
    ///  taking up.
    /// </summary>
    [SRCategory(nameof(SR.CatAppearance))]
    [Browsable(false)]
    [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
    [SRDescription(nameof(SR.TabBaseRowCountDescr))]
    public int RowCount
        => (int)PInvokeCore.SendMessage(this, PInvoke.TCM_GETROWCOUNT);
 
    /// <summary>
    ///  The index of the currently selected tab in the strip, if there
    ///  is one. If the value is -1, there is currently no selection. If the
    ///  value is 0 or greater, than the value is the index of the currently
    ///  selected tab.
    /// </summary>
    [Browsable(false)]
    [SRCategory(nameof(SR.CatBehavior))]
    [DefaultValue(-1)]
    [SRDescription(nameof(SR.selectedIndexDescr))]
    public int SelectedIndex
    {
        get => IsHandleCreated ? (int)PInvokeCore.SendMessage(this, PInvoke.TCM_GETCURSEL) : _selectedIndex;
        set
        {
            ArgumentOutOfRangeException.ThrowIfLessThan(value, -1);
 
            if (SelectedIndex != value)
            {
                if (IsHandleCreated)
                {
                    // Guard Against CreateHandle ..
                    // And also if we are setting SelectedIndex ourselves from SelectNextTab..
                    if (!GetState(State.FromCreateHandles) && !GetState(State.SelectFirstControl))
                    {
                        SetState(State.UISelection, true);
                        // Fire Deselecting .. Deselected on currently selected TabPage...
                        if (WmSelChanging())
                        {
                            SetState(State.UISelection, false);
                            return;
                        }
 
                        if (ValidationCancelled)
                        {
                            SetState(State.UISelection, false);
                            return;
                        }
                    }
 
                    PInvokeCore.SendMessage(this, PInvoke.TCM_SETCURSEL, (WPARAM)value);
 
                    if (!GetState(State.FromCreateHandles) && !GetState(State.SelectFirstControl))
                    {
                        // Fire Selecting & Selected .. Also if Selecting is Canceled..
                        // then return as we do not change the SelectedIndex...
                        SetState(State.SelectFirstControl, true);
                        if (WmSelChange())
                        {
                            SetState(State.UISelection, false);
                            SetState(State.SelectFirstControl, false);
                            return;
                        }
                        else
                        {
                            SetState(State.SelectFirstControl, false);
                        }
                    }
                }
                else
                {
                    _selectedIndex = value;
                }
            }
        }
    }
 
    /// <summary>
    ///  The selection to the given tab, provided it .equals a tab in the
    ///  list. The return value is the index of the tab that was selected,
    ///  or -1 if no tab was selected.
    /// </summary>
    [SRCategory(nameof(SR.CatAppearance))]
    [Browsable(false)]
    [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
    [SRDescription(nameof(SR.TabControlSelectedTabDescr))]
    public TabPage? SelectedTab
    {
        get
        {
            int index = SelectedIndex;
            if (index == -1 || _tabPages.Count == 0)
            {
                return null;
            }
 
            return _tabPages[index];
        }
        set
        {
            int index = FindTabPage(value);
            SelectedIndex = index;
        }
    }
 
    /// <summary>
    ///  By default, tabs are big enough to display their text, and any space
    ///  on the right of the strip is left as such. However, you can also
    ///  set it such that the tabs are stretched to fill out the right extent
    ///  of the strip, if necessary, or you can set it such that all tabs
    ///  the same width.
    /// </summary>
    [SRCategory(nameof(SR.CatBehavior))]
    [DefaultValue(TabSizeMode.Normal)]
    [SRDescription(nameof(SR.TabBaseSizeModeDescr))]
    [RefreshProperties(RefreshProperties.Repaint)]
    public TabSizeMode SizeMode
    {
        get => _sizeMode;
        set
        {
            if (_sizeMode == value)
            {
                return;
            }
 
            // valid values are 0x0 to 0x2
            SourceGenerated.EnumValidator.Validate(value);
 
            _sizeMode = value;
            RecreateHandle();
        }
    }
 
    /// <summary>
    ///  Indicates whether tooltips are being shown for tabs that have tooltips set on
    ///  them.
    /// </summary>
    [SRCategory(nameof(SR.CatBehavior))]
    [DefaultValue(false)]
    [Localizable(true)]
    [SRDescription(nameof(SR.TabBaseShowToolTipsDescr))]
    public bool ShowToolTips
    {
        get => GetState(State.ShowToolTips);
        set
        {
            if (ShowToolTips != value)
            {
                SetState(State.ShowToolTips, value);
                RecreateHandle();
            }
        }
    }
 
    internal override bool SupportsUiaProviders => true;
 
    /// <summary>
    ///  Returns the number of tabs in the strip
    /// </summary>
    [SRCategory(nameof(SR.CatAppearance))]
    [Browsable(false)]
    [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
    [SRDescription(nameof(SR.TabBaseTabCountDescr))]
    public int TabCount => _tabPages?.Count ?? 0;
 
    /// <summary>
    ///  Returns the Collection of TabPages.
    /// </summary>
    [SRCategory(nameof(SR.CatBehavior))]
    [SRDescription(nameof(SR.TabControlTabsDescr))]
    [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
    [Editor($"System.Windows.Forms.Design.TabPageCollectionEditor, {AssemblyRef.SystemDesign}", typeof(UITypeEditor))]
    [MergableProperty(false)]
    public TabPageCollection TabPages
    {
        get
        {
            return _tabCollection;
        }
    }
 
    [Browsable(false)]
    [EditorBrowsable(EditorBrowsableState.Never)]
    [Bindable(false)]
    [AllowNull]
    public override string Text
    {
        get => base.Text;
        set => base.Text = value;
    }
 
    [Browsable(false)]
    [EditorBrowsable(EditorBrowsableState.Never)]
    public new event EventHandler? TextChanged
    {
        add => base.TextChanged += value;
        remove => base.TextChanged -= value;
    }
 
    [SRCategory(nameof(SR.CatBehavior))]
    [SRDescription(nameof(SR.drawItemEventDescr))]
    public event DrawItemEventHandler? DrawItem
    {
        add => _onDrawItem += value;
        remove => _onDrawItem -= value;
    }
 
    [SRCategory(nameof(SR.CatPropertyChanged))]
    [SRDescription(nameof(SR.ControlOnRightToLeftLayoutChangedDescr))]
    public event EventHandler? RightToLeftLayoutChanged
    {
        add => Events.AddHandler(s_rightToLeftLayoutChangedEvent, value);
        remove => Events.RemoveHandler(s_rightToLeftLayoutChangedEvent, value);
    }
 
    [SRCategory(nameof(SR.CatBehavior))]
    [SRDescription(nameof(SR.selectedIndexChangedEventDescr))]
    public event EventHandler? SelectedIndexChanged
    {
        add => _onSelectedIndexChanged += value;
        remove => _onSelectedIndexChanged -= value;
    }
 
    /// <summary>
    ///  Occurs before a TabPage is selected as the top TabPage.
    /// </summary>
    [SRCategory(nameof(SR.CatAction))]
    [SRDescription(nameof(SR.TabControlSelectingEventDescr))]
    public event TabControlCancelEventHandler? Selecting
    {
        add => Events.AddHandler(s_selectingEvent, value);
        remove => Events.RemoveHandler(s_selectingEvent, value);
    }
 
    /// <summary>
    ///  Occurs after a TabPage is selected as the top TabPage.
    /// </summary>
    [SRCategory(nameof(SR.CatAction))]
    [SRDescription(nameof(SR.TabControlSelectedEventDescr))]
    public event TabControlEventHandler? Selected
    {
        add => Events.AddHandler(s_selectedEvent, value);
        remove => Events.RemoveHandler(s_selectedEvent, value);
    }
 
    /// <summary>
    ///  Occurs before the visible property of the top TabPage is set to false.
    /// </summary>
    [SRCategory(nameof(SR.CatAction))]
    [SRDescription(nameof(SR.TabControlDeselectingEventDescr))]
    public event TabControlCancelEventHandler? Deselecting
    {
        add => Events.AddHandler(s_deselectingEvent, value);
        remove => Events.RemoveHandler(s_deselectingEvent, value);
    }
 
    /// <summary>
    ///  Occurs after the visible property of the top TabPage is set to false.
    /// </summary>
    [SRCategory(nameof(SR.CatAction))]
    [SRDescription(nameof(SR.TabControlDeselectedEventDescr))]
    public event TabControlEventHandler? Deselected
    {
        add => Events.AddHandler(s_deselectedEvent, value);
        remove => Events.RemoveHandler(s_deselectedEvent, value);
    }
 
    /// <summary>
    ///  TabControl OnPaint.
    /// </summary>
    /// <hideinheritance/>
    [Browsable(false)]
    [EditorBrowsable(EditorBrowsableState.Never)]
    public new event PaintEventHandler? Paint
    {
        add => base.Paint += value;
        remove => base.Paint -= value;
    }
 
    private int AddTabPage(TabPage tabPage)
    {
        int index = AddNativeTabPage(tabPage);
        if (index >= 0)
        {
            Insert(index, tabPage);
        }
 
        return index;
    }
 
    private int AddNativeTabPage(TabPage tabPage)
    {
        int index = SendMessage(PInvoke.TCM_INSERTITEMW, TabCount + 1, tabPage);
        PInvokeCore.PostMessage(this, _tabBaseReLayoutMessage);
        return index;
    }
 
    internal void ApplyItemSize()
    {
        if (IsHandleCreated && ShouldSerializeItemSize())
        {
            PInvokeCore.SendMessage(this, PInvoke.TCM_SETITEMSIZE, 0, PARAM.FromLowHigh(_itemSize.Width, _itemSize.Height));
        }
 
        _cachedDisplayRect = Rectangle.Empty;
    }
 
    internal void BeginUpdate()
    {
        BeginUpdateInternal();
    }
 
    protected override AccessibleObject CreateAccessibilityInstance() => new TabControlAccessibleObject(this);
 
    protected override Control.ControlCollection CreateControlsInstance()
    {
        return new ControlCollection(this);
    }
 
    protected override unsafe void CreateHandle()
    {
        if (!RecreatingHandle)
        {
            using ThemingScope scope = new(Application.UseVisualStyles);
            PInvoke.InitCommonControlsEx(new INITCOMMONCONTROLSEX()
            {
                dwSize = (uint)sizeof(INITCOMMONCONTROLSEX),
                dwICC = INITCOMMONCONTROLSEX_ICC.ICC_TAB_CLASSES
            });
        }
 
        base.CreateHandle();
    }
 
    private void DetachImageList(object? sender, EventArgs e) => ImageList = null;
 
    /// <summary>
    ///  Allows the user to specify the index in TabControl.TabPageCollection of the TabPage to be hidden.
    /// </summary>
    public void DeselectTab(int index)
    {
        TabPage t = GetTabPage(index);
        if (SelectedTab == t)
        {
            if (index >= 0 && index < TabPages.Count - 1)
            {
                SelectedTab = GetTabPage(++index);
            }
            else
            {
                SelectedTab = GetTabPage(0);
            }
        }
    }
 
    /// <summary>
    ///  Allows the user to specify the TabPage in TabControl.TabPageCollection to be hidden.
    /// </summary>
    public void DeselectTab(TabPage tabPage)
    {
        ArgumentNullException.ThrowIfNull(tabPage);
 
        int index = FindTabPage(tabPage);
        DeselectTab(index);
    }
 
    /// <summary>
    ///  Allows the user to specify the name of the TabPage in TabControl.TabPageCollection to be hidden.
    /// </summary>
    public void DeselectTab(string tabPageName)
    {
        ArgumentNullException.ThrowIfNull(tabPageName);
 
        TabPage tabPage = TabPages[tabPageName]!;
        DeselectTab(tabPage);
    }
 
    protected override void Dispose(bool disposing)
    {
        if (disposing)
        {
            if (_imageList is not null)
            {
                _imageList.Disposed -= DetachImageList;
            }
        }
 
        base.Dispose(disposing);
    }
 
    internal void EndUpdate()
    {
        EndUpdate(true);
    }
 
    internal void EndUpdate(bool invalidate)
    {
        EndUpdateInternal(invalidate);
    }
 
    internal int FindTabPage(TabPage? tabPage)
    {
        for (int i = 0; i < _tabPages.Count; i++)
        {
            if (_tabPages[i].Equals(tabPage))
            {
                return i;
            }
        }
 
        return -1;
    }
 
    public Control GetControl(int index)
    {
        return GetTabPage(index);
    }
 
    internal TabPage GetTabPage(int index)
    {
        ArgumentOutOfRangeException.ThrowIfNegative(index);
        ArgumentOutOfRangeException.ThrowIfGreaterThanOrEqual(index, TabCount);
 
        return _tabPages[index];
    }
 
    /// <summary>
    ///  This has package scope so that TabStrip and TabControl can call it.
    /// </summary>
    protected virtual object[] GetItems()
    {
        if (_tabPages.Count > 0)
        {
            return _tabPages.ToArray();
        }
 
        return Array.Empty<TabPage>();
    }
 
    /// <summary>
    ///  This has package scope so that TabStrip and TabControl can call it.
    /// </summary>
    protected virtual object[] GetItems(Type baseType)
    {
        int tabPageCount = TabCount;
        object[] result = (object[])Array.CreateInstance(baseType, tabPageCount);
        if (tabPageCount > 0)
        {
            for (int i = 0; i < tabPageCount; i++)
            {
                result[i] = _tabPages[i];
            }
        }
 
        return result;
    }
 
    internal TabPage[] GetTabPages()
    {
        return (TabPage[])GetItems();
    }
 
    /// <summary>
    ///  Retrieves the bounding rectangle for the given tab in the tab strip.
    /// </summary>
    public Rectangle GetTabRect(int index)
    {
        ArgumentOutOfRangeException.ThrowIfNegative(index);
 
        if (!GetState(State.GetTabRectfromItemSize))
        {
            ArgumentOutOfRangeException.ThrowIfGreaterThanOrEqual(index, TabCount);
        }
 
        SetState(State.GetTabRectfromItemSize, false);
        RECT rect = default;
 
        // normally, we would not want to create the handle for this, but since
        // it is dependent on the actual physical display, we simply must.
        if (!IsHandleCreated)
        {
            CreateHandle();
        }
 
        PInvokeCore.SendMessage(this, PInvoke.TCM_GETITEMRECT, (WPARAM)index, ref rect);
        return rect;
    }
 
    protected string GetToolTipText(object item)
    {
        ArgumentNullException.ThrowIfNull(item);
 
        if (item is not TabPage tabPage)
        {
            throw new ArgumentException(SR.TabControlBadControl, nameof(item));
        }
 
        return tabPage.ToolTipText;
    }
 
    private void ImageListRecreateHandle(object? sender, EventArgs e)
    {
        if (IsHandleCreated)
        {
            PInvokeCore.SendMessage(this, PInvoke.TCM_SETIMAGELIST, 0, ImageList!.Handle);
        }
    }
 
    internal void Insert(int index, TabPage tabPage)
    {
        _tabPages.Insert(index, tabPage);
 
        _cachedDisplayRect = Rectangle.Empty;
        ApplyItemSize();
        if (Appearance == TabAppearance.FlatButtons)
        {
            Invalidate();
        }
    }
 
    /// <summary>
    ///  This function is used by the Insert Logic to insert a TabPage in the current TabPage in the TabPageCollection.
    /// </summary>
    private void InsertItem(int index, TabPage tabPage)
    {
        ArgumentOutOfRangeException.ThrowIfNegative(index);
        ArgumentOutOfRangeException.ThrowIfGreaterThan(index, TabCount);
        ArgumentNullException.ThrowIfNull(tabPage);
 
        index = IsHandleCreated ? SendMessage(PInvoke.TCM_INSERTITEMW, index, tabPage) : index;
        if (index >= 0)
        {
            Insert(index, tabPage);
        }
    }
 
    /// <summary>
    ///  Handling special input keys, such as pgup, pgdown, home, end, etc...
    /// </summary>
    protected override bool IsInputKey(Keys keyData)
    {
        if ((keyData & Keys.Alt) == Keys.Alt)
        {
            return false;
        }
 
        return (keyData & Keys.KeyCode) switch
        {
            Keys.PageUp or Keys.PageDown or Keys.Home or Keys.End => true,
            _ => base.IsInputKey(keyData),
        };
    }
 
    private static void NotifyAboutFocusState(TabPage? selectedTab, bool focused)
    {
        if (selectedTab is null)
        {
            return;
        }
 
        if (focused)
        {
            KeyboardToolTipStateMachine.Instance.NotifyAboutGotFocus(selectedTab);
        }
        else
        {
            KeyboardToolTipStateMachine.Instance.NotifyAboutLostFocus(selectedTab);
        }
    }
 
    protected override void OnGotFocus(EventArgs e)
    {
        NotifyAboutFocusState(SelectedTab, focused: true);
        base.OnGotFocus(e);
 
        if (IsAccessibilityObjectCreated && SelectedTab is not null)
        {
            SelectedTab.TabAccessibilityObject.RaiseAutomationEvent(UIA_EVENT_ID.UIA_AutomationFocusChangedEventId);
        }
    }
 
    /// <summary>
    ///  This is a notification that the handle has been created.
    ///  We do some work here to configure the handle.
    ///  Overriders should call base.OnHandleCreated()
    /// </summary>
    protected override void OnHandleCreated(EventArgs e)
    {
        if (!IsHandleCreated)
        {
            base.OnHandleCreated(e);
            return;
        }
 
        // Add the handle to hashtable for Ids ..
        _windowId = NativeWindow.CreateWindowId(this);
        _handleInTable = true;
 
        // Set the padding BEFORE setting the control's font (as done
        // in base.OnHandleCreated()) so that the tab control will honor both the
        // horizontal and vertical dimensions of the padding rectangle.
        if (!_padding.IsEmpty)
        {
            PInvokeCore.SendMessage(this, PInvoke.TCM_SETPADDING, 0, PARAM.FromPoint(_padding));
        }
 
        base.OnHandleCreated(e);
        _cachedDisplayRect = Rectangle.Empty;
        ApplyItemSize();
        if (_imageList is not null)
        {
            PInvokeCore.SendMessage(this, PInvoke.TCM_SETIMAGELIST, 0, _imageList.Handle);
        }
 
        if (ShowToolTips)
        {
            HWND tooltipHwnd = (HWND)PInvokeCore.SendMessage(this, PInvoke.TCM_GETTOOLTIPS);
            if (!tooltipHwnd.IsNull)
            {
                PInvoke.SetWindowPos(
                    this,
                    HWND.HWND_TOPMOST,
                    0, 0, 0, 0,
                    SET_WINDOW_POS_FLAGS.SWP_NOMOVE | SET_WINDOW_POS_FLAGS.SWP_NOSIZE | SET_WINDOW_POS_FLAGS.SWP_NOACTIVATE);
            }
        }
 
        // Add the pages
        //
        foreach (TabPage page in TabPages)
        {
            AddNativeTabPage(page);
        }
 
        // Resize the pages
        //
        ResizePages();
 
        if (_selectedIndex != -1)
        {
            try
            {
                SetState(State.FromCreateHandles, true);
                SelectedIndex = _selectedIndex;
            }
            finally
            {
                SetState(State.FromCreateHandles, false);
            }
 
            _selectedIndex = -1;
        }
 
        UpdateTabSelection(false);
    }
 
    protected override void OnHandleDestroyed(EventArgs e)
    {
        if (!Disposing)
        {
            _selectedIndex = SelectedIndex;
        }
 
        // Remove the handle from NativeWindow.
        if (_handleInTable)
        {
            _handleInTable = false;
            NativeWindow.RemoveWindowFromIDTable(_windowId);
        }
 
        base.OnHandleDestroyed(e);
    }
 
    /// <summary>
    ///  Actually goes and fires the OnDrawItem event. Inheriting controls
    ///  should use this to know when the event is fired [this is preferable to
    ///  adding an event handler on yourself for this event]. They should,
    ///  however, remember to call base.onDrawItem(e); to ensure the event is
    ///  still fired to external listeners
    /// </summary>
    protected virtual void OnDrawItem(DrawItemEventArgs e)
    {
        _onDrawItem?.Invoke(this, e);
    }
 
    /// <summary>
    ///  Actually goes and fires the OnLeave event. Inheriting controls
    ///  should use this to know when the event is fired [this is preferable to
    ///  adding an event handler on yourself for this event]. They should,
    ///  however, remember to call base.OnLeave(e); to ensure the event is
    ///  still fired to external listeners
    ///  This listener is overridden so that we can fire SAME ENTER and LEAVE
    ///  events on the TabPage.
    ///  TabPage should fire enter when the focus is on the TABPAGE and not when the control
    ///  within the TabPage gets Focused.
    ///  Similarly the Leave event should fire when the TabControl (and hence the TabPage) looses
    ///  Focus. To be Backward compatible we have added new bool which can be set to true
    ///  to the get the NEW SANE ENTER-LEAVE EVENTS ON THE TABPAGE.
    /// </summary>
    protected internal override void OnEnter(EventArgs e)
    {
        base.OnEnter(e);
        SelectedTab?.FireEnter(e);
    }
 
    /// <summary>
    ///  Actually goes and fires the OnLeave event. Inheriting controls
    ///  should use this to know when the event is fired [this is preferable to
    ///  adding an event handler on yourself for this event]. They should,
    ///  however, remember to call base.OnLeave(e); to ensure the event is
    ///  still fired to external listeners
    ///  This listener is overridden so that we can fire SAME ENTER and LEAVE
    ///  events on the TabPage.
    ///  TabPage should fire enter when the focus is on the TABPAGE and not when the control
    ///  within the TabPage gets Focused.
    ///  Similarly the Leave event should fire when the TabControl (and hence the TabPage) looses
    ///  Focus. To be Backward compatible we have added new bool which can be set to true
    ///  to the get the NEW SANE ENTER-LEAVE EVENTS ON THE TABPAGE.
    /// </summary>
    protected internal override void OnLeave(EventArgs e)
    {
        SelectedTab?.FireLeave(e);
 
        base.OnLeave(e);
    }
 
    protected override void OnLostFocus(EventArgs e)
    {
        NotifyAboutFocusState(SelectedTab, focused: false);
        base.OnLostFocus(e);
    }
 
    /// <summary>
    ///  We override this to get tabbing functionality.
    ///  If overriding this, remember to call base.onKeyDown.
    /// </summary>
    protected override void OnKeyDown(KeyEventArgs ke)
    {
        if (ke.KeyCode == Keys.Tab && (ke.KeyData & Keys.Control) != 0)
        {
            bool forward = (ke.KeyData & Keys.Shift) == 0;
            SelectNextTab(ke, forward);
        }
 
        if (ke.KeyCode == Keys.PageDown && (ke.KeyData & Keys.Control) != 0)
        {
            SelectNextTab(ke, true);
        }
 
        if (ke.KeyCode == Keys.PageUp && (ke.KeyData & Keys.Control) != 0)
        {
            SelectNextTab(ke, false);
        }
 
        base.OnKeyDown(ke);
    }
 
    internal override void OnParentHandleRecreated()
    {
        // Avoid temporarily resizing the TabControl while the parent
        // recreates its handle to avoid
        _skipUpdateSize = true;
        try
        {
            base.OnParentHandleRecreated();
        }
        finally
        {
            _skipUpdateSize = false;
        }
    }
 
    protected override void OnResize(EventArgs e)
    {
        base.OnResize(e);
        _cachedDisplayRect = Rectangle.Empty;
        UpdateTabSelection(false);
    }
 
    [EditorBrowsable(EditorBrowsableState.Advanced)]
    protected virtual void OnRightToLeftLayoutChanged(EventArgs e)
    {
        if (GetAnyDisposingInHierarchy())
        {
            return;
        }
 
        if (RightToLeft == RightToLeft.Yes)
        {
            RecreateHandle();
        }
 
        if (Events[s_rightToLeftLayoutChangedEvent] is EventHandler eh)
        {
            eh(this, e);
        }
    }
 
    /// <summary>
    ///  Actually goes and fires the onSelectedIndexChanged event. Inheriting controls
    ///  should use this to know when the event is fired [this is preferable to
    ///  adding an event handler on yourself for this event]. They should,
    ///  however, remember to call base.onSelectedIndexChanged(e); to ensure the event is
    ///  still fired to external listeners
    /// </summary>
    protected virtual void OnSelectedIndexChanged(EventArgs e)
    {
        int index = SelectedIndex;
        _cachedDisplayRect = Rectangle.Empty;
        UpdateTabSelection(GetState(State.UISelection));
        SetState(State.UISelection, false);
        _onSelectedIndexChanged?.Invoke(this, e);
        KeyboardToolTipStateMachine.Instance.NotifyAboutLostFocus(this);
        NotifyAboutFocusState(SelectedTab, focused: true);
    }
 
    /// <summary>
    ///  Raises the <see cref="OnSelecting"/> event.
    /// </summary>
    protected virtual void OnSelecting(TabControlCancelEventArgs e)
    {
        ((TabControlCancelEventHandler?)Events[s_selectingEvent])?.Invoke(this, e);
    }
 
    /// <summary>
    ///  Raises the <see cref="OnSelected"/> event.
    /// </summary>
    protected virtual void OnSelected(TabControlEventArgs e)
    {
        ((TabControlEventHandler?)Events[s_selectedEvent])?.Invoke(this, e);
 
        // Raise the enter event for this tab.
        SelectedTab?.FireEnter(EventArgs.Empty);
    }
 
    /// <summary>
    ///  Raises the <see cref="OnDeselecting"/> event.
    /// </summary>
    protected virtual void OnDeselecting(TabControlCancelEventArgs e)
    {
        ((TabControlCancelEventHandler?)Events[s_deselectingEvent])?.Invoke(this, e);
    }
 
    /// <summary>
    ///  Raises the <see cref="OnDeselected"/> event.
    /// </summary>
    protected virtual void OnDeselected(TabControlEventArgs e)
    {
        ((TabControlEventHandler?)Events[s_deselectedEvent])?.Invoke(this, e);
 
        // Raise the Leave event for this tab.
        if (SelectedTab is not null)
        {
            NotifyAboutFocusState(SelectedTab, focused: false);
            SelectedTab.FireLeave(EventArgs.Empty);
        }
    }
 
    /// <summary>
    ///  We override this to get the Ctrl and Ctrl-Shift Tab functionality.
    /// </summary>
    protected override bool ProcessKeyPreview(ref Message m)
    {
        if (ProcessKeyEventArgs(ref m))
        {
            return true;
        }
 
        return base.ProcessKeyPreview(ref m);
    }
 
    internal void UpdateSize()
    {
        if (_skipUpdateSize)
        {
            return;
        }
 
        // the spin control (left right arrows) won't update without resizing.
        // the most correct thing would be to recreate the handle, but this works
        // and is cheaper.
        BeginUpdate();
        Size size = Size;
        Size = new Size(size.Width + 1, size.Height);
        Size = size;
        EndUpdate();
    }
 
    protected override void OnFontChanged(EventArgs e)
    {
        base.OnFontChanged(e);
        _cachedDisplayRect = Rectangle.Empty;
        UpdateSize();
    }
 
    internal override void RecreateHandleCore()
    {
        TabPage[] tabPages = GetTabPages();
 
        int index = ((tabPages.Length > 0) && (SelectedIndex == -1)) ? 0 : SelectedIndex;
 
        // We don't actually want to remove the windows forms TabPages - we only
        // want to remove the corresponding TCITEM structs.
        // So, no RemoveAll()
        if (IsHandleCreated)
        {
            PInvokeCore.SendMessage(this, PInvoke.TCM_DELETEALLITEMS);
        }
 
        _tabPages.Clear();
 
        base.RecreateHandleCore();
 
        for (int i = 0; i < tabPages.Length; i++)
        {
            TabPages.Add(tabPages[i]);
        }
 
        try
        {
            SetState(State.FromCreateHandles, true);
            SelectedIndex = index;
        }
        finally
        {
            SetState(State.FromCreateHandles, false);
        }
 
        // The comctl32 TabControl seems to have some painting glitches. Briefly
        // resizing the control seems to fix these.
        UpdateSize();
    }
 
    protected void RemoveAll()
    {
        Controls.Clear();
 
        if (IsHandleCreated)
        {
            PInvokeCore.SendMessage(this, (PInvoke.TCM_DELETEALLITEMS));
        }
 
        _tabPages.Clear();
    }
 
    private void RemoveTabPage(int index)
    {
        ArgumentOutOfRangeException.ThrowIfNegative(index);
        ArgumentOutOfRangeException.ThrowIfGreaterThanOrEqual(index, TabCount);
 
        if (index < _tabPages.Count)
        {
            _tabPages.RemoveAt(index);
        }
 
        if (IsHandleCreated)
        {
            PInvokeCore.SendMessage(this, PInvoke.TCM_DELETEITEM, (WPARAM)index);
        }
 
        _cachedDisplayRect = Rectangle.Empty;
    }
 
    private void ResetItemSize()
    {
        ItemSize = s_defaultItemSize;
    }
 
    private void ResetPadding()
    {
        Padding = s_defaultPaddingPoint;
    }
 
    private void ResizePages()
    {
        Rectangle rect = DisplayRectangle;
        TabPage[] pages = GetTabPages();
        for (int i = 0; i < pages.Length; i++)
        {
            pages[i].Bounds = rect;
        }
    }
 
    /// <summary>
    ///  Called by ToolTip to poke in that Tooltip into this ComCtl so that the Native ChildToolTip is not exposed.
    /// </summary>
    internal override void SetToolTip(ToolTip toolTip)
    {
        if (toolTip is null || !ShowToolTips)
        {
            return;
        }
 
        PInvokeCore.SendMessage(this, PInvoke.TCM_SETTOOLTIPS, (WPARAM)toolTip.Handle);
        GC.KeepAlive(toolTip);
        _controlTipText = toolTip.GetToolTip(this);
    }
 
    private void SetTabPage(int index, TabPage value)
    {
        ArgumentOutOfRangeException.ThrowIfNegative(index);
        ArgumentOutOfRangeException.ThrowIfGreaterThanOrEqual(index, TabCount);
        ArgumentNullException.ThrowIfNull(value);
 
        if (IsHandleCreated)
        {
            SendMessage(PInvoke.TCM_SETITEMW, index, value);
        }
 
        // Make the Updated tab page the currently selected tab page
        if (DesignMode && IsHandleCreated)
        {
            PInvokeCore.SendMessage(this, PInvoke.TCM_SETCURSEL, (WPARAM)index);
        }
 
        _tabPages[index] = value;
    }
 
    /// <summary>
    ///  Allows the user to specify the index in TabControl.TabPageCollection of the tabPage to be shown.
    /// </summary>
    public void SelectTab(int index)
    {
        TabPage t = GetTabPage(index);
        if (t is not null)
        {
            SelectedTab = t;
        }
    }
 
    /// <summary>
    ///  Allows the user to specify the TabPage in TabControl.TabPageCollection to be shown.
    /// </summary>
    public void SelectTab(TabPage tabPage)
    {
        ArgumentNullException.ThrowIfNull(tabPage);
 
        int index = FindTabPage(tabPage);
        SelectTab(index);
    }
 
    /// <summary>
    ///  Allows the user to specify the name of the TabPage in TabControl.TabPageCollection to be shown.
    /// </summary>
    public void SelectTab(string tabPageName)
    {
        ArgumentNullException.ThrowIfNull(tabPageName);
 
        TabPage tabPage = TabPages[tabPageName]!;
        SelectTab(tabPage);
    }
 
    /// <summary>
    ///  This is called by TabControl in response to the KeyDown event to override the selection of tabPages
    ///  for different key combinations.
    ///  Control + Tab selects the next TabPage.
    ///  Control + Shift + Tab selects the previous TabPage.
    ///  Control + PageDown selects the next TabPage.
    ///  Control + PageUp selects the previous TabPage.
    /// </summary>
    private void SelectNextTab(KeyEventArgs ke, bool forward)
    {
        // WmSelChanging actually changes focus to cause validations.
        // So cache in the Focused value so that we can reuse it later
        bool focused = Focused;
 
        // Fire Deselecting .. Deselected on currently selected TabPage...
        if (WmSelChanging())
        {
            SetState(State.UISelection, false);
            return;
        }
 
        if (ValidationCancelled)
        {
            SetState(State.UISelection, false);
            return;
        }
        else
        {
            int sel = SelectedIndex;
            if (sel != -1)
            {
                int count = TabCount;
                if (forward)
                {
                    sel = (sel + 1) % count;
                }
                else
                {
                    sel = (sel + count - 1) % count;
                }
 
                // this is special casing..
                // this function is called from OnKeyDown( ) which selects the NEXT TABPAGE
                // But now we call the WmSelChanging( ) to Focus the tab page
                // This breaks the logic in UpdateTabSelection (which is called
                // thru SET of SelectedIndex) to Select the First control
                // So adding this new Flag to select the first control.
                try
                {
                    SetState(State.UISelection, true);
                    SetState(State.SelectFirstControl, true);
                    SelectedIndex = sel;
                    // This is required so that we select the first control if the TabControl is not current focused.
                    SetState(State.SelectFirstControl, !focused);
                    // Fire Selecting .. Selected on newly selected TabPage...
                    WmSelChange();
                }
                finally
                {
                    // tabControlState[State.SelectFirstControl] can be true if the TabControl is not focused
                    // But at the end of this function reset the state !!
                    SetState(State.SelectFirstControl, false);
                    ke.Handled = true;
                }
            }
        }
    }
 
    // TabControl overrides this method to return true.
    internal override bool ShouldPerformContainerValidation()
    {
        return true;
    }
 
    private bool ShouldSerializeItemSize()
    {
        return !_itemSize.Equals(s_defaultItemSize);
    }
 
    private new bool ShouldSerializePadding()
    {
        return !_padding.Equals(s_defaultPaddingPoint);
    }
 
    /// <summary>
    ///  Returns a string representation for this control.
    /// </summary>
    public override string ToString()
    {
        string s = base.ToString();
        if (TabPages is not null)
        {
            s += $", TabPages.Count: {TabPages.Count}";
            if (TabPages.Count > 0)
            {
                s += $", TabPages[0]: {TabPages[0]}";
            }
        }
 
        return s;
    }
 
    [EditorBrowsable(EditorBrowsableState.Never)]
    protected override void ScaleCore(float dx, float dy)
    {
        _currentlyScaling = true;
        base.ScaleCore(dx, dy);
        _currentlyScaling = false;
    }
 
    /// <summary>
    ///  Set the panel selections appropriately
    /// </summary>
    protected void UpdateTabSelection(bool updateFocus)
    {
        if (IsHandleCreated)
        {
            int index = SelectedIndex;
 
            // make current panel invisible
            TabPage[] tabPages = GetTabPages();
            if (index != -1)
            {
                // Changing the bounds of the TabPage during scaling
                // will force a layout to occur. After this layout
                // the TabPage will then be scaled again resulting
                // in incorrect sizes. Suspend Layout in this case.
                if (_currentlyScaling)
                {
                    tabPages[index].SuspendLayout();
                }
 
                tabPages[index].Bounds = DisplayRectangle;
 
                // After changing the Bounds of TabPages, we need to
                // make TabPages Redraw.
                // Use Invalidate directly here has no performance
                // issue, since ReSize is calling low frequency.
                tabPages[index].Invalidate();
 
                if (_currentlyScaling)
                {
                    tabPages[index].ResumeLayout(false);
                }
 
                tabPages[index].Visible = true;
                if (updateFocus)
                {
                    if (!Focused || GetState(State.SelectFirstControl))
                    {
                        SetState(State.UISelection, false);
                        bool selectNext = tabPages[index].SelectNextControl(null, true, true, false, false);
 
                        if (selectNext)
                        {
                            if (!ContainsFocus)
                            {
                                IContainerControl? c = GetContainerControl();
                                if (c is not null)
                                {
                                    while (c.ActiveControl is ContainerControl)
                                    {
                                        c = (IContainerControl)c.ActiveControl;
                                    }
 
                                    c.ActiveControl?.Focus();
                                }
                            }
                        }
                        else
                        {
                            IContainerControl? c = GetContainerControl();
                            if (c is not null && !DesignMode)
                            {
                                if (c is ContainerControl containerControl)
                                {
                                    containerControl.SetActiveControl(this);
                                }
                                else
                                {
                                    c.ActiveControl = this;
                                }
                            }
                        }
                    }
                }
            }
 
            for (int i = 0; i < tabPages.Length; i++)
            {
                if (i != SelectedIndex)
                {
                    tabPages[i].Visible = false;
                }
            }
        }
    }
 
    protected override void OnStyleChanged(EventArgs e)
    {
        base.OnStyleChanged(e);
        _cachedDisplayRect = Rectangle.Empty;
        UpdateTabSelection(false);
    }
 
    internal void UpdateTab(TabPage tabPage)
    {
        int index = FindTabPage(tabPage);
        SetTabPage(index, tabPage);
 
        // It's possible that changes to this TabPage will change the DisplayRectangle of the
        // TabControl, so invalidate and resize the size of this page.
        //
        _cachedDisplayRect = Rectangle.Empty;
        UpdateTabSelection(false);
    }
 
    private unsafe void WmNeedText(ref Message m)
    {
        NMTTDISPINFOW* ttt = (NMTTDISPINFOW*)(nint)m.LParamInternal;
 
        int commandID = (int)ttt->hdr.idFrom;
 
        string? tipText = GetToolTipText(GetTabPage(commandID));
        if (string.IsNullOrEmpty(tipText))
        {
            tipText = _controlTipText;
        }
 
        _toolTipBuffer.SetText(tipText);
        ttt->lpszText = (char*)_toolTipBuffer.Buffer;
        ttt->hinst = HINSTANCE.Null;
 
        // RightToLeft reading order
        if (RightToLeft == RightToLeft.Yes)
        {
            ttt->uFlags |= TOOLTIP_FLAGS.TTF_RTLREADING;
        }
    }
 
    private unsafe void WmReflectDrawItem(ref Message m)
    {
        DRAWITEMSTRUCT* dis = (DRAWITEMSTRUCT*)(nint)m.LParamInternal;
 
        using DrawItemEventArgs e = new(
            dis->hDC,
            Font,
            dis->rcItem,
            dis->itemID,
            dis->itemState,
            ForeColor,
            BackColor);
 
        OnDrawItem(e);
 
        m.ResultInternal = (LRESULT)1;
    }
 
    private bool WmSelChange()
    {
        TabControlCancelEventArgs tcc = new(SelectedTab, SelectedIndex, false, TabControlAction.Selecting);
        OnSelecting(tcc);
        if (!tcc.Cancel)
        {
            OnSelected(new TabControlEventArgs(SelectedTab, SelectedIndex, TabControlAction.Selected));
            OnSelectedIndexChanged(EventArgs.Empty);
 
            if (IsAccessibilityObjectCreated && SelectedTab?.ParentInternal is TabControl)
            {
                SelectedTab.TabAccessibilityObject.RaiseAutomationEvent(UIA_EVENT_ID.UIA_SelectionItem_ElementSelectedEventId);
                BeginInvoke((MethodInvoker)(() =>
                {
                    if (IsAccessibilityObjectCreated && SelectedTab?.ParentInternal is TabControl &&
                         !SelectedTab.IsDisposed && SelectedTab.TabAccessibilityObject is not null)
                    {
                        SelectedTab.TabAccessibilityObject.RaiseAutomationEvent(UIA_EVENT_ID.UIA_AutomationFocusChangedEventId);
                    }
                }));
            }
        }
        else
        {
            // user Cancelled the Selection of the new Tab.
            PInvokeCore.SendMessage(this, PInvoke.TCM_SETCURSEL, (WPARAM)_lastSelection);
            UpdateTabSelection(true);
        }
 
        return tcc.Cancel;
    }
 
    private bool WmSelChanging()
    {
        IContainerControl? c = GetContainerControl();
        if (c is not null && !DesignMode)
        {
            if (c is ContainerControl containerControl)
            {
                containerControl.SetActiveControl(this);
            }
            else
            {
                c.ActiveControl = this;
            }
        }
 
        // Fire DeSelecting .... on the current Selected Index...
        // Set the return value to a global
        // if 'cancelled' return from here else..
        // fire Deselected.
        _lastSelection = SelectedIndex;
        TabControlCancelEventArgs tcc = new(SelectedTab, SelectedIndex, false, TabControlAction.Deselecting);
        OnDeselecting(tcc);
        if (!tcc.Cancel)
        {
            OnDeselected(new TabControlEventArgs(SelectedTab, SelectedIndex, TabControlAction.Deselected));
        }
 
        return tcc.Cancel;
    }
 
    private unsafe void WmTabBaseReLayout()
    {
        BeginUpdate();
        _cachedDisplayRect = Rectangle.Empty;
        UpdateTabSelection(false);
        EndUpdate();
        Invalidate(true);
 
        // Remove other TabBaseReLayout messages from the message queue
        MSG msg = default;
        while (PInvokeCore.PeekMessage(
            &msg,
            this,
            (uint)_tabBaseReLayoutMessage,
            (uint)_tabBaseReLayoutMessage,
            PEEK_MESSAGE_REMOVE_TYPE.PM_REMOVE))
        {
        }
    }
 
    /// <summary>
    ///  The tab's window procedure. Inheriting classes can override this
    ///  to add extra functionality, but should not forget to call
    ///  base.wndProc(m); to ensure the tab continues to function properly.
    /// </summary>
    protected override unsafe void WndProc(ref Message m)
    {
        switch (m.MsgInternal)
        {
            case MessageId.WM_REFLECT_DRAWITEM:
                WmReflectDrawItem(ref m);
                break;
 
            case MessageId.WM_REFLECT_MEASUREITEM:
                // We use TCM_SETITEMSIZE instead
                break;
 
            case PInvokeCore.WM_NOTIFY:
            case MessageId.WM_REFLECT_NOTIFY:
                NMHDR* nmhdr = (NMHDR*)(nint)m.LParamInternal;
                switch (nmhdr->code)
                {
                    // new switch added to prevent the TabControl from changing to next TabPage ...
                    // in case of validation cancelled...
                    // Turn  tabControlState[State.UISelection] = false and Return So that no WmSelChange() gets fired.
                    // If validation not cancelled then tabControlState[State.UISelection] is turned ON to set the focus on to the ...
                    // next TabPage..
 
                    case PInvoke.TCN_SELCHANGING:
                        if (WmSelChanging())
                        {
                            m.ResultInternal = (LRESULT)1;
                            SetState(State.UISelection, false);
                            return;
                        }
 
                        if (ValidationCancelled)
                        {
                            m.ResultInternal = (LRESULT)1;
                            SetState(State.UISelection, false);
                            return;
                        }
                        else
                        {
                            SetState(State.UISelection, true);
                        }
 
                        break;
                    case PInvoke.TCN_SELCHANGE:
                        if (WmSelChange())
                        {
                            m.ResultInternal = (LRESULT)1;
                            SetState(State.UISelection, false);
                            return;
                        }
                        else
                        {
                            SetState(State.UISelection, true);
                        }
 
                        break;
                    case PInvoke.TTN_GETDISPINFOW:
                        // Setting the max width has the added benefit of enabling Multiline tool tips
                        PInvokeCore.SendMessage(nmhdr->hwndFrom, PInvoke.TTM_SETMAXTIPWIDTH, 0, SystemInformation.MaxWindowTrackSize.Width);
                        WmNeedText(ref m);
                        m.ResultInternal = (LRESULT)1;
                        return;
                }
 
                break;
        }
 
        if (m.MsgInternal == _tabBaseReLayoutMessage)
        {
            WmTabBaseReLayout();
            return;
        }
 
        base.WndProc(ref m);
    }
 
    private bool GetState(State state) => _tabControlState[(int)state];
 
    private void SetState(State state, bool value) => _tabControlState[(int)state] = value;
 
    private unsafe int SendMessage(uint msg, int wParam, TabPage tabPage)
    {
        TCITEMW tcitem = default;
        string text = tabPage.Text;
        PrefixAmpersands(ref text);
        if (text is not null)
        {
            tcitem.mask |= TCITEMHEADERA_MASK.TCIF_TEXT;
            tcitem.cchTextMax = text.Length;
        }
 
        int imageIndex = tabPage.ImageIndex;
        tcitem.mask |= TCITEMHEADERA_MASK.TCIF_IMAGE;
        tcitem.iImage = tabPage.ImageIndexer.ActualIndex;
 
        fixed (char* pText = text)
        {
            tcitem.pszText = pText;
            return (int)PInvokeCore.SendMessage(this, msg, (WPARAM)wParam, ref tcitem);
        }
    }
 
    private static void PrefixAmpersands(ref string value)
    {
        // Due to a comctl32 problem, ampersands underline the next letter in the
        // text string, but the accelerators don't work.
        // So in this function, we prefix ampersands with another ampersand
        // so that they actually appear as ampersands.
        if (string.IsNullOrEmpty(value))
        {
            return;
        }
 
        // If there are no ampersands, we don't need to do anything here
        int firstAmpersand = value.IndexOf('&');
        if (firstAmpersand < 0)
        {
            return;
        }
 
        // Insert extra ampersands
        StringBuilder newString = new();
        newString.Append(value.AsSpan(0, firstAmpersand));
        for (int i = firstAmpersand; i < value.Length; ++i)
        {
            if (value[i] == '&')
            {
                if (i < value.Length - 1 && value[i + 1] == '&')
                {
                    // Skip the second ampersand
                    ++i;
                }
 
                newString.Append("&&");
            }
            else
            {
                newString.Append(value[i]);
            }
        }
 
        value = newString.ToString();
    }
}