WPF MVVM 拖放

WPF MVVM Drag & Drop

我有一个使用拖放功能的应用程序。我已经根据 MVVM 概念使用行为实现了此功能。我尝试使用装饰元素来创建移动物体的错觉。但是我得到了装饰者的一种非常奇怪的行为。似乎渲染引擎为其添加了一些偏移量,因此它没有出现在所需的位置。看起来偏移量会累积。我附上了我的示例项目,所以这个问题很容易重现。

private void AssociatedObjectOnMouseMove(object sender, MouseEventArgs mouseEventArgs)
    {
        Point currentPosition = mouseEventArgs.GetPosition(RelativeElement);

        //Point currentPosition = mouseEventArgs.GetPosition(Window.GetWindow(this.AssociatedObject));
        if (!_isMousePressed || !this.AssociatedObject.IsMouseOver)
            return;

        if (Math.Abs(_originalPosition.X - currentPosition.X) < SystemParameters.MinimumHorizontalDragDistance &&
            Math.Abs(_originalPosition.Y - currentPosition.Y) < SystemParameters.MinimumVerticalDragDistance)
            return;

        IDragable context = this.AssociatedObject.DataContext as IDragable;
        if (context == null)
            return;


        Debug.WriteLine("Mouse leave");
        _adorner = new DefaultAdorner(this.AssociatedObject, new Point(0, 0), RelativeElement);

        DataObject data = new DataObject();
        data.SetData(context.DataType, context);
        System.Windows.DragDrop.DoDragDrop(this.AssociatedObject, data, DragDropEffects.Move);

        if (_adorner != null)
        {
            _adorner.Destroy();
            _adorner = null;
        }

        _isMousePressed = false;
    }


    private void AssociatedObjectOnGiveFeedback(object sender, GiveFeedbackEventArgs giveFeedbackEventArgs)
    {
        Debug.WriteLine("feedback");
        Point mouseCoordinates = Extensions.GetMouseCoordinates();
        //Debug.WriteLine("initial x: {0}; y: {1}", mouseCoordinates.X, mouseCoordinates.Y);
        Point mousePosition = RelativeElement.PointFromScreen(mouseCoordinates);

        AdornerLayer layer = AdornerLayer.GetAdornerLayer(this.AssociatedObject);
        var relative = RelativeElement.TranslatePoint(mousePosition, layer);
        Debug.WriteLine("relative to layer x: {0}; y: {1}", relative.X, relative.Y);
        var towindow = RelativeElement.TranslatePoint(mousePosition, Window.GetWindow(this.AssociatedObject));
        Debug.WriteLine("relative to layer x: {0}; y: {1}", towindow.X, towindow.Y);
        //Point mousePosition = Window.GetWindow(this.AssociatedObject).PointFromScreen(Extensions.GetMouseCoordinates());
        if (_adorner != null)
        {
            _adorner.SetMousePosition(mousePosition);
        }
    }

/// <summary>
    /// Create an adorner.
    /// The created adorner must then be added to the AdornerLayer.
    /// </summary>
    /// <param name="adornedElement">Element whose AdornerLayer will be use for displaying the adorner</param>
    /// <param name="adornerElement">Element used as adorner</param>
    /// <param name="adornerOrigin">Origin offset within the adorner</param>
    /// <param name="opacity">Adorner's opacity</param>
    public DefaultAdorner(UIElement adornedElement, Point origin, FrameworkElement relative)
        : base(adornedElement)
    {
        Rectangle rect = new Rectangle();
        rect.Width = adornedElement.RenderSize.Width;
        rect.Height = adornedElement.RenderSize.Height;

        VisualBrush visualBrush = new VisualBrush(adornedElement);
        visualBrush.Opacity = 0.5;
        visualBrush.Stretch = Stretch.None;
        rect.Fill = visualBrush;


        this._child = rect;

        this._adornerOrigin = new Point(0, 0);

        this._adornerOffset = origin;
        _relative = relative;
        AdornerLayer layer = AdornerLayer.GetAdornerLayer(adornedElement);

        Adorner[] adorners = layer.GetAdorners(adornedElement);
        if (adorners != null)
        {
            Array.ForEach(adorners, layer.Remove);   
        }

        layer.Add(this);
        InvalidateVisual();
    }

    /// <summary>
    /// Set the position of and redraw the adorner.
    /// Call when the mouse cursor position changes.
    /// </summary>
    /// <param name="position">Adorner's new position relative to AdornerLayer origin</param>
    public void SetMousePosition(Point position)
    {
        this._adornerOffset.X = position.X;
        this._adornerOffset.Y = position.Y;



        Debug.WriteLine("x: {0}; y: {1}", position.X, position.Y);
        Debug.WriteLine("x: {0}; y: {1}", position.X, position.Y);
        //this._adornerOffset.X = position.X - this._adornerOrigin.X - _child.Width / 2;
        //this._adornerOffset.Y = position.Y - this._adornerOrigin.Y - _child.Height / 2;
        UpdatePosition();
    }

    private void UpdatePosition()
    {
        AdornerLayer adornerLayer = (AdornerLayer)this.Parent;
        if (adornerLayer != null)
        {
            adornerLayer.Update(this.AdornedElement);
        }
        //AdornerLayer.GetAdornerLayer(AdornedElement).Update();
    }

    protected override int VisualChildrenCount { get { return 1; } }

    protected override Visual GetVisualChild(int index)
    {
        System.Diagnostics.Debug.Assert(index == 0, "Index must be 0, there's only one child");
        return this._child;
    }

    protected override Size MeasureOverride(Size finalSize)
    {
        this._child.Measure(finalSize);
        return this._child.DesiredSize;
    }

    protected override Size ArrangeOverride(Size finalSize)
    {
        this._child.Arrange(new Rect(finalSize));
        return finalSize;
    }

    public override GeneralTransform GetDesiredTransform(GeneralTransform transform)
    {
        Debug.WriteLine(transform);

        GeneralTransformGroup newTransform = new GeneralTransformGroup();
        MatrixTransform tr = transform as MatrixTransform;
        if (tr != null)
        {
            //newTransform.Children.Add(base.GetDesiredTransform(new MatrixTransform(new Matrix(tr.Matrix.M11, tr.Matrix.M12, tr.Matrix.M21, tr.Matrix.M22, 0, 0))));
            //newTransform.Children.Add(base.GetDesiredTransform(new MatrixTransform(new Matrix(tr.Matrix.M11, tr.Matrix.M12, tr.Matrix.M21, tr.Matrix.M22, this._adornerOffset.X, this._adornerOffset.Y))));
        }

        newTransform.Children.Add(base.GetDesiredTransform(transform));
        newTransform.Children.Add(new TranslateTransform(this._adornerOffset.X, this._adornerOffset.Y));
        return newTransform;
    }

    public void Destroy()
    {
        AdornerLayer.GetAdornerLayer(AdornedElement).Remove(this);
    }

感谢任何帮助! 这是项目的 link:https://www.dropbox.com/s/nogt3gjwmf5e3pg/SandBox.zip?dl=0

试试这个 <sandBox:DragBehavior RelativeElement="{Binding ElementName=Label}"/> 而不是 <sandBox:DragBehavior RelativeElement="{Binding ElementName=Canvas}"/>