File: System\Windows\Forms\Design\DocumentDesigner.cs
Web Access
Project: src\src\System.Windows.Forms.Design\src\System.Windows.Forms.Design.csproj (System.Windows.Forms.Design)
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
 
#nullable disable
 
using System.Collections;
using System.ComponentModel;
using System.ComponentModel.Design;
using System.ComponentModel.Design.Serialization;
using System.Drawing;
using System.Drawing.Design;
using System.Windows.Forms.Design.Behavior;
using Microsoft.Win32;
 
namespace System.Windows.Forms.Design;
 
/// <summary>
///  Provides a designer that extends the ScrollableControlDesigner and implements
///  IRootDesigner.
/// </summary>
[ToolboxItemFilter("System.Windows.Forms")]
public partial class DocumentDesigner : ScrollableControlDesigner, IRootDesigner, IToolboxUser, IOleDragClient
{
    private DesignerFrame _frame;
    private ControlCommandSet _commandSet;
    private InheritanceService _inheritanceService;
    private EventHandlerService _eventHandlerService;
    private DesignBindingValueUIHandler _designBindingValueUIHandler;
    private BehaviorService _behaviorService;
    private SelectionManager _selectionManager;
    private DesignerExtenders _designerExtenders;
    private InheritanceUI _inheritanceUI;
    private PbrsForward _pbrsFwd;
    private List<Control> _suspendedComponents;
    private UndoEngine _undoEngine;
    private bool _initializing;   // is the designer initializing?
 
    // Used to keep the state of the tab order view
    private bool _queriedTabOrder;
    private MenuCommand _tabOrderCommand;
 
    internal static IDesignerSerializationManager s_manager;
 
    // The component tray
    private ComponentTray _componentTray;
 
    private int _trayHeight = 80;
    private bool _trayLargeIcon;
    private bool _trayAutoArrange;
    private bool _trayLayoutSuspended;
 
    // ActiveX support
    private static readonly Guid s_htmlDesignTime = new("73CEF3DD-AE85-11CF-A406-00AA00C00940");
 
    private Dictionary<string, AxToolboxItem> _axTools;
    private const string AxClipFormat = "CLSID";
    private ToolboxItemCreatorCallback _toolboxCreator;
 
    /// <summary>
    ///  Property shadow for ContainerControl's AutoScaleDimensions. We shadow here so it
    ///  always returns the CurrentAutoScaleDimensions for the control. This way the control's
    ///  state always adapts to the current font / monitor.
    /// </summary>
    private SizeF AutoScaleDimensions
    {
        get
        {
            ContainerControl c = Control as ContainerControl;
            if (c is not null)
            {
                return c.CurrentAutoScaleDimensions;
            }
 
            Debug.Fail("AutoScaleDimensions should not be shadowed on non-ContainerControl objects.");
            return SizeF.Empty;
        }
        set
        {
            ContainerControl c = Control as ContainerControl;
            if (c is not null)
            {
                c.AutoScaleDimensions = value;
            }
        }
    }
 
    /// <summary>
    ///  Property shadow for ContainerControl's AutoScaleMode. We shadow here so it
    ///  never gets to the control; it can be very distracting if you change the font
    ///  and have the form you're designing suddenly move on you.
    /// </summary>
    private AutoScaleMode AutoScaleMode
    {
        get
        {
            ContainerControl c = Control as ContainerControl;
            if (c is not null)
            {
                return c.AutoScaleMode;
            }
 
            Debug.Fail("AutoScaleMode should not be shadowed on non-ContainerControl objects.");
            return AutoScaleMode.Inherit;
        }
        set
        {
            ShadowProperties[nameof(AutoScaleMode)] = value;
            ContainerControl c = Control as ContainerControl;
            if (c is not null && c.AutoScaleMode != value)
            {
                c.AutoScaleMode = value;
 
                // If we're not loading and this changes update
                // the current auto scale dimensions.
                IDesignerHost host = GetService(typeof(IDesignerHost)) as IDesignerHost;
                if (host is not null && !host.Loading)
                {
                    c.AutoScaleDimensions = c.CurrentAutoScaleDimensions;
                }
            }
        }
    }
 
    /// <summary>
    ///  BackColor property on control. We shadow this property at design time.
    /// </summary>
    private Color BackColor
    {
        get
        {
            return Control.BackColor;
        }
        set
        {
            ShadowProperties[nameof(BackColor)] = value;
            if (value.IsEmpty)
            {
                value = SystemColors.Control;
            }
 
            Control.BackColor = value;
        }
    }
 
    /// <summary>
    ///  Location property on control. We shadow this property at design time.
    /// </summary>
    [DefaultValue(typeof(Point), "0, 0")]
    private Point Location
    {
        get
        {
            return (Point)ShadowProperties[nameof(Location)];
        }
        set
        {
            ShadowProperties[nameof(Location)] = value;
        }
    }
 
    /// <summary>
    ///  We override our selection rules to make the document non-sizeable.
    /// </summary>
    public override SelectionRules SelectionRules
    {
        get
        {
            SelectionRules rules = base.SelectionRules;
            rules &= ~(SelectionRules.Moveable | SelectionRules.TopSizeable | SelectionRules.LeftSizeable);
 
            return rules;
        }
    }
 
    /// <summary>
    ///  Determines if the tab order UI is active. When tab order is active, we don't want to forward
    ///  any WndProc messages to the menu editor service (those are all non-selectable components)
    /// </summary>
    private bool TabOrderActive
    {
        get
        {
            if (!_queriedTabOrder)
            {
                _queriedTabOrder = true;
                IMenuCommandService menuCommandService = (IMenuCommandService)GetService(typeof(IMenuCommandService));
                if (menuCommandService is not null)
                    _tabOrderCommand = menuCommandService.FindCommand(StandardCommands.TabOrder);
            }
 
            return _tabOrderCommand is not null && _tabOrderCommand.Checked;
        }
    }
 
