C # Есть ли способ обнаружить щелчок мыши от дополнительных кнопок мыши с крючками?

В настоящее время я работаю над личным проектом, в котором пытаюсь создать калькулятор APM, например. Чтобы понять это, мне нужно уметь обнаруживать любую нажатую клавишу и любой щелчок мышью. Я использую хуки для отслеживания событий клавиатуры и мыши, но я не могу найти способ зафиксировать щелчок специальной кнопкой мыши ...

Когда я говорю о специальной кнопке мыши, я имею в виду дополнительные кнопки, как у игровых мышей. Пример: Logitech G600.

Я хотел бы знать, есть ли способ обнаружить щелчок мышью любой кнопкой мыши или специальной кнопкой мыши?

Я уже нашел, как это сделать для обычные кнопки и даже для кнопки навигации, но я не могу найти ничего о дополнительных кнопках.

Вот мой код, в котором я удалил всю часть клавиатуры:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.InteropServices;
using System.Threading.Tasks;
using System.Windows.Forms;
using System.ComponentModel;
using System.Diagnostics;
using System.Text;
using System.Windows.Input;

namespace MouseHook
{
    static class Program
    {
        private const int WH_MOUSE_LL = 14;
        private static LowLevelMouseProc _procMouse = HookCallbackMouse;
        private static IntPtr _hookIDMouse = IntPtr.Zero;

        private enum MouseMessages
        {
            WM_LBUTTONDOWN = 0x0201,
            WM_LBUTTONUP = 0x0202,
            WM_MOUSEMOVE = 0x0200,
            WM_MOUSEWHEEL = 0x020A,
            WM_RBUTTONDOWN = 0x0204,
            WM_RBUTTONUP = 0x0205,
            WM_XBUTTONDOWN = 0x020B

            // MISSING ADDITIONAL BUTTONS
        }

        [DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
        private static extern IntPtr SetWindowsHookEx(int idHook, LowLevelKeyboardProc lpfn, IntPtr hMod, uint dwThreadId);

        [DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
        [return: MarshalAs(UnmanagedType.Bool)]
        private static extern bool UnhookWindowsHookEx(IntPtr hhk);

        [DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
        private static extern IntPtr CallNextHookEx(IntPtr hhk, int nCode, IntPtr wParam, IntPtr lParam);

        [DllImport("kernel32.dll", CharSet = CharSet.Auto, SetLastError = true)]
        private static extern IntPtr GetModuleHandle(string lpModuleName);

        [DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
        private static extern IntPtr SetWindowsHookEx(int idHook, LowLevelMouseProc lpfn, IntPtr hMod, uint dwThreadId);

        private delegate IntPtr LowLevelMouseProc(int nCode, IntPtr wParam, IntPtr lParam);

        [STAThread]
        static void Main(string[] args)
        {
            Application.EnableVisualStyles();
            Application.SetCompatibleTextRenderingDefault(false);

            _hookIDMouse = SetHookMouse(_procMouse);

            Application.Run(new Form1());

            UnhookWindowsHookEx(_hookIDMouse);
        }

        private static IntPtr SetHookMouse(LowLevelMouseProc proc)
        {
            using (Process curProcess = Process.GetCurrentProcess())
            using (ProcessModule curModule = curProcess.MainModule)
            {
                return SetWindowsHookEx(WH_MOUSE_LL, proc, GetModuleHandle(curModule.ModuleName), 0);
            }
        }

        private static IntPtr HookCallbackMouse(int nCode, IntPtr wParam, IntPtr lParam)
        {
            // DEAL WITH ANY BUTTON OR ADDITIONAL BUTTON MOUSE

            return CallNextHookEx(_hookIDMouse, nCode, wParam, lParam);
        }
    }
}

Спасибо.


person Billard Philippe    schedule 12.03.2018    source источник
comment
Разве большинство игровых мышей не сопоставляют специальные кнопки с сочетаниями клавиш в своих драйверах устройств?   -  person Fildor    schedule 12.03.2018
comment
Возможно, я попробую найти горячие клавиши в режиме отладки, буду держать вас в курсе. Спасибо за советы.   -  person Billard Philippe    schedule 12.03.2018
comment
Зачем нужен режим отладки? Просто зайдите в настройки драйвера устройства ... вы можете проверить свои результаты с помощью отладки, но простой поиск должен быть быстрее.   -  person Fildor    schedule 12.03.2018
comment
Если нет, см. MSDN на О вводе с помощью мыши - XBUTTONS и WM_APPCOMMAND.   -  person Jimi    schedule 13.03.2018


Ответы (1)


Если я правильно понял, вы хотите найти WindowsMessages, которые отправляются после того, как вы нажмете дополнительные кнопки мыши. Чтобы их обнаружить, я предлагаю вам создать пользовательский MessageFilter, реализующий интерфейс IMessageFilter.

class MessageFilter : IMessageFilter
{
    int[] msgToIgnore = new int[] {512, 799, 49632, 49446, 1847, 15, 96, 275, 160, 1848, 674 , 513, 675, 514, 280, 161, 274, 673, 515, 516, 517, 518, 519, 520, 521, 522, 163, 164, 167, 168, 169, 165, 166};

    public bool PreFilterMessage(ref Message m)
    {
        bool match = false;

        foreach (var msg in msgToIgnore)
        {
            if (m.Msg == msg)
                match = true;
        }

        if (!match)
        {
            MessageBox.Show(m.Msg.ToString());
            return true;
        }
        else
            return false;
    }
}

MessageFilter фильтрует все WindowsMessage, отправляемые вашему приложению. Поэтому я в основном добавил массив msgToIgnore, чтобы игнорировать все остальные базовые WindowsMessages, такие как любой обычный щелчок мышью, перемещение ... или все, что происходит при запуске приложения. (Я протестировал его с пустой простой формой и щелкнул, дважды щелкнул, перетащил ... на каждую часть формы) Это игнорирует все эти WindowsMessages, но как только вы отправляете еще один WindowsMessage, например, щелкните дополнительную кнопку мыши, WindowsMessage отобразится в виде MessageBox.

Все, что вам нужно сделать, чтобы активировать это MessageFilter, - это добавить его в свое приложение, я предлагаю сделать это в конструкторе после InitializeComponent().

MessageFilter msgFilter = new MessageFilter();

Application.AddMessageFilter(msgFilter);
person L. Guthardt    schedule 12.03.2018
comment
Скажите, если я ошибаюсь, но мне нужно поймать любой щелчок (даже обычный), и мне нужно поймать его, даже если я не сфокусирован на окне (например, на калькуляторе APM). Значит, в этом случае ваше решение не сработает? Это моя беда, мне следовало сказать об этом точнее. Но спасибо за образец, он может пригодиться позже! - person Billard Philippe; 13.03.2018