File: System\Windows\Forms\Design\OleDragDropHandler.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.
 
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 Windows.Win32.System.SystemServices;
 
namespace System.Windows.Forms.Design;
 
internal partial class OleDragDropHandler
{
    // This is a bit that we stuff into the DoDragDrop
    // to indicate that the thing that is being dragged should only
    // be allowed to be moved in the current DropTarget (e.g. parent designer).
    // We use this for inherited components that can be modified (e.g. location/size) changed
    // but not removed from their parent.
    //
    protected const int AllowLocalMoveOnly = 0x04000000;
 
    private readonly SelectionUIHandler? _selectionHandler;
    private readonly IServiceProvider _serviceProvider;
 
    private bool _dragOk;
    private bool _forceDrawFrames;
    private bool _localDragInside;
    private Point _localDragOffset = Point.Empty;
    private DragDropEffects _localDragEffect;
    private object[]? _dragComps;
    private Point _dragBase = Point.Empty;
    private static Dictionary<IDataObject, IComponent>? s_currentDrags;
 
    public const string CF_CODE = "CF_XMLCODE";
    public const string CF_COMPONENTTYPES = "CF_COMPONENTTYPES";
    public const string CF_TOOLBOXITEM = "CF_NESTEDTOOLBOXITEM";
 
    public OleDragDropHandler(SelectionUIHandler? selectionHandler, IServiceProvider serviceProvider, IOleDragClient client)
    {
        _serviceProvider = serviceProvider;
        _selectionHandler = selectionHandler;
        Destination = client;
    }
 
    public static string DataFormat => CF_CODE;
 
    public static string ExtraInfoFormat => CF_COMPONENTTYPES;
 
    public static string NestedToolboxItemFormat => CF_TOOLBOXITEM;
 
    private static IComponent? GetDragOwnerComponent(IDataObject data)
    {
        return s_currentDrags is null || !s_currentDrags.TryGetValue(data, out IComponent? value) ? null : value;
    }
 
    [MemberNotNull(nameof(s_currentDrags))]
    private static void AddCurrentDrag(IDataObject data, IComponent component)
    {
        s_currentDrags ??= [];
        s_currentDrags[data] = component;
    }
 
    internal IOleDragClient Destination { get; }
 
    protected virtual bool CanDropDataObject(IDataObject? dataObj)
    {
        if (dataObj is not null)
        {
            if (dataObj is ComponentDataObjectWrapper)
            {
                object[]? dragObjects = GetDraggingObjects(dataObj, true);
                if (dragObjects is null)
                {
                    return false;
                }
 
                bool dropOk = true;
                for (int i = 0; dropOk && i < dragObjects.Length; i++)
                {
                    dropOk = dropOk && (dragObjects[i] is IComponent component) && Destination.IsDropOk(component);
                }
 
                return dropOk;
            }
 
            try
            {
                object? serializationData = dataObj.GetData(DataFormat, false);
 
                if (serializationData is null)
                {
                    return false;
                }
 
                if (!TryGetService(out IDesignerSerializationService? ds))
                {
                    return false;
                }
 
                ICollection objects = ds.Deserialize(serializationData);
                if (objects.Count > 0)
                {
                    foreach (object o in objects)
                    {
                        if (o is IComponent component && !Destination.IsDropOk(component))
                        {
                            return false;
                        }
                    }
 
                    return true;
                }
            }
            catch (Exception ex) when (!ex.IsCriticalException())
            {
                // We return false on any exception.
            }
        }
 
        return false;
    }
 
    public bool Dragging { get; private set; }
 
    public static bool FreezePainting { get; private set; }
 
    /// <summary>
    ///  This is the worker method of all CreateTool methods. It is the only one
    ///  that can be overridden.
    /// </summary>
    public IComponent[] CreateTool(ToolboxItem tool, Control? parent, int x, int y, int width, int height, bool hasLocation, bool hasSize)
    {
        return CreateTool(tool, parent, x, y, width, height, hasLocation, hasSize, null);
    }
 