    [DefaultValue(true)]
    private bool TrayAutoArrange
    {
        get => _trayAutoArrange;
        set
        {
            _trayAutoArrange = value;
            if (_componentTray is not null)
            {
                _componentTray.AutoArrange = _trayAutoArrange;
            }
        }
    }
 
    [DefaultValue(false)]
    private bool TrayLargeIcon
    {
        get => _trayLargeIcon;
        set
        {
            _trayLargeIcon = value;
            if (_componentTray is not null)
            {
                _componentTray.ShowLargeIcons = _trayLargeIcon;
            }
        }
    }
 
    [DefaultValue(80)]
    private int TrayHeight
    {
        get => _componentTray is not null ? _componentTray.Height : _trayHeight;
        set
        {
            _trayHeight = value;
            if (_componentTray is not null)
            {
                _componentTray.Height = _trayHeight;
            }
        }
    }
 
    /// <internalonly/>
    /// <summary>
    ///  Retrieves the control view instance for the given component.
    ///  For Win32 designer, this will often be the component itself.
    /// </summary>
    Control IOleDragClient.GetControlForComponent(object component)
    {
        Control c = GetControl(component);
        return c ?? (_componentTray is not null ? ((IOleDragClient)_componentTray).GetControlForComponent(component) : null);
    }
 
    internal virtual bool CanDropComponents(DragEventArgs de)
    {
        // If there is no tray we bail.
        if (_componentTray is null)
        {
            return true;
        }
 
        // Figure out if any of the components in the drag-drop are children
        // of our own tray. If so, we should prevent this drag-drop from proceeding.
 
        // Keeping GetOleDragHandler() for compat.
        _ = GetOleDragHandler();
 
        object[] dragComps = OleDragDropHandler.GetDraggingObjects(de);
 
        if (dragComps is not null)
        {
            IDesignerHost host = (IDesignerHost)GetService(typeof(IDesignerHost));
            for (int i = 0; i < dragComps.Length; i++)
            {
                if (host is null || dragComps[i] is null || dragComps[i] is not IComponent comp)
                {
                    continue;
                }
 
                if (_componentTray.IsTrayComponent(comp))
                {
                    return false;
                }
            }
        }
 
        // ToolStripItems cannot be dropped on any ParentControlDesigners since they have custom DataObject Format.
        return de.Data is not ToolStripItemDataObject;
    }
 
    private AxToolboxItem CreateAxToolboxItem(IDataObject dataObject)
    {
        // Read the stream out of the dataobject and get hold of the CLSID of the Toolbox item.
        MemoryStream stm = (MemoryStream)dataObject.GetData(AxClipFormat, true);
        int len = (int)stm.Length;
        byte[] bytes = new byte[len];
        stm.Read(bytes, 0, len);
 
        string clsid = Text.Encoding.Default.GetString(bytes);
        int index = clsid.IndexOf('}');
        clsid = clsid[..(index + 1)];
 
        // Look to see if we can find the Control key for this CLSID. If present, create a
        // new AxToolboxItem and add it to the cache.
        if (!IsSupportedActiveXControl(clsid))
        {
            return null;
        }
 
        // Look to see if we have already cached the ToolboxItem.
        if (_axTools is not null && _axTools.TryGetValue(clsid, out AxToolboxItem tool))
        {
            return tool;
        }
 
        // Create a new AxToolboxItem and add it to the cache.
        tool = new AxToolboxItem(clsid);
        _axTools ??= [];
        _axTools[clsid] = tool;
        return tool;
    }
 
    private static ToolboxItem CreateCfCodeToolboxItem(IDataObject dataObject)
    {
        object serializationData = dataObject.GetData(OleDragDropHandler.NestedToolboxItemFormat, false);
        if (serializationData is not null)
        {
            return (ToolboxItem)serializationData;
        }
 
        serializationData = dataObject.GetData(OleDragDropHandler.DataFormat, false);
        if (serializationData is not null)
        {
            return new OleDragDropHandler.CfCodeToolboxItem(serializationData);
        }
 
        return null;
    }
 
