0
0

一个更好的 AlwaysMouseWhell,兼谈如何写 DPI-Aware 的 windows 程序

Typeof 发表于 1970年01月01日 08:00 | Hits: 959

我在知乎上这篇关于 AlwaysMouseWheel 的文章说过,AWM 不支持高 dpi,使用有 bug。为什么我会发现呢?因为我有一个笔记本换上了 1080p 的屏幕,然后把 dpi 改成了 1.25x。Windows 从 vista 开始引入了 DPI 虚拟化技术,让 dwm 有权力直接放大窗口,而为了确保兼容性,许多 API——包括GetCursorPos和WindowFromPoint——都做了一些处理。

具体表现就是,在旧程序的窗口上方,GetCursorPos的结果并非鼠标的物理座标,而是经过重新计算的新位置。如果要获取真实的鼠标座标,需要使用 Vista 开始提供的新 API:GetPhysicalCursorPos。WindowFromPoint函数也不能用(事实上使用它会得到相当鬼畜的结果),应该使用WindowFromPhysicalPoint。显然AlwaysMouseWheel没有正确地使用新 api,于是它在高 dpi 下就不能正常工作了。

AlwaysMouseWheel 是个典型的鼠标钩子。在 Windows 下,写这种钩子程序最简单的办法是用AutoHotkey。可能是教主光环过于强大,我在 Github 上找到了一个类似的项目,然后制作了一个正确的版本。主要修改的地方有:

  • 使用GetPhysicalCursorPos获取鼠标座标,替换原来版本里的 AHK 函数
  • 使用WindowFromPhysicalPoint获取指针下的窗口
  • 在送滚轮消息前,将GetPhysicalCursorPos获取得到的物理座标用PhysicalToLogicalPoint转换成逻辑座标,生成用于SendMessage的lparam

各位下载后,用 autohotkey 最新版运行即可。当然,它必须在 vista 及以上版本的 windows 里方可运行。

其实支持高 dpi,涉及到窗口间互操作的就是 Logical 和 Physical 点的变换问题了。不过我还是挺纳闷的,DPI 虚拟化这个 06 年就有的技术为什么到现在还没几个软件跟进呢?

原文链接: http://typeof.net/m/a-better-focusless-mouse-wheel.html

0     0

评价列表(0)