    public IComponent[] CreateTool(ToolboxItem tool, Control? parent, int x, int y, int width, int height, bool hasLocation, bool hasSize, ToolboxSnapDragDropEventArgs? e)
    {
        // Services we will need
        //
        IToolboxService? toolboxSvc = GetService<IToolboxService>();
        IDesignerHost? host = GetService<IDesignerHost>();
        IComponent[]? comps = [];
 
        Cursor? oldCursor = Cursor.Current;
        Cursor.Current = Cursors.WaitCursor;
        DesignerTransaction? trans = null;
 
        try
        {
            try
            {
                trans = host?.CreateTransaction(string.Format(SR.DesignerBatchCreateTool, tool));
            }
            catch (CheckoutException cxe) when (cxe == CheckoutException.Canceled)
            {
                return [];
            }
 
            try
            {
                try
                {
                    // First check if we are currently in localization mode (i.e., language is non-default).
                    // If so, we should not permit addition of new components. This is an intentional
                    // change from Everett - see VSWhidbey #292249.
                    if (host is not null && CurrentlyLocalizing(host.RootComponent))
                    {
                        IUIService? uiService = GetService<IUIService>();
                        uiService?.ShowMessage(SR.LocalizingCannotAdd);
 
                        return [];
                    }
 
                    // Create a dictionary of default values that the designer can
                    // use to initialize a control with.
                    Hashtable defaultValues = [];
                    if (parent is not null)
                        defaultValues["Parent"] = parent;
 
                    // adjust the location if we are in a mirrored parent. That is because the origin
                    // will then be in the upper right rather than upper left.
                    if (parent is not null && parent.IsMirrored)
                    {
                        x += width;
                    }
 
                    if (hasLocation)
                        defaultValues["Location"] = new Point(x, y);
                    if (hasSize)
                        defaultValues["Size"] = new Size(width, height);
                    // store off extra behavior drag/drop information
                    if (e is not null)
                        defaultValues["ToolboxSnapDragDropEventArgs"] = e;
 
                    comps = tool.CreateComponents(host, defaultValues);
                }
                catch (CheckoutException checkoutEx) when (checkoutEx == CheckoutException.Canceled)
                {
                    comps = [];
                }
                catch (ArgumentException argumentEx)
                {
                    IUIService? uiService = GetService<IUIService>();
                    uiService?.ShowError(argumentEx);
                }
                catch (Exception ex)
                {
                    string exceptionMessage = string.Empty;
                    if (ex.InnerException is not null)
                    {
                        exceptionMessage = ex.InnerException.ToString();
                    }
 
                    if (string.IsNullOrEmpty(exceptionMessage))
                    {
                        exceptionMessage = ex.ToString();
                    }
 
                    if (ex is InvalidOperationException)
                    {
                        exceptionMessage = ex.Message;
                    }
 
                    if (TryGetService(out IUIService? uiService))
                    {
                        uiService.ShowError(ex, string.Format(SR.FailedToCreateComponent, tool.DisplayName, exceptionMessage));
                    }
                    else
                    {
                        throw;
                    }
                }
 
                comps ??= [];
            }
            finally
            {
                if (toolboxSvc is not null && tool.Equals(toolboxSvc.GetSelectedToolboxItem(host)))
                {
                    toolboxSvc.SelectedToolboxItemUsed();
                }
            }
        }
        finally
        {
            trans?.Commit();
 
            Cursor.Current = oldCursor;
        }
 
        // Finally, select the newly created components.
        //
        if (TryGetService(out ISelectionService? selSvc) && comps.Length > 0)
        {
            host?.Activate();
 
            List<IComponent> selectComps = new(comps);
 
            for (int i = 0; i < comps.Length; i++)
            {
                if (!TypeDescriptor.GetAttributes(comps[i]).Contains(DesignTimeVisibleAttribute.Yes))
                {
                    selectComps.Remove(comps[i]);
                }
            }
 
            selSvc.SetSelectedComponents(selectComps.ToArray(), SelectionTypes.Replace);
        }
 
        return comps;
    }
 
    /// <summary>
    ///  Determines whether we are currently in localization mode - i.e., language is not (Default).
    /// </summary>
    private static bool CurrentlyLocalizing(IComponent? rootComponent)
    {
        if (rootComponent is not null)
        {
            PropertyDescriptor? prop = TypeDescriptor.GetProperties(rootComponent)["Language"];
 
            if (prop is not null && prop.PropertyType == typeof(Globalization.CultureInfo))
            {
                Globalization.CultureInfo ci = (Globalization.CultureInfo)prop.GetValue(rootComponent)!;
                if (!ci.Equals(Globalization.CultureInfo.InvariantCulture))
                {
                    return true;
                }
            }
        }
 
        return false;
    }
 