    /// <summary>
    ///  Disposes of this designer.
    /// </summary>
    protected override void Dispose(bool disposing)
    {
        if (disposing)
        {
            IDesignerHost host = (IDesignerHost)GetService(typeof(IDesignerHost));
            Debug.Assert(host is not null, "Must have a designer host on dispose");
 
            if (host is not null)
            {
                // Remove Adorner Window which hosts DropDowns.
                ToolStripAdornerWindowService toolWindow = (ToolStripAdornerWindowService)GetService(typeof(ToolStripAdornerWindowService));
                if (toolWindow is not null)
                {
                    toolWindow.Dispose();
                    host.RemoveService<ToolStripAdornerWindowService>();
                }
 
                host.Activated -= OnDesignerActivate;
                host.Deactivated -= OnDesignerDeactivate;
 
                // If the tray wasn't destroyed, then we got some sort of imbalance
                // in our add/remove calls. Don't sweat it, but do remove the tray.
                //
                if (_componentTray is not null)
                {
                    ISplitWindowService sws = (ISplitWindowService)GetService(typeof(ISplitWindowService));
                    if (sws is not null)
                    {
                        sws.RemoveSplitWindow(_componentTray);
                        _componentTray.Dispose();
                        _componentTray = null;
                    }
 
                    host.RemoveService<ComponentTray>();
                }
 
                IComponentChangeService cs = (IComponentChangeService)GetService(typeof(IComponentChangeService));
                if (cs is not null)
                {
                    cs.ComponentAdded -= OnComponentAdded;
                    cs.ComponentChanged -= OnComponentChanged;
                    cs.ComponentRemoved -= OnComponentRemoved;
                }
 
                if (_undoEngine is not null)
                {
                    _undoEngine.Undoing -= OnUndoing;
                    _undoEngine.Undone -= OnUndone;
                }
 
                if (_toolboxCreator is not null)
                {
                    IToolboxService toolbox = (IToolboxService)GetService(typeof(IToolboxService));
                    Debug.Assert(toolbox is not null, "No toolbox service available");
                    if (toolbox is not null)
                    {
                        toolbox.RemoveCreator(AxClipFormat, host);
                        toolbox.RemoveCreator(OleDragDropHandler.DataFormat, host);
                        toolbox.RemoveCreator(OleDragDropHandler.NestedToolboxItemFormat, host);
                    }
 
                    _toolboxCreator = null;
                }
            }
 
            ISelectionService ss = (ISelectionService)GetService(typeof(ISelectionService));
            if (ss is not null)
            {
                ss.SelectionChanged -= OnSelectionChanged;
            }
 
            if (_behaviorService is not null)
            {
                _behaviorService.Dispose();
                _behaviorService = null;
            }
 
            if (_selectionManager is not null)
            {
                _selectionManager.Dispose();
                _selectionManager = null;
            }
 
            if (_componentTray is not null)
            {
                if (host is not null)
                {
                    ISplitWindowService sws = (ISplitWindowService)GetService(typeof(ISplitWindowService));
                    sws?.RemoveSplitWindow(_componentTray);
                }
 
                _componentTray.Dispose();
                _componentTray = null;
            }
 
            if (_pbrsFwd is not null)
            {
                _pbrsFwd.Dispose();
                _pbrsFwd = null;
            }
 
            if (_frame is not null)
            {
                _frame.Dispose();
                _frame = null;
            }
 
            if (_commandSet is not null)
            {
                _commandSet.Dispose();
                _commandSet = null;
            }
 
            if (_inheritanceService is not null)
            {
                _inheritanceService.Dispose();
                _inheritanceService = null;
            }
 
            if (_inheritanceUI is not null)
            {
                _inheritanceUI.Dispose();
                _inheritanceUI = null;
            }
 
            if (_designBindingValueUIHandler is not null)
            {
                IPropertyValueUIService pvUISvc = (IPropertyValueUIService)GetService(typeof(IPropertyValueUIService));
                if (pvUISvc is not null)
                {
                    pvUISvc.RemovePropertyValueUIHandler(new PropertyValueUIHandler(_designBindingValueUIHandler.OnGetUIValueItem));
                    pvUISvc = null;
                }
 
                _designBindingValueUIHandler.Dispose();
                _designBindingValueUIHandler = null;
            }
 
            if (_designerExtenders is not null)
            {
                _designerExtenders.Dispose();
                _designerExtenders = null;
            }
 
            _axTools?.Clear();
 
            if (host is not null)
            {
                host.RemoveService<BehaviorService>();
                host.RemoveService<ToolStripAdornerWindowService>();
                host.RemoveService<SelectionManager>();
                host.RemoveService<IInheritanceService>();
                host.RemoveService<IEventHandlerService>();
                host.RemoveService<IOverlayService>();
                host.RemoveService<ISplitWindowService>();
                host.RemoveService<InheritanceUI>();
            }
        }
 
        base.Dispose(disposing);
    }
 
    /// <summary>
    ///  Returns an array of Glyph objects representing the selection
    ///  borders and grab handles for the related Component. Note that
    ///  based on 'selType' the Glyphs returned will either: represent
    ///  a fully resizeable selection border with grab handles, a locked
    ///  selection border, or a single 'hidden' selection Glyph.
    /// </summary>
    public override GlyphCollection GetGlyphs(GlyphSelectionType selectionType)
    {
        GlyphCollection glyphs = [];
 
        if (selectionType != GlyphSelectionType.NotSelected)
        {
            Point loc = BehaviorService.ControlToAdornerWindow((Control)Component);
            Rectangle translatedBounds = new(loc, ((Control)Component).Size);
            bool primarySelection = (selectionType == GlyphSelectionType.SelectedPrimary);
 
            bool locked = false;
            PropertyDescriptor prop = TypeDescriptor.GetProperties(Component)["Locked"];
            if (prop is not null)
            {
                locked = (bool)prop.GetValue(Component);
            }
 
            bool autoSize = false;
            prop = TypeDescriptor.GetProperties(Component)["AutoSize"];
            if (prop is not null)
            {
                autoSize = (bool)prop.GetValue(Component);
            }
 
            AutoSizeMode mode = AutoSizeMode.GrowOnly;
            prop = TypeDescriptor.GetProperties(Component)["AutoSizeMode"];
            if (prop is not null)
            {
                mode = (AutoSizeMode)prop.GetValue(Component);
            }
 
            SelectionRules rules = SelectionRules;
 
            if (locked)
            {
                // the lock glyph
                glyphs.Add(new LockedHandleGlyph(translatedBounds, primarySelection));
 
                // the four locked border glyphs
                glyphs.Add(new LockedBorderGlyph(translatedBounds, SelectionBorderGlyphType.Top));
                glyphs.Add(new LockedBorderGlyph(translatedBounds, SelectionBorderGlyphType.Bottom));
                glyphs.Add(new LockedBorderGlyph(translatedBounds, SelectionBorderGlyphType.Left));
                glyphs.Add(new LockedBorderGlyph(translatedBounds, SelectionBorderGlyphType.Right));
            }
 
            // we check if the control is a form because we only want to disable resizing
            // for components that are not Forms.
            else if (autoSize && (mode == AutoSizeMode.GrowAndShrink) && !(Control is Form))
            {
                // the non-resizeable grab handle
                glyphs.Add(new NoResizeHandleGlyph(translatedBounds, rules, primarySelection, null));
 
                // the four resizeable border glyphs
                glyphs.Add(new NoResizeSelectionBorderGlyph(translatedBounds, rules, SelectionBorderGlyphType.Top, null));
                glyphs.Add(new NoResizeSelectionBorderGlyph(translatedBounds, rules, SelectionBorderGlyphType.Bottom, null));
                glyphs.Add(new NoResizeSelectionBorderGlyph(translatedBounds, rules, SelectionBorderGlyphType.Left, null));
                glyphs.Add(new NoResizeSelectionBorderGlyph(translatedBounds, rules, SelectionBorderGlyphType.Right, null));
            }
 
            else
            {
                glyphs.Add(new GrabHandleGlyph(translatedBounds, GrabHandleGlyphType.MiddleRight, StandardBehavior, primarySelection));
                glyphs.Add(new GrabHandleGlyph(translatedBounds, GrabHandleGlyphType.LowerRight, StandardBehavior, primarySelection));
                glyphs.Add(new GrabHandleGlyph(translatedBounds, GrabHandleGlyphType.MiddleBottom, StandardBehavior, primarySelection));
                glyphs.Add(new SelectionBorderGlyph(translatedBounds, rules, SelectionBorderGlyphType.Top, null));
                glyphs.Add(new SelectionBorderGlyph(translatedBounds, rules, SelectionBorderGlyphType.Bottom, StandardBehavior));
                glyphs.Add(new SelectionBorderGlyph(translatedBounds, rules, SelectionBorderGlyphType.Left, null));
                glyphs.Add(new SelectionBorderGlyph(translatedBounds, rules, SelectionBorderGlyphType.Right, StandardBehavior));
            }
        }
 
        return glyphs;
    }
 
