【问题标题】:Clipboard updated event fired multiple times多次触发剪贴板更新事件
【发布时间】:2014-10-23 01:53:57
【问题描述】:

我正在尝试制作一个全局多值剪贴板。我使用堆栈来存储值。我正在使用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() ,但没有任何工作。

可以做些什么来解决它。

【问题讨论】:

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


    【解决方案1】:

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

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

    【讨论】:

      【解决方案2】:

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

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

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

      【讨论】:

        猜你喜欢
        • 2021-08-01
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        相关资源
        最近更新 更多