    private static void DisableDragDropChildren(ICollection controls, List<Control> allowDropCache)
    {
        foreach (Control c in controls)
        {
            if (c is not null)
            {
                if (c.AllowDrop)
                {
                    allowDropCache.Add(c);
                    c.AllowDrop = false;
                }
 
                if (c.HasChildren)
                {
                    DisableDragDropChildren(c.Controls, allowDropCache);
                }
            }
        }
    }
 
    private Point DrawDragFrames(
        object[]? comps,
        Point oldOffset,
        DragDropEffects oldEffect,
        Point newOffset,
        bool drawAtNewOffset)
    {
        Control parentControl = Destination.GetDesignerControl();
 
        if (_selectionHandler is null)
        {
            Debug.Fail("selectionHandler should not be null");
            return Point.Empty;
        }
 
        if (comps is null)
        {
            return Point.Empty;
        }
 
        for (int i = 0; i < comps.Length; i++)
        {
            Control comp = Destination.GetControlForComponent(comps[i])!;
 
            Color backColor = SystemColors.Control;
            try
            {
                backColor = comp.BackColor;
            }
            catch (Exception ex) when (!ex.IsCriticalException())
            {
            }
 
            // If we are moving, we must make sure that the location property of the component
            // is not read only. Otherwise, we can't move the thing.
            bool readOnlyLocation = true;
 
            PropertyDescriptor? loc = TypeDescriptor.GetProperties(comps[i])["Location"];
            if (loc is not null)
            {
                readOnlyLocation = loc.IsReadOnly;
            }
 
            Rectangle newRect;
            // first, undraw the old rect
            if (!oldOffset.IsEmpty)
            {
                if ((oldEffect & DragDropEffects.Move) == 0 ||
                    !readOnlyLocation)
                {
                    newRect = comp.Bounds;
 
                    if (drawAtNewOffset)
                    {
                        newRect.X = oldOffset.X;
                        newRect.Y = oldOffset.Y;
                    }
                    else
                    {
                        newRect.Offset(oldOffset.X, oldOffset.Y);
                    }
 
                    newRect = _selectionHandler.GetUpdatedRect(comp.Bounds, newRect, false);
                    DrawReversibleFrame((HWND)parentControl.Handle, newRect, backColor);
                }
            }
 
            if (!newOffset.IsEmpty)
            {
                if ((oldEffect & DragDropEffects.Move) == 0 ||
                    !readOnlyLocation)
                {
                    newRect = comp.Bounds;
                    if (drawAtNewOffset)
                    {
                        newRect.X = newOffset.X;
                        newRect.Y = newOffset.Y;
                    }
                    else
                    {
                        newRect.Offset(newOffset.X, newOffset.Y);
                    }
 
                    newRect = _selectionHandler.GetUpdatedRect(comp.Bounds, newRect, false);
                    DrawReversibleFrame((HWND)parentControl.Handle, newRect, backColor);
                }
            }
        }
 
        return newOffset;
    }
 
    private static void DrawReversibleFrame(HWND handle, Rectangle rectangle, Color backColor)
    {
        // Bug # 71547 <subhag> to make drag rect visible if any the dimensions of the control are 0
        if (rectangle.Width == 0)
            rectangle.Width = 5;
        if (rectangle.Height == 0)
            rectangle.Height = 5;
 
        // Copy of ControlPaint.DrawReversibleFrame, see VSWhidbey 581670
        // If ControlPaint ever gets overloaded, we should replace the code below by calling it:
        // ControlPaint.DrawReversibleFrame(handle, rectangle, backColor, FrameStyle.Thick);
 
        // ------ Duplicate code----------------------------------------------------------
        R2_MODE rop2;
        Color graphicsColor;
 
        if (backColor.GetBrightness() < .5)
        {
            rop2 = R2_MODE.R2_NOTXORPEN;
            graphicsColor = Color.White;
        }
        else
        {
            rop2 = R2_MODE.R2_XORPEN;
            graphicsColor = Color.Black;
        }
 
        using GetDcScope dc = new(handle);
        using CreatePenScope pen = new(backColor, 2);
 
        using SetRop2Scope rop2Scope = new(dc, rop2);
        using SelectObjectScope brushSelection = new(dc, PInvokeCore.GetStockObject(GET_STOCK_OBJECT_FLAGS.NULL_BRUSH));
        using SelectObjectScope penSelection = new(dc, pen);
 
        PInvokeCore.SetBkColor(dc, (COLORREF)(uint)ColorTranslator.ToWin32(graphicsColor));
        PInvoke.Rectangle(dc, rectangle.X, rectangle.Y, rectangle.Right, rectangle.Bottom);
        // ------ Duplicate code----------------------------------------------------------
    }
 