    /// <summary>
    ///  Examines the current selection for a suitable frame designer. This
    ///  is used when we are creating a new component so we know what control
    ///  to parent the component to. This will always return a frame designer,
    ///  and may walk all the way up the control parent chain to this designer
    ///  if it needs to.
    /// </summary>
    private ParentControlDesigner GetSelectedParentControlDesigner()
    {
        ISelectionService s = (ISelectionService)GetService(typeof(ISelectionService));
        ParentControlDesigner parentControlDesigner = null;
 
        if (s is not null)
        {
            // We first try the primary selection. If that is null
            // or isn't a Control, we then walk the set of selected
            // objects. Failing all of this, we default to us.
            //
            object sel = s.PrimarySelection;
 
            if (sel is null || !(sel is Control))
            {
                sel = null;
 
                ICollection sels = s.GetSelectedComponents();
 
                foreach (object obj in sels)
                {
                    if (obj is Control)
                    {
                        sel = obj;
                        break;
                    }
                }
            }
 
            if (sel is not null)
            {
                // Now that we have our currently selected component
                // we can walk up the parent chain looking for a frame
                // designer.
                //
                Debug.Assert(sel is Control, "Our logic is flawed - sel should be a Control");
                Control c = (Control)sel;
 
                IDesignerHost host = (IDesignerHost)GetService(typeof(IDesignerHost));
 
                if (host is not null)
                {
                    while (c is not null)
                    {
                        ParentControlDesigner designer = host.GetDesigner(c) as ParentControlDesigner;
 
                        if (designer is not null)
                        {
                            parentControlDesigner = designer;
                            break;
                        }
 
                        c = c.Parent;
                    }
                }
            }
        }
 
        parentControlDesigner ??= this;
 
        return parentControlDesigner;
    }
 
    /// <summary>
    ///  Determines if the given tool is supported by this designer.
    ///  If a tool is supported then it will be enabled in the toolbox
    ///  when this designer regains focus. Otherwise, it will be disabled.
    ///  Once a tool is marked as enabled or disabled it may not be
    ///  queried again.
    /// </summary>
    protected virtual bool GetToolSupported(ToolboxItem tool)
    {
        return true;
    }
 
