多次触发剪贴板更新事件

Res*_*ict 3 c# clipboard copy-paste keyboard-hook setwindowshookex

我正在尝试制作一个全局多值剪贴板。我使用了一个堆栈来存储这些值。我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);
    }
Run Code Online (Sandbox Code Playgroud)

我的 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;
            }
        }
Run Code Online (Sandbox Code Playgroud)

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

我想当我在粘贴拦截中更改剪贴板值时,它会触发 WM_CLIPBOARDUPDATE 事件。我尝试取消注册监听器,我尝试使用标志变量“localChange”,我尝试使用 block(),但没有正在工作。

可以做些什么来解决它。

Chr*_*ton 5

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

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