    public unsafe bool DoBeginDrag(object[] components, SelectionRules rules, int initialX, int initialY)
    {
        // if we're in a sizing operation, or the mouse isn't down, don't do this!
        if ((rules & SelectionRules.AllSizeable) != SelectionRules.None || Control.MouseButtons == MouseButtons.None)
        {
            return true;
        }
 
        Control c = Destination.GetDesignerControl();
 
        Dragging = true;
        _localDragInside = false;
 
        _dragComps = components;
        _dragBase = new Point(initialX, initialY);
        _localDragOffset = Point.Empty;
        c.PointToClient(new Point(initialX, initialY));
 
        DragDropEffects allowedEffects = DragDropEffects.Copy | DragDropEffects.None | DragDropEffects.Move;
 
        // check to see if any of the components are inherited. if so, don't allow them to be moved.
        // We replace DragDropEffects.Move with a local bit called AllowLocalMoveOnly which means it
        // can be moved around on the current dropsource/target, but not to another target. Since only
        // we understand this bit, other drop targets will not allow the move to occur
        //
        for (int i = 0; i < components.Length; i++)
        {
            InheritanceAttribute attr = (InheritanceAttribute)TypeDescriptor.GetAttributes(components[i])[typeof(InheritanceAttribute)]!;
 
            if (!attr.Equals(InheritanceAttribute.NotInherited) && !attr.Equals(InheritanceAttribute.InheritedReadOnly))
            {
                allowedEffects &= ~DragDropEffects.Move;
                allowedEffects |= (DragDropEffects)AllowLocalMoveOnly;
            }
        }
 
        DataObject data = new ComponentDataObjectWrapper(new ComponentDataObject(_serviceProvider, components));
 
        // We make sure we're painted before we start the drag. Then, we disable window painting to
        // ensure that the drag can proceed without leaving artifacts lying around. We should be calling LockWindowUpdate,
        // but that causes a horrible flashing because GDI+ uses direct draw.
        MSG msg = default;
        while (PInvokeCore.PeekMessage(&msg, HWND.Null, PInvokeCore.WM_PAINT, PInvokeCore.WM_PAINT, PEEK_MESSAGE_REMOVE_TYPE.PM_REMOVE))
        {
            PInvoke.TranslateMessage(msg);
            PInvoke.DispatchMessage(&msg);
        }
 
        // don't do any new painting...
        bool oldFreezePainting = FreezePainting;
 
        // ASURT 90345 -- this causes some subtle bugs, so i'm turning it off to see if we really need it, and if we do
        // if we can find a better way.
        //
        // freezePainting = true;
 
        AddCurrentDrag(data, Destination.Component);
 
        // Walk through all the children recursively and disable drag-drop
        // for each of them. This way, we will not accidentally try to drop
        // ourselves into our own children.
        //
        List<Control> allowDropChanged = [];
        foreach (object comp in components)
        {
            if (comp is Control { HasChildren: true } ctl)
            {
                DisableDragDropChildren(ctl.Controls, allowDropChanged);
            }
        }
 
        DragDropEffects effect = DragDropEffects.None;
        DesignerTransaction? trans = null;
        if (TryGetService(out IDesignerHost? host))
        {
            trans = host.CreateTransaction(string.Format(SR.DragDropDragComponents, components.Length));
        }
 
        try
        {
            effect = c.DoDragDrop(data, allowedEffects);
            trans?.Commit();
        }
        finally
        {
            s_currentDrags.Remove(data);
 
            // Reset the AllowDrop for the components being dragged.
            //
            foreach (Control ctl in allowDropChanged)
            {
                ctl.AllowDrop = true;
            }
 
            FreezePainting = oldFreezePainting;
 
            ((IDisposable?)trans)?.Dispose();
        }
 
        bool isMove = (effect & DragDropEffects.Move) != 0 || ((int)effect & AllowLocalMoveOnly) != 0;
 
        // since the EndDrag will clear this
        bool isLocalMove = isMove && _localDragInside;
 
        ISelectionUIService? selectionUISvc = GetService<ISelectionUIService>();
        Debug.Assert(selectionUISvc is not null, "Unable to get selection ui service when adding child control");
 
        if (selectionUISvc is not null)
        {
            // We must check to ensure that UI service is still in drag mode. It is
            // possible that the user hit escape, which will cancel drag mode.
            //
            if (selectionUISvc.Dragging)
            {
                // cancel the drag if we aren't doing a local move
                selectionUISvc.EndDrag(!isLocalMove);
            }
        }
 
        if (!_localDragOffset.IsEmpty && effect != DragDropEffects.None)
        {
            DrawDragFrames(_dragComps, _localDragOffset, _localDragEffect, Point.Empty, drawAtNewOffset: false);
        }
 
        _localDragOffset = Point.Empty;
        _dragComps = null;
        Dragging = _localDragInside = false;
        _dragBase = Point.Empty;
 
        return false;
    }
 