    /// <summary>
    ///  Initializes the designer with the given component. The designer can
    ///  get the component's site and request services from it in this call.
    /// </summary>
    public override void Initialize(IComponent component)
    {
        _initializing = true;
        base.Initialize(component);
        _initializing = false;
 
        // If the back color of the control has not been established, force it to be
        // "Control" so it doesn't walk up the parent chain and grab the document window's
        // back color.
        //
        PropertyDescriptor backProp = TypeDescriptor.GetProperties(Component.GetType())["BackColor"];
        if (backProp is not null && backProp.PropertyType == typeof(Color) && !backProp.ShouldSerializeValue(Component))
        {
            Control.BackColor = SystemColors.Control;
        }
 
        IDesignerHost host = (IDesignerHost)GetService(typeof(IDesignerHost));
        IExtenderProviderService exps = (IExtenderProviderService)GetService(typeof(IExtenderProviderService));
        if (exps is not null)
        {
            _designerExtenders = new DesignerExtenders(exps);
        }
 
        if (host is not null)
        {
            host.Activated += OnDesignerActivate;
            host.Deactivated += OnDesignerDeactivate;
 
            ServiceCreatorCallback callback = new(OnCreateService);
            host.AddService<IEventHandlerService>(callback);
 
            // M3.2 CONTROL ARRAY IS CUT
 
            // Create the view for this component. We first create the designer frame so we can provide
            // the overlay and split window services, and then later on we initialize the frame with
            // the designer view.
            _frame = new DesignerFrame(component.Site);
 
            IOverlayService os = _frame;
            host.AddService(os);
            host.AddService<ISplitWindowService>(_frame);
 
            _behaviorService = new BehaviorService(Component.Site, _frame);
            host.AddService(_behaviorService);
 
            _selectionManager = new SelectionManager(host, _behaviorService);
 
            host.AddService(_selectionManager);
            host.AddService<ToolStripAdornerWindowService>(callback);
 
            // And component add and remove events for our tray
            //
            IComponentChangeService cs = (IComponentChangeService)GetService(typeof(IComponentChangeService));
            if (cs is not null)
            {
                cs.ComponentAdded += OnComponentAdded;
                cs.ComponentChanged += OnComponentChanged;
                cs.ComponentRemoved += OnComponentRemoved;
            }
 
            // We must do the inheritance scan early, but not so early that we haven't hooked events
            // to handle invisible components. We also use the variable "inheritanceService"
            // as a check in OnCreateHandle -- we cannot call base.OnCreateHandle if we have
            // not done an inheritance scan yet, because this will cause the base control
            // class to hook all of the controls we may want to inherit. So, we do the
            // scan, assign the variable, and then call OnCreateHandle if needed.
 
            _inheritanceUI = new InheritanceUI();
            host.AddService(_inheritanceUI);
 
            InheritanceService isvc = new DocumentInheritanceService(this);
            host.AddService<IInheritanceService>(isvc);
 
            s_manager = host.GetService(typeof(IDesignerSerializationManager)) as IDesignerSerializationManager;
            isvc.AddInheritedComponents(component, component.Site.Container);
            s_manager = null;
 
            _inheritanceService = isvc;
            if (Control.IsHandleCreated)
            {
                OnCreateHandle();
            }
 
            IPropertyValueUIService pvUISvc = (IPropertyValueUIService)component.Site.GetService(typeof(IPropertyValueUIService));
 
            if (pvUISvc is not null)
            {
                _designBindingValueUIHandler = new DesignBindingValueUIHandler();
                pvUISvc.AddPropertyValueUIHandler(new PropertyValueUIHandler(_designBindingValueUIHandler.OnGetUIValueItem));
            }
 
            // Add the DocumentDesigner as a creator of CLSID toolbox items.
            IToolboxService toolbox = (IToolboxService)host.GetService(typeof(IToolboxService));
            if (toolbox is not null)
            {
                _toolboxCreator = new ToolboxItemCreatorCallback(OnCreateToolboxItem);
                toolbox.AddCreator(_toolboxCreator, AxClipFormat, host);
                toolbox.AddCreator(_toolboxCreator, OleDragDropHandler.DataFormat, host);
                toolbox.AddCreator(_toolboxCreator, OleDragDropHandler.NestedToolboxItemFormat, host);
            }
 
            // Listen for the completed load. When finished, we need to select the form. We don'
            // want to do it before we're done, however, or else the dimensions of the selection rectangle
            // could be off because during load, change events are not fired.
            host.LoadComplete += OnLoadComplete;
        }
 
        // Setup our menu command structure.
        Debug.Assert(component.Site is not null, "Designer host should have given us a site by now.");
        _commandSet = new ControlCommandSet(component.Site);
 
        // Finally hook the designer view into the frame. We do this last because the frame may
        // cause the control to be created, and if this happens before the inheritance scan we
        // will subclass the inherited controls before we get a chance to attach designers.
        _frame.Initialize(Control);
        _pbrsFwd = new PbrsForward(_frame, component.Site);
 
        // And force some shadow properties that we change in the course of
        // initializing the form.
        Location = new Point(0, 0);
    }
 
    /// <summary>
    ///  Checks to see if the give CLSID is an ActiveX control
    ///  that we support. This ignores designtime controls.
    /// </summary>
    private static bool IsSupportedActiveXControl(string clsid)
    {
        RegistryKey key = null;
        RegistryKey designtimeKey = null;
 
        try
        {
            string controlKey = $"CLSID\\{clsid}\\Control";
            key = Registry.ClassesRoot.OpenSubKey(controlKey);
            if (key is not null)
            {
                // ASURT 36817:
                // We are not going to support design-time controls for this revision. We use the guids under
                // HKCR\Component Categories to decide which categories to avoid. Currently the only one is
                // the "HTML Design-time Control" category implemented by VID controls.
                //
                string category = $"CLSID\\{clsid}\\Implemented Categories\\{{{s_htmlDesignTime}}}";
                designtimeKey = Registry.ClassesRoot.OpenSubKey(category);
                return (designtimeKey is null);
            }
 
            return false;
        }
        finally
        {
            key?.Close();
 
            designtimeKey?.Close();
        }
    }
 
    private void OnUndone(object source, EventArgs e)
    {
        // resume all suspended comps we found earlier
        if (_suspendedComponents is not null)
        {
            foreach (Control c in _suspendedComponents)
            {
                c.ResumeLayout(false/*performLayout*/);
                c.PerformLayout();
            }
        }
    }
 
    private void OnUndoing(object source, EventArgs e)
    {
        // attempt to suspend all components within the icontainer
        // plus the root component's parent.
        if (GetService(typeof(IDesignerHost)) is IDesignerHost host)
        {
            IContainer container = host.Container;
            if (container is not null)
            {
                _suspendedComponents = new(container.Components.Count + 1);
 
                foreach (IComponent comp in container.Components)
                {
                    if (comp is Control control)
                    {
                        control.SuspendLayout();
                        // add this control to our suspended components list so we can resume
                        // later
                        _suspendedComponents.Add(control);
                    }
                }
 
                // Also suspend the root component's parent.
                if (host.RootComponent is Control root)
                {
                    Control rootParent = root.Parent;
                    if (rootParent is not null)
                    {
                        rootParent.SuspendLayout();
                        _suspendedComponents.Add(rootParent);
                    }
                }
            }
        }
    }
 
    /// <summary>
    ///  Called when a component is added to the design container.
    ///  If the component isn't a control, this will demand create
    ///  the component tray and add the component to it.
    /// </summary>
    private void OnComponentAdded(object source, ComponentEventArgs ce)
    {
        if (!TryGetService(out IDesignerHost host))
        {
            return;
        }
 
        IComponent component = ce.Component;
 
        // This is the mirror to logic in ParentControlDesigner. The component should be
        // added somewhere, and this logic decides where.
 
        // If the component is a toolstrip or a top level form, we should add to the tray.
 
        IDesigner designer = host.GetDesigner(component);
        bool addControl = designer is ToolStripDesigner
            || designer is not ControlDesigner cd
            || (cd.Control is Form form && form.TopLevel);
 
        if (!addControl || !TypeDescriptor.GetAttributes(component).Contains(DesignTimeVisibleAttribute.Yes))
        {
            return;
        }
 
        if (_componentTray is null && TryGetService(out ISplitWindowService sws))
        {
            _componentTray = new ComponentTray(this, Component.Site);
            sws.AddSplitWindow(_componentTray);
 
            _componentTray.Height = _trayHeight;
            _componentTray.ShowLargeIcons = _trayLargeIcon;
            _componentTray.AutoArrange = _trayAutoArrange;
 
            host.AddService(_componentTray);
        }
 
        if (_componentTray is not null)
        {
            // Suspend the layout of the tray through the loading
            // process. This way, we won't continuously try to layout
            // components in auto arrange mode. We will instead let
            // the controls restore themselves to their persisted state
            // and then let auto-arrange kick in once.
 
            if (host is not null && host.Loading && !_trayLayoutSuspended)
            {
                _trayLayoutSuspended = true;
                _componentTray.SuspendLayout();
            }
 
            _componentTray.AddComponent(component);
        }
    }
 
