剪贴板更新事件多次激发

本文关键字:更新 事件 剪贴板 | 更新日期: 2023-09-27 18:27:41

我正在尝试制作一个全局多值剪贴板。我使用了一个堆栈来存储这些值。我正在使用WinProc()来捕获全局复制操作,在该操作中我将值推送到堆栈上。类似地,我使用windows键盘挂钩来捕获Ctrl-V(粘贴)操作。这两个函数的代码如下所示。我已经从中复制并修改了代码。

        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事件。我试过注销listner,我试过使用标志变量"localChange",我试着使用block(),但什么都不起作用。

可以做些什么来解决它。

剪贴板更新事件多次激发

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

您可以使用剪贴板所有权,也可以注入专用剪贴板格式将其"标记"为您的。这是我很久以前写的一篇文章,解释了如何做到这一点,目的是告诉剪贴板查看器不要捕获数据。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格式,然后忽略数据。这也是密码管理器等应用程序避免敏感数据干扰剪贴板管理器的一种方法。

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

由于注册热键将使系统只通知您的应用程序,因此您可以控制使用剪贴板的操作。稍后,您将不得不将Ctrl-V组合发送到目标(预期)应用程序。这将模拟用户为目标应用程序发出的粘贴命令。

这样做的缺点是,您必须知道哪个应用程序在焦点上,以便稍后将密钥组合发送到。