    public void DoEndDrag()
    {
        _dragComps = null;
        Dragging = false;
        _localDragInside = false;
    }
 
    public void DoOleDragDrop(DragEventArgs de)
    {
        // ASURT 43757: By the time we come here, it means that the user completed the drag-drop and
        // we compute the new location/size of the controls if needed and set the property values.
        // We have to stop freezePainting right here, so that controls can get a chance to validate
        // their new rectangles.
        //
        FreezePainting = false;
 
        if (_selectionHandler is null)
        {
            Debug.Fail("selectionHandler should not be null");
            de.Effect = DragDropEffects.None;
            return;
        }
 
        // make sure we've actually moved
        if ((Dragging && de.X == _dragBase.X && de.Y == _dragBase.Y) ||
            de.AllowedEffect == DragDropEffects.None ||
            (!Dragging && !_dragOk))
        {
            de.Effect = DragDropEffects.None;
            return;
        }
 
        bool localMoveOnly = ((int)de.AllowedEffect & AllowLocalMoveOnly) != 0 && _localDragInside;
 
        // if we are dragging inside the local dropsource/target, and and AllowLocalMoveOnly flag is set,
        // we just consider this a normal move.
        //
        bool moveAllowed = (de.AllowedEffect & DragDropEffects.Move) != DragDropEffects.None || localMoveOnly;
        bool copyAllowed = (de.AllowedEffect & DragDropEffects.Copy) != DragDropEffects.None;
 
        if ((de.Effect & DragDropEffects.Move) != 0 && !moveAllowed)
        {
            // Try copy instead?
            de.Effect = DragDropEffects.Copy;
        }
 
        // make sure the copy is allowed
        if ((de.Effect & DragDropEffects.Copy) != 0 && !copyAllowed)
        {
            // if copy isn't allowed, don't do anything
 
            de.Effect = DragDropEffects.None;
            return;
        }
 
        if (localMoveOnly && (de.Effect & DragDropEffects.Move) != 0)
        {
            de.Effect |= (DragDropEffects)AllowLocalMoveOnly | DragDropEffects.Move;
        }
        else if ((de.Effect & DragDropEffects.Copy) != 0)
        {
            de.Effect = DragDropEffects.Copy;
        }
 
        if (_forceDrawFrames || _localDragInside)
        {
            // Undraw the drag rect
            _localDragOffset = DrawDragFrames(
                _dragComps,
                _localDragOffset,
                _localDragEffect,
                Point.Empty,
                _forceDrawFrames);
 
            _forceDrawFrames = false;
        }
 
        Cursor? oldCursor = Cursor.Current;
 
        try
        {
            Cursor.Current = Cursors.WaitCursor;
 
            if (_dragOk || (_localDragInside && de.Effect == DragDropEffects.Copy))
            {
                // add em to this parent.
                IDesignerHost host = GetService<IDesignerHost>()!;
                IContainer? container = host.RootComponent.Site!.Container;
 
                object[] components;
                IDataObject dataObj = de.Data!;
                bool updateLocation = false;
 
                if (dataObj is ComponentDataObjectWrapper wrapper)
                {
                    ComponentDataObject cdo = wrapper.InnerData;
 
                    // if we're moving ot a different container, do a full serialization
                    // to make sure we pick up design time props, etc.
                    //
                    IComponent? dragOwner = GetDragOwnerComponent(wrapper);
                    bool newContainer = dragOwner is null || Destination.Component is null || dragOwner.Site!.Container != Destination.Component.Site!.Container;
                    bool collapseChildren = false;
                    if (de.Effect == DragDropEffects.Copy || newContainer)
                    {
                        // this causes new elements to be created
                        //
                        cdo.Deserialize(_serviceProvider, (de.Effect & DragDropEffects.Copy) == 0);
                    }
                    else
                    {
                        collapseChildren = true;
                    }
 
                    updateLocation = true;
                    components = cdo.Components;
 
                    if (collapseChildren)
                    {
                        components = GetTopLevelComponents(components);
                    }
                }
                else
                {
                    object? serializationData = dataObj.GetData(DataFormat, true);
 
                    if (serializationData is null)
                    {
                        Debug.Fail("data object didn't return any data, so how did we allow the drop?");
                        components = Array.Empty<IComponent>();
                    }
                    else
                    {
                        dataObj = new ComponentDataObject(_serviceProvider, serializationData);
                        components = ((ComponentDataObject)dataObj).Components;
                        updateLocation = true;
                    }
                }
 
                // now we need to offset the components locations from the drop mouse
                // point to the parent, since their current locations are relative
                // the mouse pointer
                if (components is not null && components.Length > 0)
                {
                    Debug.Assert(container is not null, "Didn't get a container from the site!");
                    string? name;
 
                    DesignerTransaction? trans = null;
 
                    try
                    {
                        trans = host.CreateTransaction(SR.DragDropDropComponents);
                        if (!Dragging)
                        {
                            host.Activate();
                        }
 
                        List<IComponent> selectComps = [];
 
                        for (int i = 0; i < components.Length; i++)
                        {
                            if (components[i] is not IComponent comp)
                            {
                                continue;
                            }
 
                            try
                            {
                                name = comp.Site?.Name;
 
                                Control? oldDesignerControl = null;
                                if (updateLocation)
                                {
                                    oldDesignerControl = Destination.GetDesignerControl();
                                    PInvokeCore.SendMessage(oldDesignerControl, PInvokeCore.WM_SETREDRAW, (WPARAM)(BOOL)false);
                                }
 
                                Point dropPt = Destination.GetDesignerControl().PointToClient(new Point(de.X, de.Y));
 
                                // First check if the component we are dropping have a TrayLocation, and if so, use it
                                PropertyDescriptor? loc = TypeDescriptor.GetProperties(comp)["TrayLocation"];
                                // it didn't, so let's check for the regular Location
                                loc ??= TypeDescriptor.GetProperties(comp)["Location"];
 
                                if (loc is not null && !loc.IsReadOnly)
                                {
                                    Rectangle bounds = default;
                                    Point pt = (Point)loc.GetValue(comp)!;
                                    bounds.X = dropPt.X + pt.X;
                                    bounds.Y = dropPt.Y + pt.Y;
                                    bounds = _selectionHandler.GetUpdatedRect(Rectangle.Empty, bounds, false);
                                }
 
                                if (!Destination.AddComponent(comp, name!, false))
                                {
                                    // this means that we just moved the control
                                    // around in the same designer.
 
                                    de.Effect = DragDropEffects.None;
                                }
                                else
                                {
                                    // make sure the component was added to this client
                                    if (Destination.GetControlForComponent(comp) is null)
                                    {
                                        updateLocation = false;
                                    }
                                }
 
                                if (updateLocation)
                                {
                                    if (Destination is ParentControlDesigner parentDesigner)
                                    {
                                        Control c = Destination.GetControlForComponent(comp)!;
                                        dropPt = parentDesigner.GetSnappedPoint(c.Location);
                                        c.Location = dropPt;
                                    }
                                }
 
                                if (oldDesignerControl is not null)
                                {
                                    // ((ComponentDataObject)dataObj).ShowControls();
                                    PInvokeCore.SendMessage(oldDesignerControl, PInvokeCore.WM_SETREDRAW, (WPARAM)(BOOL)true);
                                    oldDesignerControl.Invalidate(true);
                                }
 
                                if (TypeDescriptor.GetAttributes(comp).Contains(DesignTimeVisibleAttribute.Yes))
                                {
                                    selectComps.Add(comp);
                                }
                            }
                            catch (CheckoutException checkoutException) when (checkoutException == CheckoutException.Canceled)
                            {
                                break;
                            }
                        }
 
                        host.Activate();
 
                        // select the newly added components
                        ISelectionService selService = GetService<ISelectionService>()!;
 
                        selService.SetSelectedComponents(selectComps.ToArray(), SelectionTypes.Replace);
                        _localDragInside = false;
                    }
                    finally
                    {
                        trans?.Commit();
                    }
                }
            }
 
            if (_localDragInside)
            {
                ISelectionUIService? selectionUISvc = GetService<ISelectionUIService>();
                Debug.Assert(selectionUISvc is not null, "Unable to get selection ui service when adding child control");
 
                if (selectionUISvc is not null)
                {
                    // We must check to ensure that UI service is still in drag mode. It is
                    // possible that the user hit escape, which will cancel drag mode.
                    //
                    if (selectionUISvc.Dragging && moveAllowed)
                    {
                        Rectangle offset = new(de.X - _dragBase.X, de.Y - _dragBase.Y, 0, 0);
                        selectionUISvc.DragMoved(offset);
                    }
                }
            }
 
            _dragOk = false;
        }
        finally
        {
            Cursor.Current = oldCursor;
        }
    }
 