    /// <summary>
    ///  Called when a component is removed from the design container.
    ///  Here we check to see there are no more components on the tray
    ///  and if not, we will remove the tray.
    /// </summary>
    private void OnComponentRemoved(object source, ComponentEventArgs ce)
    {
        // ToolStrip is designableAsControl but has a ComponentTray Entry ... so special case it out.
        bool designableAsControl = ce.Component is Control
            && ce.Component is not ToolStrip
            && !(ce.Component is Form form && form.TopLevel);
 
        if (!designableAsControl && _componentTray is not null)
        {
            _componentTray.RemoveComponent(ce.Component);
 
            if (_componentTray.ComponentCount == 0)
            {
                ISplitWindowService sws = (ISplitWindowService)GetService(typeof(ISplitWindowService));
                if (sws is not null)
                {
                    sws.RemoveSplitWindow(_componentTray);
                    IDesignerHost host = (IDesignerHost)GetService(typeof(IDesignerHost));
                    host?.RemoveService<ComponentTray>();
 
                    _componentTray.Dispose();
                    _componentTray = null;
                }
            }
        }
    }
 
    /// <summary>
    ///  Called when the context menu should be displayed. This displays the document
    ///  context menu.
    /// </summary>
    protected override void OnContextMenu(int x, int y)
    {
        IMenuCommandService mcs = (IMenuCommandService)GetService(typeof(IMenuCommandService));
        if (mcs is not null)
        {
            ISelectionService selSvc = (ISelectionService)GetService(typeof(ISelectionService));
            if (selSvc is not null)
            {
                // Here we check to see if we're the only component selected. If not, then
                // we'll display the standard component menu.
                //
                if (selSvc.SelectionCount == 1 && selSvc.GetComponentSelected(Component))
                {
                    mcs.ShowContextMenu(MenuCommands.ContainerMenu, x, y);
                }
 
                // Try to see if the component selected has a contextMenuStrip to show
                else
                {
                    Component selComp = selSvc.PrimarySelection as Component;
                    if (selComp is not null)
                    {
                        IDesignerHost host = (IDesignerHost)GetService(typeof(IDesignerHost));
                        if (host is not null)
                        {
                            ComponentDesigner compDesigner = host.GetDesigner(selComp) as ComponentDesigner;
                            if (compDesigner is not null)
                            {
                                compDesigner.ShowContextMenu(x, y);
                                return;
                            }
                        }
                    }
 
                    mcs.ShowContextMenu(MenuCommands.SelectionMenu, x, y);
                }
            }
        }
    }
 
    /// <summary>
    ///  This is called immediately after the control handle has been created.
    /// </summary>
    protected override void OnCreateHandle()
    {
        // Don't call base unless our inheritance service is already running.
        if (_inheritanceService is not null)
        {
            base.OnCreateHandle();
        }
    }
 
    /// <summary>
    ///  Creates some of the more infrequently used services we offer.
    /// </summary>
    private object OnCreateService(IServiceContainer container, Type serviceType)
    {
        if (serviceType == typeof(IEventHandlerService))
        {
            _eventHandlerService ??= new EventHandlerService(_frame);
 
            return _eventHandlerService;
        }
        else if (serviceType == typeof(ToolStripAdornerWindowService))
        {
            return new ToolStripAdornerWindowService(Component.Site, _frame);
        }
 
        Debug.Fail($"Called back to create a service we don't know how to create: {serviceType.Name}");
        return null;
    }
 
    /// <summary>
    ///  Called when our document becomes active. We paint our form's
    ///  border the appropriate color here.
    /// </summary>
    private ToolboxItem OnCreateToolboxItem(object serializedData, string format)
    {
        if (serializedData is not IDataObject dataObject)
        {
            Debug.Fail("Toolbox service didn't pass us a data object; that should never happen");
            return null;
        }
 
        // CLSID format.
        if (format.Equals(AxClipFormat))
        {
            return CreateAxToolboxItem(dataObject);
        }
 
        // CF_CODE format
        if (format.Equals(OleDragDropHandler.DataFormat) ||
            format.Equals(OleDragDropHandler.NestedToolboxItemFormat))
        {
            return CreateCfCodeToolboxItem(dataObject);
        }
 
        return null;
    }
 
    /// <summary>
    ///  Called when our document becomes active. Here we try to
    ///  select the appropriate toolbox tab.
    /// </summary>
    private void OnDesignerActivate(object source, EventArgs evevent)
    {
        if (_undoEngine is null)
        {
            _undoEngine = GetService(typeof(UndoEngine)) as UndoEngine;
            if (_undoEngine is not null)
            {
                _undoEngine.Undoing += OnUndoing;
                _undoEngine.Undone += OnUndone;
            }
        }
    }
 
    /// <summary>
    ///  Called by the host when we become inactive. Here we update the
    ///  title bar of our form so it's the inactive color.
    /// </summary>
    private unsafe void OnDesignerDeactivate(object sender, EventArgs e)
    {
        Control control = Control;
        if (control is not null && control.IsHandleCreated)
        {
            PInvokeCore.SendMessage(control, PInvokeCore.WM_NCACTIVATE, (WPARAM)(BOOL)false);
            PInvoke.RedrawWindow(control, lprcUpdate: null, HRGN.Null, REDRAW_WINDOW_FLAGS.RDW_FRAME);
        }
    }
 
