【问题标题】:How to track mouse X/Y position and print it to a label? [duplicate]如何跟踪鼠标 X/Y 位置并将其打印到标签上? [复制]
【发布时间】:2012-01-15 16:44:12
【问题描述】:

我知道如何从 MSDN 上的示例中修改光标位置。

我的问题是如何在鼠标移动时检查鼠标的位置,然后打印 X 和 Y 位置来表示标签?

编辑:假设我想在整个屏幕上跟踪我的鼠标位置。

编辑 2:我的应用程序将在后台/最小化。

我已经在使用鼠标钩了:

namespace Program
{
    public partial class MouseHook
    {
        [DllImport("user32.dll", CharSet = CharSet.Auto, CallingConvention = CallingConvention.StdCall)]
        private static extern void mouse_event(int dwFlags, int dx, int dy, int dwData, IntPtr dwExtraInfo);

        private const int MOUSEEVENTF_LEFTDOWN = 0x02;
        private const int MOUSEEVENTF_LEFTUP = 0x04;
        private const int MOUSEEVENTF_RIGHTDOWN = 0x08;
        private const int MOUSEEVENTF_RIGHTUP = 0x10;

        public void DoMouseClick()
        {
            int X = Cursor.Position.X;
            int Y = Cursor.Position.Y;

            IntPtr newP = new IntPtr(Convert.ToInt64("0", 16));
            mouse_event(MOUSEEVENTF_LEFTDOWN | MOUSEEVENTF_LEFTUP, X, Y, 0, newP);
        }
    }
}

【问题讨论】:

  • 这个问题已经被问过很多次了。在搜索框中键入“+setwindowshookex +wh_mouse_ll”。或者只使用 50 毫秒计时器和 Cursor.Position
  • @Hans Passant,计时器解决方案似乎是检查 x/y 位置的最简单方法。我现在只需要弄清楚如何检查位置。为此 +1 :)
  • @Cody Gray,如果问题是在注明答案前 3 年发布的,这个问题怎么会重复?

标签: c# winforms mousemove low-level-api


【解决方案1】:

Here 是有关系统级鼠标事件处理调用(低级鼠标进程)的 msdn 文档。

Here 是使用低级鼠标 procs 更改滚动事件的示例。

在第二个链接的答案中,使用来自here 的 WM_MOUSEMOVE 而不是 WM_MOUSEWHEEL。

需要注意的一点:当鼠标悬停在具有提升权限的程序上时,该程序要继续捕获鼠标事件,必须以提升的权限启动该程序。

代码(未测试):

using System;

using System.Diagnostics;
using System.Windows.Forms;
using System.Runtime.InteropServices;

namespace CatchMouseMove
{
    class InterceptMouse
    {
        const int INPUT_MOUSE = 0;
        const int MOUSEEVENTF_WHEEL = 0x0800;
        const int WH_MOUSE_LL = 14; 


        private static LowLevelMouseProc _proc = HookCallback;
        private static IntPtr _hookID = IntPtr.Zero;

        public static void Main()
        {
            _hookID = SetHook(_proc);

            if (_hookID == null)
            {
                MessageBox.Show("SetWindowsHookEx Failed");
                return;
            }
            Application.Run();
            UnhookWindowsHookEx(_hookID);
        }

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

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

        private static IntPtr HookCallback(int nCode, IntPtr wParam, IntPtr lParam)
        {
            int xPos = 0;
            int yPos = 0;

            if (nCode >= 0 && MouseMessages.WM_MOUSEMOVE == (MouseMessages)wParam)
            {    
                xPos = GET_X_LPARAM(lParam); 
                yPos = GET_Y_LPARAM(lParam);

                //do stuff with xPos and yPos
            }
            return CallNextHookEx(_hookID, nCode, wParam, lParam);
        }


        private enum MouseMessages
        {
            WM_MOUSEMOVE = 0x0200
        }

        [StructLayout(LayoutKind.Sequential)]
        private struct POINT
        {
            public int x;
            public int y;
        }

        [StructLayout(LayoutKind.Sequential)]
        private struct MSLLHOOKSTRUCT
        {
            public POINT pt;
            public int mouseData;
            public int flags;
            public int time;
            public IntPtr dwExtraInfo;
        }

        public struct INPUT
        {
            public int type;
            public MOUSEINPUT mi;
        }

        [StructLayout(LayoutKind.Sequential)]
        public struct MOUSEINPUT
        {
            public int dx;
            public int dy;
            public int mouseData;
            public uint dwFlags;
            public int time;
            public int dwExtraInfo;
        }



        [DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
        private static extern IntPtr SetWindowsHookEx(int idHook,
            LowLevelMouseProc 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);

    }

}

【讨论】:

    【解决方案2】:

    您需要设置 Windows 挂钩才能实现此目的。 MSDN 文章How to set a Windows hook in Visual C# .NET 展示了如何设置鼠标挂钩。

    我试过了,它会在整个表单中捕获鼠标,即使鼠标光标在控件上也是如此。

    【讨论】:

    • 这个例子很好,除了它是为一个独立的线程设置的,它可以找到你的表单而不是直接绑定到表单。
    • 可以,但是很容易适应。
    【解决方案3】:

    您可以使用MouseMove event 的EventArgs,因为它保存了鼠标坐标。从那里您可以轻松地将标签的文本属性设置为您将从e(MouseMove EventArgs)获取的 X 或 Y 坐标。

    private void Form_MouseMove(object sender, MouseEventArgs e)
    {
        // Update the mouse coordinates displayed in the textbox.
        myTextBox.Text = e.Location.ToString();
    }
    

    【讨论】:

    • 好的,我明白了。但是因为我必须有一个对象来执行处理程序,例如:textBox1.TextChanged += new EventArgs(textBox1_TextChanged); 我会调用哪个对象?
    • @Help: TextChanged 是错误的事件。您想处理MouseMove 事件。在该事件处理方法中,更新您的文本框:textBox1.Text = e.Location.ToString();
    • @Cody Gray,我知道。但我的问题是我要引用什么对象,这意味着我将 textBox1 替换为什么?
    • @Help:哦,可能是你的表单。您会遇到的问题是,只要光标移到窗体顶部的子控件上,它就不会在窗体上引发事件。
    • 您真的需要跟踪整个窗口中的鼠标移动,还是只需要一些控件(如窗格等)?
    猜你喜欢
    • 2014-12-13
    • 2011-12-09
    • 2013-12-08
    • 1970-01-01
    • 2011-09-07
    • 2018-06-11
    • 2021-09-09
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多