    public void DoOleDragEnter(DragEventArgs de)
    {
        /*
        this causes focus rectangles to be drawn, which we don't want to happen.
 
        Control dragHost = client.GetDesignerControl();
 
        if (dragHost != null && dragHost.CanSelect) {
            dragHost.Focus();
        }*/
 
        if (!Dragging && CanDropDataObject(de.Data) && de.AllowedEffect != DragDropEffects.None)
        {
            if (!Destination.CanModifyComponents)
            {
                return;
            }
 
            _dragOk = true;
 
            // this means it's not us doing the drag
            if ((de.KeyState & (int)MODIFIERKEYS_FLAGS.MK_CONTROL) != 0 && (de.AllowedEffect & DragDropEffects.Copy) != 0)
            {
                de.Effect = DragDropEffects.Copy;
            }
            else if ((de.AllowedEffect & DragDropEffects.Move) != 0)
            {
                de.Effect = DragDropEffects.Move;
            }
            else
            {
                de.Effect = DragDropEffects.None;
                return;
            }
        }
        else if (Dragging && de.AllowedEffect != DragDropEffects.None)
        {
            _localDragInside = true;
            if ((de.KeyState & (int)MODIFIERKEYS_FLAGS.MK_CONTROL) != 0
                && (de.AllowedEffect & DragDropEffects.Copy) != 0
                && Destination.CanModifyComponents)
            {
                de.Effect = DragDropEffects.Copy;
            }
 
            bool localMoveOnly = ((int)de.AllowedEffect & AllowLocalMoveOnly) != 0 && _localDragInside;
 
            if (localMoveOnly)
            {
                de.Effect |= (DragDropEffects)AllowLocalMoveOnly;
            }
 
            if ((de.AllowedEffect & DragDropEffects.Move) != 0)
            {
                de.Effect |= DragDropEffects.Move;
            }
        }
        else
        {
            de.Effect = DragDropEffects.None;
        }
    }
 