    /// <summary>
    ///  Called when the designer is finished loading. Here we select the form.
    /// </summary>
    private void OnLoadComplete(object sender, EventArgs e)
    {
        // Remove the handler; we're done looking at this.
        ((IDesignerHost)sender).LoadComplete -= OnLoadComplete;
 
        // Restore the tray layout.
        if (_trayLayoutSuspended && _componentTray is not null)
            _componentTray.ResumeLayout();
 
        // Select this component.
        ISelectionService ss = (ISelectionService)GetService(typeof(ISelectionService));
        if (ss is not null)
        {
            ss.SelectionChanged += OnSelectionChanged;
            ss.SetSelectedComponents(new object[] { Component }, SelectionTypes.Replace);
            Debug.Assert(ss.PrimarySelection == Component, "Bug in selection service:  form should have primary selection.");
        }
    }
 
    private void OnComponentChanged(object sender, ComponentChangedEventArgs e)
    {
        Control ctrl = e.Component as Control;
        if (ctrl is not null && ctrl.IsHandleCreated)
        {
            PInvoke.NotifyWinEvent(
                (uint)AccessibleEvents.LocationChange,
                ctrl,
                (int)OBJECT_IDENTIFIER.OBJID_CLIENT,
                (int)PInvoke.CHILDID_SELF);
 
            if (_frame.Focused)
            {
                PInvoke.NotifyWinEvent(
                    (uint)AccessibleEvents.Focus,
                    ctrl,
                    (int)OBJECT_IDENTIFIER.OBJID_CLIENT,
                    (int)PInvoke.CHILDID_SELF);
            }
        }
    }
 
    /// <summary>
    ///  Called by the selection service when the selection has changed. We do a number
    ///  of selection-related things here.
    /// </summary>
    private void OnSelectionChanged(object sender, EventArgs e)
    {
        if (!TryGetService(out ISelectionService svc))
        {
            return;
        }
 
        ICollection selComponents = svc.GetSelectedComponents();
 
        // Setup the correct active accessibility selection / focus data
        foreach (object selObj in selComponents)
        {
            if (selObj is Control c)
            {
                PInvoke.NotifyWinEvent(
                    (uint)AccessibleEvents.SelectionAdd,
                    c,
                    (int)OBJECT_IDENTIFIER.OBJID_CLIENT,
                    (int)PInvoke.CHILDID_SELF);
            }
        }
 
        if (svc.PrimarySelection is Control primary)
        {
            PInvoke.NotifyWinEvent(
                (uint)AccessibleEvents.Focus,
                primary,
                (int)OBJECT_IDENTIFIER.OBJID_CLIENT,
                (int)PInvoke.CHILDID_SELF);
        }
 
        // See if there are visual controls selected. If so, we add a context attribute.
        // Otherwise, we remove the attribute. We do not count the form.
        IHelpService hs = (IHelpService)GetService(typeof(IHelpService));
 
        if (hs is not null)
        {
            ushort type = 0;
            string[] types =
            [
                "VisualSelection",
                    "NonVisualSelection",
                    "MixedSelection"
            ];
 
            foreach (object obj in selComponents)
            {
                if (obj is Control)
                {
                    if (obj != Component)
                    {
                        type |= 1;
                    }
                }
                else
                {
                    type |= 2;
                }
 
                if (type == 3)
                {
                    break;
                }
            }
 
            // Remove any prior attribute
            for (int i = 0; i < types.Length; i++)
            {
                hs.RemoveContextAttribute("Keyword", types[i]);
            }
 
            if (type != 0)
            {
                hs.AddContextAttribute("Keyword", types[type - 1], HelpKeywordType.GeneralKeyword);
            }
        }
    }
 
    /// <summary>
    ///  Allows a designer to filter the set of properties
    ///  the component it is designing will expose through the
    ///  TypeDescriptor object. This method is called
    ///  immediately before its corresponding "Post" method.
    ///  If you are overriding this method you should call
    ///  the base implementation before you perform your own
    ///  filtering.
    /// </summary>
    protected override void PreFilterProperties(IDictionary properties)
    {
        PropertyDescriptor prop;
 
        base.PreFilterProperties(properties);
 
        // Add any properties that the Tray will persist.
        properties["TrayHeight"] = TypeDescriptor.CreateProperty(typeof(DocumentDesigner), "TrayHeight", typeof(int),
                                                        BrowsableAttribute.No,
                                                        DesignOnlyAttribute.Yes,
                                                        new SRDescriptionAttribute("FormDocumentDesignerTraySizeDescr"),
                                                        CategoryAttribute.Design);
 
        properties["TrayLargeIcon"] = TypeDescriptor.CreateProperty(typeof(DocumentDesigner), "TrayLargeIcon", typeof(bool),
                                                           BrowsableAttribute.No,
                                                           DesignOnlyAttribute.Yes,
                                                           CategoryAttribute.Design);
 
        // Expose the doublebuffered property on control
        properties["DoubleBuffered"] = TypeDescriptor.CreateProperty(typeof(Control), "DoubleBuffered", typeof(bool),
                                                            BrowsableAttribute.Yes,
                                                            DesignOnlyAttribute.No);
 
        // Handle shadowed properties
        //
        string[] shadowProps =
        [
            "Location",
            "BackColor"
        ];
 
        string[] noBrowseProps =
        [
            "Anchor",
            "Dock",
            "TabIndex",
            "TabStop",
            "Visible"
        ];
 
        Attribute[] empty = [];
 
        for (int i = 0; i < shadowProps.Length; i++)
        {
            prop = (PropertyDescriptor)properties[shadowProps[i]];
            if (prop is not null)
            {
                properties[shadowProps[i]] = TypeDescriptor.CreateProperty(typeof(DocumentDesigner), prop, empty);
            }
        }
 
        prop = (PropertyDescriptor)properties["AutoScaleDimensions"];
        if (prop is not null)
        {
            properties["AutoScaleDimensions"] = TypeDescriptor.CreateProperty(typeof(DocumentDesigner), prop, DesignerSerializationVisibilityAttribute.Visible);
        }
 
        prop = (PropertyDescriptor)properties["AutoScaleMode"];
        if (prop is not null)
        {
            properties["AutoScaleMode"] = TypeDescriptor.CreateProperty(typeof(DocumentDesigner), prop, DesignerSerializationVisibilityAttribute.Visible, BrowsableAttribute.Yes);
        }
 
        for (int i = 0; i < noBrowseProps.Length; i++)
        {
            prop = (PropertyDescriptor)properties[noBrowseProps[i]];
            if (prop is not null)
            {
                properties[noBrowseProps[i]] = TypeDescriptor.CreateProperty(prop.ComponentType, prop, BrowsableAttribute.No, DesignerSerializationVisibilityAttribute.Hidden);
            }
        }
    }
 
