从后台C#app发送Win API粘贴cmd

时间:2012-08-24 01:04:38

标签: c# winforms winapi winforms-interop

目标:编写一个在后台运行的C#应用​​程序,侦听组合键Win-V,当发生这种情况时,将剪贴板内容粘贴到当前活动窗口(某些任意应用程序) 。基本上我正在尝试模仿PureText,但我并不打算先将文本转换为纯文本。

问题:粘贴到当前活动的窗口无效。

详细信息:要在后台收听按键操作,我正在使用A Simple C# Global Low Level Keyboard Hook中的 globalKeyboardHook 类。我能够捕获Win-V事件,但我无法正确发送粘贴命令。我可以使用 SendKeys.Send keybd_event 功能发送粘贴。然而,他们向管道发送了另一个“V”按下,该管道被 gkh_KeyDown 事件捕获并导致多个粘贴事件被触发。

我希望我需要使用 SendMessage PostMessage ,但到目前为止,我所做的所有尝试都失败了。下面是最后一个函数 SendCtrlV 的完整代码,是感兴趣的代码。这些评论解释了我迄今为止所尝试的一切。你能看到我错过的东西吗?

using System;
using System.IO;
using System.Runtime.InteropServices;
using System.Windows.Forms;
using Utilities;

namespace KeyHookTest
{
    public partial class Form1 : Form
    {
        private bool LWin_down;
        private bool V_down;
        globalKeyboardHook gkh = new globalKeyboardHook();

        [DllImport("user32.dll", CharSet = CharSet.Auto)]
        static public extern IntPtr GetForegroundWindow();

        [DllImport("user32.dll")]
        static extern void keybd_event(byte bVk, byte bScan, uint dwFlags, uint dwExtraInfo);

        [DllImport("user32.dll")]
        private static extern int SendMessage(IntPtr hwnd, int msg, int wParam, int lParam);

        [DllImport("user32.dll")]
        public static extern IntPtr PostMessage(IntPtr hWnd, uint Msg, IntPtr wParam, IntPtr lParam);  

        public Form1()
        {
            InitializeComponent();
        }

        private void Form1_Load(object sender, EventArgs e)
        {
            gkh.HookedKeys.Add(Keys.V);
            gkh.HookedKeys.Add(Keys.LWin);
            gkh.KeyDown += new KeyEventHandler(gkh_KeyDown);
            gkh.KeyUp += new KeyEventHandler(gkh_KeyUp);
        }

        void gkh_KeyUp(object sender, KeyEventArgs e)
        {
            if (e.KeyCode == Keys.LWin)
                LWin_down = false;
            else
                V_down = false;
        }

        void gkh_KeyDown(object sender, KeyEventArgs e)
        {
            if (e.KeyCode == Keys.LWin)
                LWin_down = true;
            else
                V_down = true;

            if (LWin_down && V_down)
            {
                LogDebug("Enter Win+V");

                try
                {
                    SendCtrlV();
                }
                catch { }
            }

        }

        private void SendCtrlV()
        {
            uint KEYEVENTF_KEYUP = 2;
            int KEYDOWN = 0x0100;
            int KEYUP = 0x0101;
            byte KEY_LCONTROL1 = 0x11;
            IntPtr KEY_LCONTROL2 = new IntPtr(0x11);
            byte KEY_V1 = 0x56;
            IntPtr KEY_V2 = new IntPtr(0x56);
            int WM_PASTE1 = 0x302;
            uint WM_PASTE2 = 0x302;

            IntPtr hWnd = GetForegroundWindow();

            // Works, but causes multiple gkh_KeyDown to fire so it's slow and buggy
            /*keybd_event(KEY_LCONTROL1, 0, 0, 0);
            keybd_event(KEY_V1, 0, 0, 0);
            keybd_event(KEY_V1, 0, KEYEVENTF_KEYUP, 0);
            keybd_event(KEY_LCONTROL1, 0, KEYEVENTF_KEYUP, 0);*/

            // Works, but causes multiple gkh_KeyDown to fire so it's slow and buggy
            //SendKeys.Send("^v");

            // Doesn't work, causes UAC prompt
            //SendKeys.Send("{^}v");

            // Doesn't work, nothing gets pasted to the foregroundwindow
            //SendMessage(hWnd, WM_PASTE1, 0, 0);

            // Doesn't work, nothing gets pasted to the foregroundwindow
            //PostMessage(hWnd, WM_PASTE2, IntPtr.Zero, IntPtr.Zero);

            // Doesn't work, nothing gets pasted to the foregroundwindow
            /*SendMessage(hWnd, KEYDOWN, KEY_LCONTROL1, 0);
            SendMessage(hWnd, KEYDOWN, KEY_V1, 0);
            SendMessage(hWnd, KEYUP, KEY_V1, 0);
            SendMessage(hWnd, KEYUP, KEY_LCONTROL1, 0);*/

            // Doesn't work, nothing gets pasted to the foregroundwindow
            /*PostMessage(hWnd, 0x0100, KEY_LCONTROL2, IntPtr.Zero);
            PostMessage(hWnd, 0x0100, KEY_V2, IntPtr.Zero);
            PostMessage(hWnd, 0x0101, KEY_V2, IntPtr.Zero);
            PostMessage(hWnd, 0x0101, KEY_LCONTROL2, IntPtr.Zero);*/
        }

        private void LogDebug(string msg)
        {
            string logpath = Environment.GetEnvironmentVariable("USERPROFILE") + @"\Desktop\KeyHookTest.txt";
            File.AppendAllText(logpath, DateTime.Now.ToString("HH:mm:ss:fff") + ": " + msg + "\r\n");
        }
    }
}

2 个答案:

答案 0 :(得分:2)

这些额外的链接帮助我找到答案:

这对我有用:

private void SendCtrlV()
{
    IntPtr hWnd = GetFocusedHandle();
    PostMessage(hWnd, WM_PASTE, IntPtr.Zero, IntPtr.Zero);
}

static IntPtr GetFocusedHandle()
{
    var info = new GuiThreadInfo();
    info.cbSize = Marshal.SizeOf(info);
    if (!GetGUIThreadInfo(0, ref info))
        throw new Win32Exception();
    return info.hwndFocus;
}

答案 1 :(得分:0)

它可以工作,但是如果您希望它有效,则必须使用TextBox的本机窗口句柄