- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我遇到了 WM_NCHITTEST
的奇怪行为消息。
总而言之,一旦我将鼠标悬停在目标(即:Hooked)控件上并保持鼠标静止(或空闲),我就会收到 无休止地百位 WM_NCHITTEST
每秒消息。无论我是否对 WndProc
进行子类化都会发生这种情况该控件的 WindowProc()
, 或者如果我 覆盖 WndProc
子类中的方法(为了简单起见,我在下面的代码中子类化)。
据我从在线 Win32 API 文档和其他来源中找到的信息,我怀疑此消息是否以这种频率触发,但我可能是错的。或者可能有一个我完全错过的明显解释,或者我不知道的 API 发生了一些变化。无论如何,我真的很想知道它是什么,或者正在发生什么。
我已经在两个不同的系统上测试了相同的代码(下面的示例),结果相同,尽管两个系统都在相同的 Delphi/OS 版本和配置中。我尝试在 IDE 之外运行应用程序(因此没有调试 Hook ),在调试和发布配置(后者没有调试信息)中,针对 32 位和 64 位,我总是得到相同的结果。
我正在 Win10 Pro 64 位版本 20H2(我相信是最新的 Windows 版本)下使用 Delphi XE7 Enterprise 进行开发。
这是一个非常简单的程序来重现我所经历的:TForm
与 TPanel
, TCheckBox
, 和 TLabel
.面板是勾选复选框时被钩住的控件,标签显示的是多少WM_NCHITTEST
消息由 WndProc()
接收方法:
unit Unit5;
interface
uses
Winapi.Windows, Winapi.Messages, System.SysUtils, System.Variants, System.Classes, Vcl.Graphics,
Vcl.Controls, Vcl.Forms, Vcl.Dialogs, Vcl.ExtCtrls, Vcl.StdCtrls;
type
TForm5 = class(TForm)
CheckBox1: TCheckBox;
Label1: TLabel;
Panel1: TPanel;
procedure FormDestroy(Sender: TObject);
procedure CheckBox1Click(Sender: TObject);
private
FHookedCtrl: TControl;
FHookedCtrlWndProc: TWndMethod;
FMessageCount: Integer;
procedure SetHookedCtrl(const Value: TControl);
public
procedure ControlWndProc(var Message: TMessage);
property HookedCtrl: TControl read FHookedCtrl write SetHookedCtrl;
end;
var
Form5: TForm5;
implementation
{$R *.dfm}
{ TForm5 }
procedure TForm5.CheckBox1Click(Sender: TObject);
begin
//checkbox activates or deactivates the hook
if CheckBox1.Checked then
//hook the panel's WndProc by subclassing
HookedCtrl := Panel1
//release the hook on WndProc
else HookedCtrl := nil;
end;
procedure TForm5.ControlWndProc(var Message: TMessage);
begin
case Message.Msg of
WM_NCHITTEST:
begin
//show how many messages received with the label's caption
Inc(FMessageCount);
Label1.Caption := FormatFloat('##,##0 messages', FMessageCount);
end;
end;
//not really handling the messsage, just counting.
FHookedCtrlWndProc(Message);
end;
procedure TForm5.FormDestroy(Sender: TObject);
begin
//make sure to clear the hook if assigned
HookedCtrl := nil;
end;
procedure TForm5.SetHookedCtrl(const Value: TControl);
begin
if (Value <> FHookedCtrl) then
begin
if Assigned(FHookedCtrl) then
begin
//release the hook
FHookedCtrl.WindowProc := FHookedCtrlWndProc;
FHookedCtrlWndProc := nil;
FMessageCount := 0;
end;
FHookedCtrl := Value;
if Assigned(FHookedCtrl) then
begin
//hook the panel (i.e. Value)
FHookedCtrlWndProc := FHookedCtrl.WindowProc;
FHookedCtrl.WindowProc := ControlWndProc;
end;
end;
end;
end.
重现:运行应用程序,选中 CheckBox,将鼠标悬停在面板上并让它保持空闲(静止)。就我而言,我收到了 100 个
WM_NCHITTEST
每秒消息,它永远不会停止。这应该发生吗?
最佳答案
我用过 Microsoft Spy++工具来查看会发生什么以及何时发生。
它是 WM_NCHITTEST 处理程序中的以下行
Label1.Caption := FormatFloat('##,##0 messages', FMessageCount);
这导致了这个问题。当您删除它时,所有这些
WM_NCHITTEST
都没有了。消息。要查看消息数量,请使用
TTimer
以 1 秒的间隔并在标签中显示消息计数。你会看到你得到一个
WM_NCHITTEST
每次计时器触发时(如果您有一个空的 OnTimer 处理程序,您仍然会收到一条消息),当然还有鼠标移动时。
unit Unit5;
interface
uses
Winapi.Windows, Winapi.Messages, System.SysUtils, System.Variants, System.Classes, Vcl.Graphics,
Vcl.Controls, Vcl.Forms, Vcl.Dialogs, Vcl.ExtCtrls, Vcl.StdCtrls;
type
TForm5 = class(TForm)
Label1: TLabel;
CheckBox1: TCheckBox;
Panel1: TPanel;
Timer1: TTimer;
procedure FormDestroy(Sender: TObject);
procedure CheckBox1Click(Sender: TObject);
procedure Timer1Timer(Sender: TObject);
private
FHookedCtrl: TControl;
FHookedCtrlWndProc: TWndMethod;
FMessageCount: Integer;
procedure SetHookedCtrl(const Value: TControl);
public
procedure ControlWndProc(var Message: TMessage);
property HookedCtrl: TControl read FHookedCtrl write SetHookedCtrl;
end;
var
Form5: TForm5;
implementation
{$R *.dfm}
{ TForm5 }
procedure TForm5.CheckBox1Click(Sender: TObject);
begin
//checkbox activates or deactivates the hook
if CheckBox1.Checked then
//hook the panel's WndProc by subclassing
HookedCtrl := Panel1
else
//release the hook on WndProc
HookedCtrl := nil;
end;
procedure TForm5.ControlWndProc(var Message: TMessage);
begin
case Message.Msg of
WM_NCHITTEST:
//Count how many messages received
Inc(FMessageCount);
end;
//not really handling the messsage, just counting.
FHookedCtrlWndProc(Message);
end;
procedure TForm5.FormDestroy(Sender: TObject);
begin
//make sure to clear the hook if assigned
HookedCtrl := nil;
end;
procedure TForm5.SetHookedCtrl(const Value: TControl);
begin
if (Value <> FHookedCtrl) then begin
if Assigned(FHookedCtrl) then begin
//release the hook
FHookedCtrl.WindowProc := FHookedCtrlWndProc;
FHookedCtrlWndProc := nil;
FMessageCount := 0;
end;
FHookedCtrl := Value;
if Assigned(FHookedCtrl) then begin
//hook the panel (i.e. Value)
FHookedCtrlWndProc := FHookedCtrl.WindowProc;
FHookedCtrl.WindowProc := ControlWndProc;
end;
end;
end;
procedure TForm5.Timer1Timer(Sender: TObject);
begin
// Show how many message received
Label1.Caption := FormatFloat('##,##0 messages', FMessageCount);
end;
end.
关于delphi - WM_NCHITTEST 是否应该由 Win10 以每秒 100 次的频率永久生成,即使鼠标处于空闲状态?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/66501361/
我注意到当光标位于标题栏内(不在边框上)时,消息 WM_NCHitTest 不会发送到表单。 我尝试使用以下任一方法拦截它 procedure WMNCHitTest(var Message: TWM
所描述的第 2 个显示器位于主显示器左侧的设置导致 WM_NCHITTEST 发送负值,根据 this 显然不支持该负值发布。 我有一个用 win32 编写的自定义控件,它类似于组控件。它有一个小的可
我正在玩 Windows API 试图了解它的行为方式,我意识到我可以完全删除 WNDPROC 并使用裸事件循环处理事情,如下所示: #include static struct { HWN
我正在通过指定边框样式 NONE 和 WM_NCHITTEST 的自定义处理来创建自定义绘制窗口。我已将某些区域定义为“我的窗口标题”,并在该区域中返回 WM_NCHITTEST 的 HTCAPTIO
case WM_NCHITTEST: { LRESULT hit = DefWindowProc( hWnd, uMsg, wParam, lParam ); if ( h
我正在尝试允许用户移动/调整 CEF 无边界窗口(使用 WS_POPUP 标志创建)。 鼠标位置在 CEF 浏览器中捕获(使用 Javascript)并调用 C++ 回调。然后,我通过 IPC 从 C
我在 Visual C++ 2010 中使用 wxWidgets。 我的目标之一是能够 move 我用窗口的任何部分(客户端或其他)创建的框架。为此,我过去曾使用 WM_NCHITTEST 让 Win
如何在 C# 代码中获取 WM_NCHITTEST 消息的坐标? 我希望获得最快的方式,因为性能是一项要求。 最佳答案 直到今天早上,我都会 100% 同意 Thomas Levesques 的回答,
我正在使用本文的代码http://melander.dk/articles/alphasplash/在表单中显示 32 位位图,但是当我尝试使用纯色位图而不是图像时,未收到 WM_NCHITTEST
我正在使用 win32 创建一个带有自定义标题栏和边框的窗口。我的问题是,当我使用 WM_NCHITTEST WM_LBUTTONDOWN 时,点击事件不会被触发。当我没有使用 WM_NCHITTES
我正在制作一个应用程序,它将自身挂接到目标应用程序,并且在被用户激活时,阻止所有键盘和鼠标窗口消息到达目标应用程序的窗口进程。我的应用程序通过将传入的输入消息(例如 WM_MOUSEMOVE)转换为
我创建了一个无边界窗口,它使用一个 wndProc() 函数强制 WM_NCHITTEST case 返回 HTCAPTION;,它允许用户拖动窗口,无论光标位于何处。 问题是我已经设置了自定义光标,
所以我创建了这个带有自定义 move 和调整大小的弹出窗口,按住 ctrl 键并单击客户端中的任意位置以 move 和 alt 调整边界大小。问题是,现在我正在为这个窗口内部创建子窗口,这里它“出错”
我用 C++ 创建了一个小型 MFC 文档 View 应用程序,但在从 CStatic 继承的类中接收消息时遇到了一些问题。我已经设法创建了 CStatic 派生,它在我的 View 中可见,但是我的
在什么情况下,消息WM_NCHITTEST 的lParam 会是一个不适合32 位整数的值? 由于我们的 WPF 应用程序中存在未处理的异常,我们的一位客户在他的 64 位计算机上遇到了崩溃,我很难找
我遇到了 WM_NCHITTEST 的奇怪行为消息。 总而言之,一旦我将鼠标悬停在目标(即:Hooked)控件上并保持鼠标静止(或空闲),我就会收到 无休止地百位 WM_NCHITTEST每秒消息。无
我是一名优秀的程序员,十分优秀!