c# - 确定哪个控件最接近鼠标指针

标签 c# .net winforms graphics controls

在我的 C# (.NET 2) 应用程序中,我想确定哪个控件最靠近鼠标。

我可以想出一些不太适合的方法。我可以使用 Control.Location 属性,但这只会给我顶部/左侧,鼠标可能位于控件的另一侧。我可以计算控件的中心点,但大型控件会扭曲这一点(靠近控件的边缘算作靠近控件)。

所以基本上我在 Canvas 上有一堆矩形和一个点。我需要找到离该点最近的矩形。

(理想情况下,我也想知道点和矩形之间的距离)。

有什么想法吗?

最佳答案

您需要找到以下内容:
- 到最近角落的距离
- 到最近边缘的距离
-(可选)到中心的距离

基本上,您需要这三个值中的较小值。取两个控件的最小值以确定哪个更接近。

通过迭代表单上的所有控件并创建下面类的集合来加载表单。

要找到离某个点最近的控件,请迭代集合(请参阅底部的代码)。以您目前找到的最小距离跟踪控件。如果需要,您可以测试 ContainsPoint() ... 如果找到一个控件,其中的点落在控件边界内,那么您就拥有了控件(只要您没有重叠控件)。否则,当您到达集合的末尾时,您找到的距中心/边缘距离最短的控件就是您的控件。

public class HitControl {

    public Control ThisControl;

    private Rectangle ControlBounds;
    private Point Center;

    public HitControl (Control FormControl) {
        ControlBounds = FormControl.Bounds;
        Center = new Point(ControlBounds.X + (ControlBounds.Width/2), ControlBounds.Y + (ControlBounds.Height/2));
    }

    //  Calculate the minimum distance from the left, right, and center
    public double DistanceFrom(Point TestPoint) {

        //  Note:  You don't need to consider control center points unless
        //  you plan to allow for controls placed over other controls... 
        //  Then you need to test the distance to the centers, as well, 
        //  and pick the shortest distance of to-edge, to-side, to-corner

        bool withinWidth = TestPoint.X > ControlBounds.X && TestPoint.X < ControlBounds.X + ControlBounds.Width;
        bool withinHeight = TestPoint.Y > ControlBounds.Y && TestPoint.Y < ControlBounds.Y + ControlBounds.Height;

        int EdgeLeftXDistance = Math.Abs(ControlBounds.X - TestPoint.X);
        int EdgeRightXDistance = Math.Abs(ControlBounds.X + ControlBounds.Width - TestPoint.X);

        int EdgeTopYDistance = Math.Abs(ControlBounds.Y - TestPoint.Y);
        int EdgeBottomYDistance = Math.Abs(ControlBounds.Y + ControlBounds.Height - TestPoint.Y);

        int EdgeXDistance = Math.Min(EdgeLeftXDistance, EdgeRightXDistance);
        int EdgeYDistance = Math.Min(EdgeTopYDistance, EdgeBottomYDistance);


        // Some points to consider for rectangle (100, 100, 100, 100):
        //  - (140, 90):  Distance to top edge
        //  - (105, 10):  Distance to top edge
        //  - (50, 50):   Distance to upper left corner
        //  - (250, 50):  Distance to upper right corner
        //  - (10, 105):  Distance to left edge
        //  - (140, 105):  Distance to top edge
        //  - (105, 140):  Distance to left edge
        //  - (290, 105):  Distance to right edge
        //  - (205, 150):  Distance to right edge
        //  ... and so forth


        //  You're within the control
        if (withinWidth && withinHeight) {
            return Math.Min(EdgeXDistance, EdgeYDistance);
        }

        //  You're above or below the control
        if (withinWidth) {
            return EdgeYDistance;
        }

        //  You're to the left or right of the control
        if (withinHeight) {
            return EdgeXDistance;
        }

        //  You're in one of the four outside corners around the control.
        //  Find the distance to the closest corner
        return Math.Sqrt(EdgeXDistance ^ 2 + EdgeYDistance ^ 2);


    }

    public bool ContainsPoint (Point TestPoint) {
        return ControlBounds.Contains(TestPoint);
    }


}



//  Initialize and use this collection
List<HitControl> hitControls = (from Control control in Controls
                                select new HitControl(control)).ToList();

Point testPoint = new Point(175, 619);
double distance;
double shortestDistance = 0;
HitControl closestControl = null;

foreach (HitControl hitControl in hitControls) {

    //  Optional... works so long as you don't have overlapping controls
    //  If you do, comment this block out
    if (hitControl.ContainsPoint(testPoint)) {
        closestControl = hitControl;
        break;
    }

    distance = hitControl.DistanceFrom(testPoint);
    if (shortestDistance == 0 || distance < shortestDistance) {
        shortestDistance = distance;
        closestControl = hitControl;
    }
}

if (closestControl != null) {
    Control foundControl = closestControl.ThisControl;
}

关于c# - 确定哪个控件最接近鼠标指针,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3969344/

相关文章:

c# - 如何在 Selenium 中选择多个元素(不是 <Select> - <Option>)

c# - 没有内置方法的舍入

c# - 如果存在无效的属性值,Asp.net core 不会绑定(bind) post 模型

c# - 在写入文件之前压缩 XML

c# - 对象初始化可以简化:IDE0017

c# - .Net Regex.Escape 无法正常工作?

c# - 复选框绑定(bind)慢

c# - Object cannot be cast from DBNull to other types 错误显示?

vb.net - 在VB.NET中删除DataGridView控件中的一行

c# - Crystal Reports 版本 13.0 中报告失败错误?