如何将鼠标滚轮事件传递给WindowsFormsHost?

3

我在WPF应用程序中有一个WinForms控件。幸运的是,在WPF中,鼠标滚轮事件通常由鼠标光标下的控件处理...但如果它是一个WinForms控件,我必须先单击控件才能让它获得键盘焦点。如何绕过这种行为,并在WinForms控件没有焦点时发送鼠标滚轮事件?

(注意:对于WindowsFormsHost,MouseWheelPreviewMouseWheel 事件似乎不起作用。如果您在WindowsFormsHost的父级中添加 PreviewMouseWheel 处理程序,则当鼠标位于非WPF控件上方时,它不会被调用。显然,WinForms控件也不会收到 MouseWheel 事件;该事件似乎完全消失了。)


你尝试在鼠标悬停事件中将焦点设置在 WFH 上了吗? - Larry
不,那样做是不正确的行为:键盘焦点不应该因为鼠标移动而改变。我的应用程序有文本框,所以用户可能会注意到焦点意外地改变了。WindowsFormsHost也不会接收MouseEnterMouseMove事件,尽管它包含的控件会接收到。 - Qwertie
1个回答

5

一种可能的解决方案是处理所有WM_MOUSEWHEEL消息,并手动为未聚焦的控件引发MouseWheel事件。我附加的行为实现了这个想法:

public static class CaptureMouseWheelWhenUnfocusedBehavior
{
    private static readonly HashSet<WindowsFormsHost> TrackedHosts =
        new HashSet<WindowsFormsHost>();

    private static readonly System.Windows.Forms.IMessageFilter MessageFilter =
        new MouseWheelMessageFilter();

    private sealed class MouseWheelMessageFilter : System.Windows.Forms.IMessageFilter
    {
        [DllImport("User32.dll"), SuppressUnmanagedCodeSecurity]
        private static extern IntPtr WindowFromPoint(System.Drawing.Point point);

        private static System.Drawing.Point LocationFromLParam(IntPtr lParam)
        {
            int x = (int)((((long)lParam) >> 0) & 0xffff);
            int y = (int)((((long)lParam) >> 16) & 0xffff);
            return new System.Drawing.Point(x, y);
        }

        private static bool ConsiderRedirect(WindowsFormsHost host)
        {
            var control = host.Child;
            return control != null &&
                  !control.IsDisposed &&
                   control.IsHandleCreated &&
                   control.Visible &&
                  !control.Focused;
        }

        private static int DeltaFromWParam(IntPtr wParam)
        {
            return (short)((((long)wParam) >> 16) & 0xffff);
        }

        private static System.Windows.Forms.MouseButtons MouseButtonsFromWParam(IntPtr wParam)
        {
            const int MK_LBUTTON = 0x0001;
            const int MK_MBUTTON = 0x0010;
            const int MK_RBUTTON = 0x0002;
            const int MK_XBUTTON1 = 0x0020;
            const int MK_XBUTTON2 = 0x0040;
            int buttonFlags = (int)((((long)wParam) >> 0) & 0xffff);
            var buttons = System.Windows.Forms.MouseButtons.None;
            if(buttonFlags != 0)
            {
                if((buttonFlags & MK_LBUTTON) == MK_LBUTTON)
                {
                    buttons |= System.Windows.Forms.MouseButtons.Left;
                }
                if((buttonFlags & MK_MBUTTON) == MK_MBUTTON)
                {
                    buttons |= System.Windows.Forms.MouseButtons.Middle;
                }
                if((buttonFlags & MK_RBUTTON) == MK_RBUTTON)
                {
                    buttons |= System.Windows.Forms.MouseButtons.Right;
                }
                if((buttonFlags & MK_XBUTTON1) == MK_XBUTTON1)
                {
                    buttons |= System.Windows.Forms.MouseButtons.XButton1;
                }
                if((buttonFlags & MK_XBUTTON2) == MK_XBUTTON2)
                {
                    buttons |= System.Windows.Forms.MouseButtons.XButton2;
                }
            }
            return buttons;
        }

