微软终于开始学苹果一样好好做触摸板了(就是键盘空格键下面那一大块)。然而鉴于以前没有好好做,以至于 WPF 程序甚至都没有对触摸板的横向滚动提供支持(竖向滚动是直接使用了 MouseWheel,汗……)。但有些功能真希望能够支持横向滚动!

本文将介绍让触摸板支持横向滚动的方法,本质上也是用 MouseWheel,但却支持了横向。

我们需要从 Windows 的窗口消息中获取 WM_MOUSEHWHEEL 消息。对,就是鼠标滚轮消息!以前我们只取了纵向数据,现在我们要取横向数据。

首先,我们需要能够监听得到消息才行。重写 WindowOnSourceInitialized 方法可以开始监听消息;如果代码没办法写到 Window 中,可以通过 Window.GetWindow(DependencyObject) 获取到窗口实例后监听它的 SourceInitialized 事件。如果拿不到这样的时机,则只要在任何 SourceInitialized 之后的时机(比如 Loaded )都可以写下面方法内部的两行代码。

protected override void OnSourceInitialized(EventArgs e) { var source = PresentationSource.FromVisual(_board); ((HwndSource) source)?.AddHook(Hook); } private IntPtr Hook(IntPtr hwnd, int msg, IntPtr wParam, IntPtr lParam, ref bool handled) { // 在这里添加消息的处理。 return IntPtr.Zero; }
C

接下来,我们开始处理 WM_MOUSEHWHEEL

const int WM_MOUSEHWHEEL = 0x020E; private IntPtr Hook(IntPtr hwnd, int msg, IntPtr wParam, IntPtr lParam, ref bool handled) { switch (msg) { case WM_MOUSEHWHEEL: int tilt = (short) HIWORD(wParam); OnMouseTilt(tilt); return (IntPtr) 1; } return IntPtr.Zero; } /// <summary> /// 取指针所在高位数值。 /// </summary> private static int HIWORD(IntPtr ptr) { var val32 = ptr.ToInt32(); return ((val32 >> 16) & 0xFFFF); } /// <summary> /// 取指针所在低位数值。 /// </summary> private static int LOWORD(IntPtr ptr) { var val32 = ptr.ToInt32(); return (val32 & 0xFFFF); } private void OnMouseTilt(int tilt) { // 这里就是触摸板横向滚动的时机,参数是横向滚动的数值,就像鼠标滚轮纵向滚动的数值一样。 }
C

OnMouseTilt 中就可以写我们触摸板横向滚动的处理代码。

以上代码都可以封装成通用的方法,在 OnMouseTilt 中抛出一个类似于 MouseWheel 一样的事件是非常好的选择。

微软的 Microsoft Sculpt Comfort Mouse 鼠标滚轮也是支持横向滚动的,以上方法也可以支持。


但是,原文关于取 LOWORDHIWORD 有问题,在 64 位系统上,反向滚动会造成 Arithmetic Overflow, 所以下面是修改后的代码:

/// <summary> /// 取指针所在高位数值。 /// </summary> private static int HIWORD(IntPtr ptr) { unchecked { if (Environment.Is64BitOperatingSystem) { var val64 = ptr.ToInt64(); return (short) ((val64 >> 16) & 0xFFFF); } var val32 = ptr.ToInt32(); return (short) ((val32 >> 16) & 0xFFFF); } } /// <summary> /// 取指针所在低位数值。 /// </summary> private static int LOWORD(IntPtr ptr) { unchecked { if (Environment.Is64BitOperatingSystem) { var val64 = ptr.ToInt64(); return (short)(val64 & 0xFFFF); } var val32 = ptr.ToInt32(); return (short)(val32 & 0xFFFF); } }
C

下面一段代码是评论中t3brown的评论,可以获取鼠标所在位置的滚动视图并进行滚动,十分方便:

private void OnMouseTilt(int tilt) { UIElement element = Mouse.DirectlyOver as UIElement; if (element == null) return; ScrollViewer scrollViewer = element is ScrollViewer viewer ? viewer : FindParent<ScrollViewer>(element); if (scrollViewer == null) return; scrollViewer.ScrollToHorizontalOffset(scrollViewer.HorizontalOffset + tilt); } public static T FindParent<T>(DependencyObject child) where T : DependencyObject { DependencyObject parentObject = VisualTreeHelper.GetParent(child); if (parentObject == null) return null; T parent = parentObject as T; if (parent != null) return parent; else return FindParent<T>(parentObject); }
C
最后修改:2021 年 11 月 22 日 12 : 39 PM
真的不买杯奶茶嘛....qwq