其他分享
首页 > 其他分享> > 剪贴板更新事件被多次触发

剪贴板更新事件被多次触发

作者:互联网

我正在尝试制作一个全球多值剪贴板.我已经使用堆栈来存储值.我正在使用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(),但什么也没有正在工作.

该怎么办才能解决.

解决方法:

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

您可能可以使用剪贴板所有权,也可以注入专用剪贴板格式以将其“标记”为您自己的格式.这是一篇关于我该如何做的文章(很久以前由我提供),该文章旨在告诉剪贴板查看器不要捕获数据.
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的私有剪贴板格式,并在放置真实数据的同时(以相同的打开/更新/关闭顺序)将其添加到剪贴板.诸如Web浏览器,文字处理器,记事本等程序将不在乎.但是剪贴板查看器(例如您自己的剪贴板查看器,或我的ClipMate)将看到剪贴板上显示的CF_CLIPBOARD_VIEWER_IGNORE格式,然后忽略数据.这也是密码管理器等应用程序避免使用敏感数据使剪贴板管理器混乱的方式.

标签:keyboard-hook,clipboard,copy-paste,setwindowshookex,c
来源: https://codeday.me/bug/20191029/1956860.html