gpt4 book ai didi

.net - WPF MVVM拖放

转载 作者:行者123 更新时间:2023-12-03 10:44:36 26 4
gpt4 key购买 nike

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

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);
}

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

最佳答案

试试这个<sandBox:DragBehavior RelativeElement="{Binding ElementName=Label}"/>代替<sandBox:DragBehavior RelativeElement="{Binding ElementName=Canvas}"/>

关于.net - WPF MVVM拖放,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32312145/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com