8

目標: バックグラウンドで実行され、キーの組み合わせ Win-V をリッスンし、それが発生すると、クリップボードの内容を現在アクティブなウィンドウ (任意のアプリ) に貼り付ける C# アプリを作成します。基本的に私は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");
        }
    }
}
4

2 に答える 2

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;
}
于 2012-08-24T12:43:28.270 に答える