2

我试图通过在 Windows 7 中使用 user32.dll 模拟鼠标移动和点击来自动绘制类似于绘图的程序。

这是我所拥有的以及我如何使用它:

设置

    [DllImport("user32.dll")]
    static extern void mouse_event(int dwFlags, int dx, int dy, int dwData, int dwExtraInfo);


    [Flags]
    public enum MouseEventFlags
    {
        LEFTDOWN = 0x00000002,
        LEFTUP = 0x00000004,
        MIDDLEDOWN = 0x00000020,
        MIDDLEUP = 0x00000040,
        MOVE = 0x00000001,
        ABSOLUTE = 0x00008000,
        RIGHTDOWN = 0x00000008,
        RIGHTUP = 0x00000010
    }

    public void LeftMouseDown()
    {
        mouse_event((int)(MouseEventFlags.LEFTDOWN), Cursor.Position.X, Cursor.Position.Y, 0, 0);
    }

    public void LeftMouseUp()
    {
        mouse_event((int)(MouseEventFlags.LEFTUP), Cursor.Position.X, Cursor.Position.Y, 0, 0);
    }

绘图时

            foreach (var contour in contours)
            {
                LeftMouseDown();
                foreach (var point in contour)
                {
                    var x = point.X + offsetX;
                    var y = point.Y + offsetY;
                    Cursor.Position = new Point(x, y);
                    //LeftMouseDown();
                    System.Threading.Thread.Sleep(2);
                }
                LeftMouseUp();
            }

我试图模拟的是鼠标被点击并按住,移动到每个点的一堆点,contour然后在移动到下一个点之前松开contour

问题在于,这只是在第一次移动时按住鼠标按钮,然后再松开。

TL;博士

当以编程方式移动鼠标时,如何保持鼠标左键单击?

我正在尝试在第三方应用程序中模拟绘图。(Microsoft LINQ 的 Whiteboard,IM 客户端。)

4

2 回答 2

2

总结评论线索:这不是预期的结果,因为代码只模拟鼠标点击,而不是鼠标运动。使用 Cursor.Position 移动光标直接更改光标位置,绕过 Windows 输入事件队列。因此不会向具有焦点的窗口生成任何通知(WM_MOUSEMOVE 消息)。

通过将 MouseMove() 方法添加到使用 mouse_event() 和 MouseEventFlags.MOVE 的帮助程序类来修复

于 2012-04-12T17:57:06.387 回答
1

一个迟到的回应,但我认为你的错误是在打电话mouse_event。基本上,mouse_event报告相对鼠标移动(如果没有MOUSEEVENTF_ABSOLUTE),但您传递绝对位置,从而使其行为错误。此外,在执行鼠标按下之前,您应该将光标移动到按下鼠标按钮的位置。

所以这是我的修复:

// I changed a bit of the method signature, but that doesn't really matter
[DllImport("user32.dll", CallingConvention = CallingConvention.StdCall)]
static extern void mouse_event(MouseEventFlags flags, uint dx, uint dy, uint delta, IntPtr extraInfo);

[Flags]
enum MouseEventFlags : uint
{
    Absolute = 0x8000,
    LeftDown = 0x0002,
    LeftUp = 0x0004,
    MiddleDown = 0x0020,
    MiddleUp = 0x0040,
    Move = 0x0001,
    RightDown = 0x0008,
    RightUp = 0x0010,
    Wheel = 0x0800,
    XDown = 0x0080,
    XUp = 0x0100,
    HWheel = 0x1000,
}

public void LeftMouseDown()
{
    // Simulate left down, notice that RELATIVE movement is 0
    mouse_event(MouseEventFlags.LeftDown, 0, 0, 0, IntPtr.Zero);
}

public void LeftMouseUp()
{
    // Simulate left up, notice that RELATIVE movement is 0 too
    mouse_event(MouseEventFlags.LeftUp, 0, 0, 0, IntPtr.Zero);
}

移动鼠标时...

foreach (var contour in contours)
{
    // simulate mouse down AFTER cursor is moved to the first point (IMPORTANT!)
    var x = contour[0].X + offsetX;
    var y = contour[0].Y + offsetY;
    Cursor.Position = new Point(x, y);
    LeftMouseDown();
    foreach (var point in contour)
    {
        x = point.X + offsetX;
        y = point.Y + offsetY;
        Cursor.Position = new Point(x, y);
        System.Threading.Thread.Sleep(2);
    }
    // cursor is already at the final position
    LeftMouseUp();
}

附带说明一下,通过一些简单的测试,设置Cursor.Position 确实会生成WM_MOUSEMOVE消息。

于 2013-03-29T16:19:13.223 回答