    public void DoOleDragLeave()
    {
        if (Dragging || _forceDrawFrames)
        {
            _localDragInside = false;
            _localDragOffset = DrawDragFrames(
                _dragComps,
                _localDragOffset,
                _localDragEffect,
                Point.Empty,
                _forceDrawFrames);
 
            if (_forceDrawFrames && _dragOk)
            {
                _dragBase = Point.Empty;
                _dragComps = null;
            }
 
            _forceDrawFrames = false;
        }
 
        _dragOk = false;
    }
 
    public void DoOleDragOver(DragEventArgs de)
    {
        if (!Dragging && !_dragOk)
        {
            de.Effect = DragDropEffects.None;
            return;
        }
 
        bool copy = (de.KeyState & (int)MODIFIERKEYS_FLAGS.MK_CONTROL) != 0
            && (de.AllowedEffect & DragDropEffects.Copy) != 0
            && Destination.CanModifyComponents;
 
        // we pretend AllowLocalMoveOnly is a normal move when we are over the originating container.
        //
        bool localMoveOnly = ((int)de.AllowedEffect & AllowLocalMoveOnly) != 0 && _localDragInside;
        bool move = (de.AllowedEffect & DragDropEffects.Move) != 0 || localMoveOnly;
 
        if ((copy || move) && (Dragging || _forceDrawFrames))
        {
            Point convertedPoint = Destination.GetDesignerControl().PointToClient(new Point(de.X, de.Y));
 
            // draw the shadow rectangles.
            Point newOffset = _forceDrawFrames ? convertedPoint : new Point(de.X - _dragBase.X, de.Y - _dragBase.Y);
 
            // Only allow drops on the client area.
            if (!Destination.GetDesignerControl().ClientRectangle.Contains(convertedPoint))
            {
                copy = false;
                move = false;
                newOffset = _localDragOffset;
            }
 
            if (newOffset != _localDragOffset)
            {
                DrawDragFrames(_dragComps, _localDragOffset, _localDragEffect, newOffset, _forceDrawFrames);
                _localDragOffset = newOffset;
                _localDragEffect = de.Effect;
            }
        }
 
        if (copy)
        {
            de.Effect = DragDropEffects.Copy;
        }
        else if (move)
        {
            de.Effect = DragDropEffects.Move;
        }
        else
        {
            de.Effect = DragDropEffects.None;
        }
 
        if (localMoveOnly)
        {
            de.Effect |= (DragDropEffects)AllowLocalMoveOnly;
        }
    }
 
