我在Silverlight 4中对我的RichTextBox有拖放行为,但我遇到了一个小的捕捉问题。
我希望用户能够“抓住”RichTextBox的任何边框并拖放它。 RichTextBox应该相对于用户“抓住”RichTextBox的位置进行拖动。但相反,只要用户开始拖动,RichTextBox就会“捕捉”到鼠标位置的中间位置而不是相对于鼠标位置的位置。
所以,如果我抓住右下角,如下所示......
https://skydrive.live.com/redir?resid=DCC93DD825EF3F43!658
它会在拖动开始时“捕捉”到鼠标位置的中间位置,如下所示......
https://skydrive.live.com/redir?resid=DCC93DD825EF3F43!659
这是我的拖动代码。我假设我的数学错误(在MouseMove事件中)???
public class CustomRichTextBox : RichTextBox
{
private bool isDragging = false;
protected override void OnMouseLeftButtonDown(MouseButtonEventArgs e)
{
base.OnMouseLeftButtonDown(e);
bool isDragAllowed = false;
Point pt = e.GetPosition(this);
if (pt.Y >= 0 && pt.Y <= this.BorderThickness.Top)
// Allow dragging if the mouse is down on the top border
isDragAllowed = true;
else if (pt.Y >= (this.RenderSize.Height - this.BorderThickness.Bottom) && pt.Y <= this.RenderSize.Height)
// Allow dragging if the mouse is down on the bottom border
isDragAllowed = true;
else if (pt.X >= 0 && pt.X <= this.BorderThickness.Left)
// Allow dragging if the mouse is down on the left border
isDragAllowed = true;
else if (pt.X >= (this.RenderSize.Width - this.BorderThickness.Right) && pt.X <= this.RenderSize.Width)
// Allow dragging if the mouse is down on the right border
isDragAllowed = true;
if (!isDragAllowed)
return;
this.CaptureMouse();
isDragging = true;
}
protected override void OnMouseMove(MouseEventArgs e)
{
base.OnMouseMove(e);
if (isDragging)
{
CustomRichTextBox elem = this;
CompositeTransform ct = (CompositeTransform)elem.RenderTransform;
UIElement parent = (UIElement)elem.Parent;
ct.TranslateX = e.GetPosition(parent).X - (parent.RenderSize.Width / 2);
ct.TranslateY = e.GetPosition(parent).Y - (parent.RenderSize.Height / 2);
}
}
protected override void OnMouseLeftButtonUp(MouseButtonEventArgs e)
{
base.OnMouseLeftButtonUp(e);
isDragging = false;
this.ReleaseMouseCapture();
}
}
答案 0 :(得分:1)
这里缺少的是参考点到拖动开始的位置。在鼠标移动事件中,您尝试将控件转换为鼠标的当前位置,而不是当前和原始位置之间的差异。
或者通过使用相对坐标,您可以相对于控件存储鼠标单击的位置,并简单地从OnMove中的当前位置中减去它。这是以下示例中发生的情况。
请注意,这是没有CompositeTransform的WPF。 提取父级的RenderSize看起来像是一个错误,或者它可能是WPF和Silverlight的不同,无论如何我希望这有助于解决这个问题:
public class CustomRichTextBox : RichTextBox
{
private bool isDragging = false;
private Point draggingPoint;
protected override void OnPreviewMouseLeftButtonDown(MouseButtonEventArgs e)
{
base.OnPreviewMouseLeftButtonDown(e);
bool isDragAllowed = false;
Point pt = e.GetPosition(this);
if (pt.Y >= 0 && pt.Y <= this.BorderThickness.Top)
// Allow dragging if the mouse is down on the top border
isDragAllowed = true;
else if (pt.Y >= (this.RenderSize.Height - this.BorderThickness.Bottom) && pt.Y <= this.RenderSize.Height)
// Allow dragging if the mouse is down on the bottom border
isDragAllowed = true;
else if (pt.X >= 0 && pt.X <= this.BorderThickness.Left)
// Allow dragging if the mouse is down on the left border
isDragAllowed = true;
else if (pt.X >= (this.RenderSize.Width - this.BorderThickness.Right) && pt.X <= this.RenderSize.Width)
// Allow dragging if the mouse is down on the right border
isDragAllowed = true;
if (!isDragAllowed)
return;
draggingPoint = pt;
this.CaptureMouse();
isDragging = true;
}
protected override void OnPreviewMouseMove(MouseEventArgs e)
{
base.OnPreviewMouseMove(e);
if (isDragging)
{
CustomRichTextBox elem = this;
TransformGroup ct = new TransformGroup();
UIElement parent = (UIElement)elem.Parent;
TranslateTransform tr = new TranslateTransform(
e.GetPosition(parent).X - elem.RenderSize.Width + this.BorderThickness.Left - draggingPoint.X,
e.GetPosition(parent).Y - elem.RenderSize.Height + this.BorderThickness.Top - draggingPoint.Y);
ct.Children.Add(tr);
elem.RenderTransform = ct;
}
}
protected override void OnPreviewMouseLeftButtonUp(MouseButtonEventArgs e)
{
base.OnPreviewMouseLeftButtonUp(e);
isDragging = false;
this.ReleaseMouseCapture();
}
}