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);
        }
    }
}

Благодарю.

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);
Другие вопросы по тегам