    public void DoOleGiveFeedback(GiveFeedbackEventArgs e)
    {
        if (_selectionHandler is null)
        {
            Debug.Fail("selectionHandler should not be null");
        }
 
        e.UseDefaultCursors = ((!_localDragInside && !_forceDrawFrames) || ((e.Effect & (DragDropEffects.Copy)) != 0)) || e.Effect == DragDropEffects.None;
        if (!e.UseDefaultCursors && _selectionHandler is not null)
        {
            _selectionHandler.SetCursor();
        }
    }
 
    private static object[]? GetDraggingObjects(IDataObject? dataObj, bool topLevelOnly)
    {
        object[]? components = null;
 
        if (dataObj is ComponentDataObjectWrapper wrapper)
        {
            ComponentDataObject cdo = wrapper.InnerData;
 
            components = cdo.Components;
        }
 
        return !topLevelOnly || components is null ? components : GetTopLevelComponents(components);
    }
 
    public static object[]? GetDraggingObjects(IDataObject? dataObj) => GetDraggingObjects(dataObj, topLevelOnly: false);
 
    public static object[]? GetDraggingObjects(DragEventArgs de) => GetDraggingObjects(de.Data);
 
    private static object[] GetTopLevelComponents(ICollection comps)
    {
        // Filter the top-level components.
        if (comps is not IList)
        {
            comps = new ArrayList(comps);
        }
 
        IList compList = (IList)comps;
        List<object> topLevel = [];
        foreach (object comp in compList)
        {
            Control? c = comp as Control;
            if (c is null && comp is not null)
            {
                topLevel.Add(comp);
            }
            else if (c is not null)
            {
                if (c.Parent is null || !compList.Contains(c.Parent))
                {
                    topLevel.Add(c);
                }
            }
        }
 
        return [.. topLevel];
    }
 
    protected object? GetService(Type t)
    {
        return _serviceProvider.GetService(t);
    }
 
    protected T? GetService<T>() where T : class
    {
        return _serviceProvider.GetService(typeof(T)) as T;
    }
 
    protected bool TryGetService<T>([NotNullWhen(true)] out T? service) where T : class
    {
        service = _serviceProvider.GetService(typeof(T)) as T;
        return service is not null;
    }
 
    protected virtual void OnInitializeComponent(IComponent comp, int x, int y, int width, int height, bool hasLocation, bool hasSize)
    {
    }
}