场景
我正在开发一个用于镜像目标窗口的用户控件,它在内部使用 Win32 DWM API 来注册/取消注册缩略图,并在父窗体调整大小或移动时更新缩略图位置和位置。
这是我的应用程序的结构:
应用程序/我的用户控件在“正常”条件下按预期工作(也就是说,当在操作系统中使用默认的 Windows 主题时),我录制了下一个演示用户控件用法和行为的视频,所以您可以更好地了解这一切,还可以看到控件按预期工作:
问题
当我设法为操作系统使用不同的主题时,问题就开始了,特别是任何为 Windows 10 中的窗口添加非隐形边框的主题,这可以像使用第三方软件一样重现 WindowBlinds (主题名为“Flat Dark”),也许修改注册表中的某些 Windows 度量值也可以重现 Windows 10 中可见边框的添加,但我不记得如何通过注册表执行此操作,抱歉。
嗯,最主要的是,在 Windows 10 下,当设法使用具有非隐形边框的窗口时(通过提到的第 3 方软件或通过其他可能的方式),我在用户控制类中的算法为了检索其相对于父窗体的坐标,它中断了,然后我得到了意外的坐标,因此 DWM 缩略图没有绘制在它应该绘制的正确位置。
我录制了下一个视频,您可以在其中看到差异并理解问题:
在视频中,我首先展示了程序在“正常”条件下运行,然后关闭程序,更改操作系统主题,再次运行程序,从这一点可以看到 DWM 缩略图未绘制在正确的坐标...
我的所有猜想都表明,当表单/窗口应用了不可见的 Windows 10 边框时,我遇到的问题与表单的客户端/非客户端区域有关。
为什么我这么认为?因为如果我将主题更改为具有可见边框的窗口,然后我会像这样删除表单的边框:
this.FormBorderStyle = FormBorderStyle.None;
...然后我的应用程序再次正常工作,而我的表单是无边界的,所以在这些特定情况下,这一定是与我的表单的客户端/非客户端区域相关的问题,我不明白我的意思当我在这些情况下计算我的控件的相对位置时,当窗体有边框时,我做错了。
源代码
最后,我在这里分享完整的解决方案,它包括我正在开发的用户控件以及一个演示应用程序(与您在上面的视频中看到的相同)。
请注意,源代码是用 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
在 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;
}
}
...它在我解释的情况下将意外坐标分配给 relativePos
,这就是我需要解决的问题,也是我要求的,我需要确定真正的相对坐标无论父窗体窗口的边框大小如何,我都可以有效地(通用地)对父窗体进行用户控制...
最佳答案
无论控件类型或控件的父级或控件在控件树中的位置有多深,这里有一个扩展方法可帮助您找到相对于宿主窗体的控件边界:
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);
}
}
例如:
var r = textBox1.GetBoundsRelativeToForm();
我重现了这个问题,我发现位置计算正确。但是 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)
实际上,使用这段代码,我将边框宽度和标题栏高度添加到结果中。
关于c# - 当它在 splittercontainer 中时,查找相对于父窗体的控件位置,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51715305/