剪贴板更新事件多次触发

时间:2014-10-23 01:53:57

标签: c# clipboard copy-paste keyboard-hook setwindowshookex

我正在尝试创建一个全局多值剪贴板。我用堆栈来存储值。我正在使用WinProc()捕获全局复制操作,我在其中推送堆栈上的值。类似地,我使用Windows键盘钩来捕获Ctrl-V(粘贴)操作。这两个函数的代码如下。我复制并修改了this中的代码。

        private int KbHookProc(int nCode, IntPtr wParam, IntPtr lParam)
    {
        if (nCode >= 0)
        {
            var hookStruct = (KbLLHookStruct)Marshal.PtrToStructure(lParam, typeof(KbLLHookStruct));

            // Quick and dirty check. You may need to check if this is correct. See GetKeyState for more info.
            bool ctrlDown = GetKeyState(VK_LCONTROL) != 0 || GetKeyState(VK_RCONTROL) != 0;

            if (ctrlDown && hookStruct.vkCode == 0x56) // Ctrl+V
            {
                if (clipBoardStack.Count > 0)
                {
                    lock (this)
                    {
                        localChange = true;
                        RemoveClipboardFormatListener(this.Handle);     // Remove our window from the clipboard's format listener list.
                        System.Threading.Thread.Sleep(200);
                        Clipboard.SetText(clipBoardStack.Pop());
                        AddClipboardFormatListener(this.Handle);
                        System.Threading.Thread.Sleep(200);
                    }

                }

            }
        }

        // Pass to other keyboard handlers. Makes the Ctrl+V pass through.
        return CallNextHookEx(_hookHandle, nCode, wParam, lParam);
    }

我的WinProc覆盖如下。我也从SO复制了它,但不记得链接。

        protected override void WndProc(ref Message m)
    {
        base.WndProc(ref m);

        if (m.Msg == WM_CLIPBOARDUPDATE)
        {
            if (!localChange)//Only store the data in stack when it comes from outside. Just to prevent the side effect of Paste Operation
            {
                IDataObject iData = Clipboard.GetDataObject();      // Clipboard's data.

                if (iData.GetDataPresent(DataFormats.Text))
                {
                    lock (this)
                    {
                        string text = (string)iData.GetData(DataFormats.Text);
                        clipBoardStack.Push(text);                            
                    }
                }
            }
            else
            {
                localChange = false;
            }
        }

复制操作运行良好。它填充堆栈,但是当我使用粘贴操作时,它会触发WM_CLIPBOARDUPDATE事件。这使得堆栈再次使用最新值填充。

我认为当我在粘贴拦截中更改剪贴板值时,它会触发WM_CLIPBOARDUPDATE事件。我试图取消注册列表器,我试图使用标志变量'localChange',我试图使用block()但没有什么工作。

可以做些什么来解决它。

2 个答案:

答案 0 :(得分:3)

您仍然可以侦听剪贴板更新,但在创建剪贴板更新时需要忽略它们。即不要对自己的回声做出反应。

您可以使用剪贴板所有权,或者您可以注入私有剪贴板格式以“标记”它作为您的。这篇文章(由我很久以前)解释了如何做到这一点,其目的是让剪贴板查看器不捕获数据。 http://www.clipboardextender.com/developing-clipboard-aware-programs-for-windows/ignoring-clipboard-updates-with-the-cf_clipboard_viewer_ignore-clipboard-format

该文章的基本思想是创建一个名为CF_CLIPBOARD_VIEWER_IGNORE的私有剪贴板格式,并将其与放置实际数据的同时(以相同的打开/更新/关闭顺序)添加到剪贴板。诸如网络浏览器,文字处理器,记事本等程序将无关紧要。但是剪贴板查看器(例如您自己的剪贴板或我的ClipMate)会在剪贴板上看到CF_CLIPBOARD_VIEWER_IGNORE格式,然后忽略该数据。这也是密码管理器等应用程序避免使用敏感数据混乱剪贴板管理器的一种方法。

答案 1 :(得分:1)

您可以在系统中注册HotKey(Ctrl-V),以便系统可以将控制权发送到应用程序中的HotKey处理程序。在那里你可以更新剪贴板。

由于注册HotKey会使系统仅通知您的应用程序,因此您可以控制使用ClipBoard执行的操作。稍后,您必须将Ctrl-V组合发送到目标(预期)应用程序。这将模拟用户为目标应用程序发出粘贴命令。

这样做的缺点是你必须知道哪个应用程序是关注的,以便稍后发送密钥组合。