当控件位于 splittercontainer 内时,查找相对于父窗体的控件位置

Ele*_*ios 4 c# vb.net rectangles winforms windows-10

设想


我正在开发一个用于镜像目标窗口的用户控件,它在内部使用 Win32 DWM API 来注册/取消注册缩略图,并在父窗体调整大小或移动时更新缩略图位置和位置。

这是我的应用程序的结构:

在此输入图像描述

应用程序/我的用户控件在“正常”条件下按预期工作(这意味着,在操作系统中使用默认的 Windows 主题时),我录制了下一个视频,演示了用户控件的使用和行为,因此您可以获得对这一切有更好的了解,并看到控制按预期工作:

问题


当我设法为操作系统使用不同的主题时,问题就开始了,特别是在 Windows 10 中为窗口添加不可见边框的任何主题,这可以像使用WindowBlinds等第三方软件(主题名为“ Flat Dark ”),也许修改注册表中的某些 Windows 指标值也可以重现 Windows 10 中添加可见边框的情况,但我不记得如何通过注册表执行此操作,抱歉。

好吧,最主要的是,在 Windows 10 中,当设法使用具有非隐形边框的窗口时(通过提到的第 3 方软件或通过其他可能的方式),我在用户控件类中使用的算法来检索其相对于父窗体的坐标,它会中断,然后我得到意外的坐标,因此 DWM 缩略图未绘制在应绘制的正确位置。

我录制了下一个视频,您可以在其中看到差异并理解问题:

在视频中,我首先展示了在“正常”条件下运行的程序,然后关闭程序,更改操作系统主题,再次运行程序,从这一点上您可以看到 DWM 缩略图没有以正确的方式绘制坐标...

我所有的猜测都表明,当表单/窗口应用了不可见的 Windows 10 边框时,我遇到的问题与表单的客户端/非客户端区域有关。

为什么我这么想?因为如果我将主题更改为具有可见边框的窗口,然后我像这样删除表单的边框:

this.FormBorderStyle = FormBorderStyle.None;
Run Code Online (Sandbox Code Playgroud)

...然后我的应用程序再次正常工作,而我的表单是无边框的,因此在这些特定情况下,这一定是与我的表单的客户/非客户区域相关的问题,而且我不明白我做错了什么当我在这些情况下计算控件的相对位置时,当表单有边框时。

源代码


最后,我在这里分享完整的解决方案,它包括我正在开发的用户控件以及一个演示应用程序(与您在上面的视频中看到的相同)。

请注意,源代码是用 VB.NET 编写的,但这一事实与我在这个问题中标记的语言无关,因为我接受 C# 或 VB.NET 中的任何解决方案,所以请不要指责这一点,因为标记的语言问题中的问题是一回事,而用一种特定语言编写的共享解决方案是另一回事。

没有必要下载和检查源代码,所有源代码中唯一相关的部分是 的坐标分配relativePos,这里:

Public Class ElektroDwmThumbnail : Inherits UserControl

    Protected Function GetThumbnailRectangle() As Rectangle
        Dim relativePos As Point = Me.ParentForm.PointToClient(Me.PointToScreen(Point.Empty))
        ' ...
        Dim dstRectangle As New Rectangle(relativePos, thumbnailSize)
        Return dstRectangle
    End Function

End Class
Run Code Online (Sandbox Code Playgroud)

在 C# 中将是:

public class ElektroDwmThumbnail: UserControl {

    protected Rectangle GetThumbnailRectangle() {
        Point relativePos = this.ParentForm.PointToClient(this.PointToScreen(Point.Empty));
        // ...
        Rectangle dstRectangle = new Rectangle(relativePos, thumbnailSize);
        return dstRectangle;
    }

}
Run Code Online (Sandbox Code Playgroud)

...它在我解释的情况下分配了意外的坐标relativePos,这就是我需要解决的问题和我所要求的,我需要有效地确定用户控件与父窗体的真实相对坐标(通用),无论父窗体窗口的边框大小如何......

Rez*_*aei 5

无论控件类型或控件的父级或控件在控件树中的位置如何,这里都有一个扩展方法可以帮助您找到相对于宿主窗体的控件边界:

using System;
using System.Drawing;
using System.Windows.Forms;

public static class ControlExtensions
{
    public static Rectangle GetBoundsRelativeToForm(this Control c)
    {
        if (c == null)
            throw new ArgumentNullException(nameof(c));

        var form = c.FindForm();
        if (form == null)
            throw new InvalidOperationException("The control is not located on a form.");

        var parent = c.Parent;
        if (parent == null)
            throw new InvalidOperationException("The control does not have a parent.");

        var p = form.PointToClient(parent.PointToScreen(c.Location));
        return new Rectangle(p, c.Size);
    }
}
Run Code Online (Sandbox Code Playgroud)

例如:

var r = textBox1.GetBoundsRelativeToForm();
Run Code Online (Sandbox Code Playgroud)

我重现了该问题,并且发现位置计算正确。但是DwmRegisterThumbnail假设整个窗口区域作为客户区域,而期望使用客户区域。

我认为这是主题的问题,作为快速解决方案,我通过以下方式更正了位置:

Dim p0 As Point = Me.ParentForm.PointToScreen(Point.Empty)
Dim p1 As Point = Me.ParentForm.DesktopLocation
Dim relativePos As Point = Me.ParentForm.PointToClient(Me.PointToScreen(Point.Empty)) 
relativePos.X += (p0.X - p1.X)
relativePos.Y += (p0.Y - p1.Y) 
Run Code Online (Sandbox Code Playgroud)

事实上,使用这段代码,我将边框宽度和标题栏高度添加到结果中。