繁体   English   中英

选择矩形(通过鼠标在画布上绘制)无法识别位于该选择矩形内的矩形(C# WPF)

[英]A selection-rectangle (drawn by mouse on canvas) does not recognize rectangles, which are inside that selection-rectangle (C# WPF)

我现在真的很绝望。 我只是无法更进一步,原则上它不应该那么困难。 我尝试在过去 2 周内解决它。 以下任务/问题:我的 C# WPF 项目中有一个名为“MarkPointsMap”的 canvas 项目,上面有不同种类的标记点(3种不同类型的不同颜色的标记点) 一个 MarkPointsMap 可以在整个 MarkPointsMap 中包含大约 3000-4000 个不同的 MarkPoints(该画布的子项)。 MarkPoints 的位置从一个.tab 文件中加载,该文件被转换为一个名为“MarkPoints”的列表,并在加载相应的.tab 文件后作为子项添加到“MarkPointsMap”canvas 上。

这是 MarkPointsMap 的屏幕截图: 这是 MarkPointsMap 的屏幕截图:

例如,大部分是小矩形(绿色的),一些中间的矩形(黄色的)和两个大的矩形(红色的)。 应使用选择矩形选择多个标记点。 在我的项目中,我已经可以绘制选择矩形,但是“if this.selectRect.Contains(MarkPointsMap.Children)”(见下文)部分不起作用。

选择矩形如下所示:

选择矩形如下所示:

以下方法“drawMarkPoints”,在程序中加载.tab文件后调用,在canvas“MarkPointsMap”上绘制标记点:

public void drawMarkPoints()
    {
        MarkPointsMap.Children.Clear();

        foreach (MarkPoint markpoint in this.Marks.markpoints)
        {
            if (markpoint.type.relevant)
            {
                Rectangle markpointArea = new Rectangle();
                markpointArea.Tag = markpoint;
                //!!
                SolidColorBrush mySolidColorBrush = new SolidColorBrush();
                mySolidColorBrush.Color = markpoint.fillColor;
                markpointArea.Fill = mySolidColorBrush;
                markpointArea.StrokeThickness = 0.2;
                markpointArea.Stroke = Brushes.Black;

                markpointArea.Width = markpoint.symbol.Width;
                markpointArea.Height = markpoint.symbol.Height;
                //set the markpoints in canvas 
                Canvas.SetLeft(markpointArea, getCanvasCoordinates(markpoint.center).X - 0.5 * markpoint.symbol.Width); 
                Canvas.SetTop(markpointArea, getCanvasCoordinates(markpoint.center).Y - 0.5 * markpoint.symbol.Height);

                MarkPointsMap.Children.Add(markpointArea);
          
            }
        }
    }

选择矩形在 canvas“MarkPointsMap”中绘制,带有三个鼠标事件(MarkPointsMap_MouseDown、MarkPointsMap_MouseMove、MarkPointsMap_MouseUp):

public Rect itemRect;
public Rect selectRect;
private point anchorPoint; 
private Rectangle manualDragRect = new Rectangle(); 

private void MarkPointsMap_MouseDown(object sender, MouseButtonEventArgs e)
    {
        if (e.ChangedButton == MouseButton.Left)
        {
            isLeftMouseButtonDownOnWindow = true;
            anchorPoint = e.MouseDevice.GetPosition(MarkPointsMap);
            MarkPointsMap.CaptureMouse();
            e.Handled = true;
            manualDragRect = new Rectangle
            {
                Stroke = Brushes.Red,
                StrokeThickness = 2
            };
            MarkPointsMap.Children.Add(manualDragRect); 
        }       
    }

private void MarkPointsMap_MouseMove(object sender, MouseEventArgs e)
    {
        if (!MarkPointsMap.IsMouseCaptured)
            return;

        Point location = e.MouseDevice.GetPosition(MarkPointsMap);
        double minX = Math.Min(location.X, anchorPoint.X);
        double minY = Math.Min(location.Y, anchorPoint.Y);
        double maxX = Math.Max(location.X, anchorPoint.X);
        double maxY = Math.Max(location.Y, anchorPoint.Y);

        Canvas.SetTop(manualDragRect, minY);
        Canvas.SetLeft(manualDragRect, minX);

        double height = maxY - minY;
        double width = maxX - minX;

        manualDragRect.Height = Math.Abs(height);
        manualDragRect.Width = Math.Abs(width);
        Point xy_2 = new Point((Canvas.GetLeft(manualDragRect) + 0.5 * width), (Canvas.GetTop(manualDragRect) + 0.5 * height));
        this.selectRect = new Rect(xy_2.X, xy_2.Y, width, height);
    }


private void MarkPointsMap _MouseUp(object sender, MouseButtonEventArgs e)
    {
        DefectMap.ReleaseMouseCapture();

        foreach (MarkPoint markpoint in this.Marks.markpoints) 
        {
            Rect itemRect = new Rect(markpoint.center.X, markpoint.center.Y, markpoint.symbol.Width, markpoint.symbol.Height);
            if (selectRect.Contains(itemRect))
            {
                MessageBox.Show("Test!"); // it does not reach this code, if I circle several markpoints with the red selection-rectangle called “selectRect”
                
            }

        }
    }

为什么这不起作用? 我想这与从矩形(使用 derictive 的 System.Windows.Shapes)到 struct Rect:IFormattable 的转换有关。 未达到的“if (selectRect.Contains(itemRect)”应在透视图中将选择矩形内的所有标记点着色为红色,然后计算中间的点(x-Coordinate,y-Coordinate) -选择矩形的点并将这个中间点添加回 original.tab 文件。
任何想法或提示,我该如何继续? 提前致谢。 此致!

看来你的计算是错误的。 您正在抵消selectRect 您将其相对于渲染的manualDragRect 这样,您测试的区域与您通过呈现的选择范围定义的区域完全不同。

此外,您正在Canvas上捕获鼠标,但在某些不同的控件上释放捕获。 它也应该在MarkPointsMap上发布。 还要小心将鼠标事件标记为已处理。 如果输入处理不是自定义控件的一部分,您通常应该避免使用它。

您应该考虑使用带有CanvasItemsControl作为面板。 然后,您可以将绘图移动到标记并为MarkPoint数据定义DataTemplate

我建议使用从manualDragRect形状返回的原始Rect 这意味着您应该删除selectRect字段和相关计算:

private void MarkPointsMap_MouseUp(object sender, MouseButtonEventArgs e)
{
  MarkPointsMap.ReleaseMouseCapture();

  foreach (MarkPoint markpoint in this.Marks.markpoints)
  {
    Rect itemRect = new Rect(markpoint.center.X, markpoint.center.Y, markpoint.symbol.Width, markpoint.symbol.Height);

    Rect selectionBounds = manualDragRect.RenderedGeometry.Bounds;
    selectionBounds.Offset(Canvas.GetLeft(manualDragRect), Canvas.GetTop(manualDragRect));
    if (selectionBounds.Contains(itemRect))
    {
      MessageBox.Show("Test!");
    }

    // TODO::Calculate center of selectionBounds
    Point selectionBoundsCenter = ...;
  }
}

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM