|
// 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.
//
OleDragDropHandler ddh = 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.
if (de.Data is ToolStripItemDataObject)
{
return false;
}
return true;
}
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)
{
IDesignerHost host = (IDesignerHost)GetService(typeof(IDesignerHost));
if (host is not null)
{
IComponent component = ce.Component;
bool addControl = true;
// This is the mirror to logic in ParentControlDesigner. The component should be
// added somewhere, and this logic decides where.
// LETS SEE IF WE ARE TOOLSTRIP in which case we want to get added
// to the componentTray even though this is a control..
// We should think of implementing an interface so that we can have many more
// controls behaving like this.
if (host.GetDesigner(component) is not ToolStripDesigner td)
{
ControlDesigner cd = host.GetDesigner(component) as ControlDesigner;
if (cd is not null)
{
if (cd.Control is not Form form || !form.TopLevel)
{
addControl = false;
}
}
}
if (addControl &&
TypeDescriptor.GetAttributes(component).Contains(DesignTimeVisibleAttribute.Yes))
{
if (_componentTray is null)
{
ISplitWindowService sws = (ISplitWindowService)GetService(typeof(ISplitWindowService));
if (sws is not null)
{
_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);
}
}
}
|