        public bool PreFilterMessage(ref System.Windows.Forms.Message m)
        {
            const int WM_MOUSEWHEEL = 0x020A;
            if(m.Msg == WM_MOUSEWHEEL)
            {
                var location = LocationFromLParam(m.LParam);
                var hwnd = WindowFromPoint(location);
                foreach(var host in TrackedHosts)
                {
                    if(!ConsiderRedirect(host)) continue;
                    if(hwnd == host.Child.Handle)
                    {
                        var delta = DeltaFromWParam(m.WParam);
                        {
                            // raise event for WPF control
                            var mouse = InputManager.Current.PrimaryMouseDevice;
                            var args = new MouseWheelEventArgs(mouse, Environment.TickCount, delta);
                            args.RoutedEvent = WindowsFormsHost.MouseWheelEvent;
                            host.RaiseEvent(args);
                        }
                        {
                            // raise event for winforms control
                            var buttons = MouseButtonsFromWParam(m.WParam);
                            var args = new System.Windows.Forms.MouseEventArgs(
                                buttons, 0, location.X, location.Y, delta);
                            var method = typeof(System.Windows.Forms.Control).GetMethod(
                                "OnMouseWheel",
                                System.Reflection.BindingFlags.Instance |
                                System.Reflection.BindingFlags.NonPublic);
                            method.Invoke(host.Child, new object[] { args });
                        }
                        return true;

                    }
                }
            }
            return false;
        }
    }

    public static bool GetIsEnabled(DependencyObject obj)
    {
        return (bool)obj.GetValue(IsEnabledProperty);
    }

    public static void SetIsEnabled(DependencyObject obj, bool value)
    {
        obj.SetValue(IsEnabledProperty, value);
    }

    public static readonly DependencyProperty IsEnabledProperty =
        DependencyProperty.RegisterAttached(
            "IsEnabled",
            typeof(bool),
            typeof(CaptureMouseWheelWhenUnfocusedBehavior),
            new PropertyMetadata(false, OnIsEnabledChanged));

    private static void OnIsEnabledChanged(DependencyObject o, DependencyPropertyChangedEventArgs e)
    {
        var wfh = o as WindowsFormsHost;
        if(wfh == null) return;

        if((bool)e.NewValue)
        {
            wfh.Loaded += OnHostLoaded;
            wfh.Unloaded += OnHostUnloaded;
            if(wfh.IsLoaded && TrackedHosts.Add(wfh))
            {
                if(TrackedHosts.Count == 1)
                {
                    System.Windows.Forms.Application.AddMessageFilter(MessageFilter);
                }
            }
        }
        else
        {
            wfh.Loaded -= OnHostLoaded;
            wfh.Unloaded -= OnHostUnloaded;
            if(TrackedHosts.Remove(wfh))
            {
                if(TrackedHosts.Count == 0)
                {
                    System.Windows.Forms.Application.RemoveMessageFilter(MessageFilter);
                }
            }
        }
    }

    private static void OnHostLoaded(object sender, EventArgs e)
    {
        var wfh = (WindowsFormsHost)sender;
        if(TrackedHosts.Add(wfh))
        {
            if(TrackedHosts.Count == 1)
            {
                System.Windows.Forms.Application.AddMessageFilter(MessageFilter);
            }
        }
    }

    private static void OnHostUnloaded(object sender, EventArgs e)
    {
        var wfh = (WindowsFormsHost)sender;
        if(TrackedHosts.Remove(wfh))
        {
            if(TrackedHosts.Count == 0)
            {
                System.Windows.Forms.Application.RemoveMessageFilter(MessageFilter);
            }
        }
    }
}

它可以在XAML中使用:

<WindowsFormsHost namespace:CaptureMouseWheelWhenUnfocusedBehavior.IsEnabled="True" />

以及后台代码:

CaptureMouseWheelWhenUnfocusedBehavior.SetIsEnabled(host, true);

更新:增加了代码以触发 WinForms 控件的事件。


这个代码很好,但是它只将鼠标滚轮事件传递给WindowsFormsHost,而不会在WinForms控件上引发事件。如果您愿意,我可以编辑答案,添加必要反射命令以调用Control.OnMouseWheel方法,因为该方法是受保护的,无法直接调用。 - Qwertie
1
这是一个改进的 LocationFromLParam 方法版本,适用于多显示器系统:short x = (short) ((((long) lParam) >> 0) & 0xffff); short y = (short) ((((long) lParam) >> 16) & 0xffff); return new System.Drawing.Point(x, y); - Anton
非常好!小修正:OnMouseWheel事件参数应包括客户端坐标,而不是屏幕坐标,因此需要将它们转换为: var clientpoint = host.Child.PointToClient(location); var args = new System.Windows.Forms.MouseEventArgs( buttons, 0, clientpoint.X, clientpoint.Y, delta); - Andreas Kahler

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接