    /// <summary>
    ///  Resets the back color to be based on the parent's back color.
    /// </summary>
    private void ResetBackColor()
    {
        BackColor = Color.Empty;
    }
 
    private bool ShouldSerializeAutoScaleDimensions()
    {
        // Visual inheritance always adds default
        // value attributes that adopt the current values. This
        // isn't right for auto scale, however,because we always
        // want to write out the auto scale values. So, we have
        // to be a bit sleazy here and trick the inheritance engine
        // to think that these properties currently have their
        // default values.
        return !_initializing && AutoScaleMode != AutoScaleMode.None && AutoScaleMode != AutoScaleMode.Inherit;
    }
 
    private bool ShouldSerializeAutoScaleMode()
    {
        // Visual inheritance always adds default
        // value attributes that adopt the current values. This
        // isn't right for auto scale, however,because we always
        // want to write out the auto scale values. So, we have
        // to be a bit sleazy here and trick the inheritance engine
        // to think that these properties currently have their
        // default values.
        return !_initializing && ShadowProperties.Contains(nameof(AutoScaleMode));
    }
 
    /// <summary>
    ///  Returns true if the BackColor property should be persisted in code gen.
    /// </summary>
    private bool ShouldSerializeBackColor()
    {
        // We push Color.Empty into our shadow cash during
        // init and also whenever we are reset. We do this
        // so we can push a real color into the controls
        // back color to stop it from walking the parent chain.
        // But, we want it to look like we didn't push a color
        // so code gen won't write out "Color.Control"
        if (!ShadowProperties.Contains(nameof(BackColor)) || ((Color)ShadowProperties[nameof(BackColor)]).IsEmpty)
        {
            return false;
        }
 
        return true;
    }
 
    /// <summary>
    ///  This will be called when the user double-clicks on a
    ///  toolbox item. The document designer should create
    ///  a component for the given tool.
    /// </summary>
    protected virtual void ToolPicked(ToolboxItem tool)
    {
        // If the tab order UI is showing, don't allow us to place a tool.
        //
        IMenuCommandService mcs = (IMenuCommandService)GetService(typeof(IMenuCommandService));
        if (mcs is not null)
        {
            MenuCommand cmd = mcs.FindCommand(StandardCommands.TabOrder);
            if (cmd is not null && cmd.Checked)
            {
                return;
            }
        }
 
        IDesignerHost host = (IDesignerHost)GetService(typeof(IDesignerHost));
        host?.Activate();
 
        // Just find the currently selected frame designer and ask it to create the tool.
        //
        try
        {
            ParentControlDesigner designer = GetSelectedParentControlDesigner();
            if (!InvokeGetInheritanceAttribute(designer).Equals(InheritanceAttribute.InheritedReadOnly))
            {
                InvokeCreateTool(designer, tool);
                IToolboxService toolboxService = (IToolboxService)GetService(typeof(IToolboxService));
                toolboxService?.SelectedToolboxItemUsed();
            }
        }
        catch (Exception e)
        {
            DisplayError(e);
            if (e.IsCriticalException())
            {
                throw;
            }
        }
    }
 
    /// <internalonly/>
    /// <summary>
    /// The list of technologies that this designer can support
    /// for its view. Examples of different technologies are
    /// WinForms and Web Forms. Other object models can be
    /// supported at design time, but they most be able to
    /// provide a view in one of the supported technologies.
    /// </summary>
    ViewTechnology[] IRootDesigner.SupportedTechnologies
    {
        get
        {
            return [ViewTechnology.Default, (ViewTechnology)1];
        }
    }
 
    /// <summary>
    ///  The view for this document. The designer should assume that the view will be shown shortly
    ///  after this call is made and make any necessary preparations.
    /// </summary>
    object IRootDesigner.GetView(ViewTechnology technology) =>
        technology is not ViewTechnology.Default and not (ViewTechnology)1
            ? throw new ArgumentException(null, nameof(technology))
            : (object)_frame;
 
    /// <summary>
    ///  Determines if the given tool is supported by this designer.
    ///  If a tool is supported then it will be enabled in the toolbox
    ///  when this designer regains focus. Otherwise, it will be disabled.
    ///  Once a tool is marked as enabled or disabled it may not be
    ///  queried again.
    /// </summary>
    bool IToolboxUser.GetToolSupported(ToolboxItem tool)
    {
        return GetToolSupported(tool);
    }
 
    /// <internalonly/>
    /// <summary>
    /// <para>Selects the specified tool.</para>
    /// </summary>
    void IToolboxUser.ToolPicked(ToolboxItem tool)
    {
        using (ScaleHelper.EnterDpiAwarenessScope(DPI_AWARENESS_CONTEXT.DPI_AWARENESS_CONTEXT_SYSTEM_AWARE))
        {
            ToolPicked(tool);
        }
    }
}