我正在开发一个用于镜像目标窗口的用户控件,它在内部使用Win32 DWM API来注册/取消注册缩略图,并在父窗体调整大小或移动时更新缩略图的位置和位置。
这是我的应用程序的结构:
应用程序/我的用户控件在“正常”条件下正常工作(这意味着,在操作系统中使用默认Windows主题时),我录制了下一个视频,演示了用户控件的用法和行为,因此您可以更好地了解所有这些内容,并看到控件按预期工作:
当我设法为操作系统使用不同的主题(特别是为Windows 10中的Windows添加非可见边框的任何主题)时,问题就开始出现,可以像使用{{3}这样的第三方软件来重现该问题。 }(主题名为“ WindowBlinds”),也许修改注册表中的某些Windows指标值可能也会重现Windows 10中可见边框的添加,但是我不记得如何通过注册表来做到这一点,对不起
好吧,最主要的是,在Windows 10中,当设法使用具有不可见边框的窗口(通过提到的第三方软件或通过其他可能的方式)时,我在用户控件类中拥有的算法检索其相对于父窗体的相对坐标,它会中断,然后我得到了意外的坐标,因此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
分配了意外坐标,这就是我需要解决的问题以及我要的是,我需要确定用户的真实相对坐标-无论父窗体窗口的边框大小如何,都可以有效地(通用)控制父窗体...
答案 0 :(得分:3)
不管控件的类型,控件的父级或控件在控件树中的位置,此扩展方法都可以帮助您找到相对于宿主形式的控件范围:
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)
事实上,使用此代码,我在结果中添加了边框宽度和标题栏高度。
答案 1 :(得分:1)
下面的示例代码将帮助您找到控件相对于表单的位置。表单可以是无边界的或有边界的。
使用以下方法作为控件类型的扩展名。或者只是从参数中删除该关键字,使其成为常规方法
public static Point GetPositionInForm(this Control ctrl)
{
Point p = ctrl.Location;
Control parent = ctrl.Parent;
Form frm = ctrl.FindForm();
Rectangle screenRectangle = frm.RectangleToScreen(frm.ClientRectangle);
int titleHeight = screenRectangle.Top - frm.Top;
int leftMargin = screenRectangle.Left - frm.Left;
p.Offset(leftMargin, titleHeight);
while (!(parent is Form))
{
p.Offset(parent.Location.X, parent.Location.Y);
parent = parent.Parent;
}
return p;
}