Trello 如何访问用户的剪贴板?

IT技术 javascript coffeescript clipboard trello
2021-01-30 10:35:52

当您将鼠标悬停在Trello 中的卡片上并按Ctrl+ 时C,此卡片的 URL 将复制到剪贴板。他们怎么做到的?

据我所知,不涉及 Flash 电影。我已经安装Flashblock,并且 Firefox 网络选项卡显示没有加载 Flash 电影。(这是通常的方法,例如 ZeroClipboard。)

他们是如何实现这种魔力的?

(此时我想我有一个顿悟:你不能在页面上选择文本,所以我假设他们有一个不可见的元素,他们通过 JavaScript 代码创建一个文本选择,并且Ctrl+C触发浏览器的默认行为,复制那个不可见的元素节点的文本值。)

5个回答

披露: 我编写了 Trello 使用的代码下面的代码是 Trello 用来完成剪贴板技巧的实际源代码。


我们实际上并没有“访问用户的剪贴板”,而是通过在用户按下Ctrl+时选择一些有用的东西来帮助用户C

听起来你已经想通了;我们利用了这样一个事实,即当您想按Ctrl+ 时C,您必须先按一下Ctrl键。Ctrl按键被按下时,我们弹出一个包含我们想要在剪贴板上结束的文本的文本区域,并选择其中的所有文本,因此当C按下时选择全部设置(然后我们在Ctrl键出现时隐藏 textarea 。)

具体来说,Trello 是这样做的:

TrelloClipboard = new class
  constructor: ->
    @value = ""

    $(document).keydown (e) =>
      # Only do this if there's something to be put on the clipboard, and it
      # looks like they're starting a copy shortcut
      if !@value || !(e.ctrlKey || e.metaKey)
        return

      if $(e.target).is("input:visible,textarea:visible")
        return

      # Abort if it looks like they've selected some text (maybe they're trying
      # to copy out a bit of the description or something)
      if window.getSelection?()?.toString()
        return

      if document.selection?.createRange().text
        return

      _.defer =>
        $clipboardContainer = $("#clipboard-container")
        $clipboardContainer.empty().show()
        $("<textarea id='clipboard'></textarea>")
        .val(@value)
        .appendTo($clipboardContainer)
        .focus()
        .select()

    $(document).keyup (e) ->
      if $(e.target).is("#clipboard")
        $("#clipboard-container").empty().hide()

  set: (@value) ->

在 DOM 中,我们有:

<div id="clipboard-container"><textarea id="clipboard"></textarea></div>

剪贴板内容的 CSS:

#clipboard-container {
  position: fixed;
  left: 0px;
  top: 0px;
  width: 0px;
  height: 0px;
  z-index: 100;
  display: none;
  opacity: 0;
}
#clipboard {
  width: 1px;
  height: 1px;
  padding: 0px;
}

... CSS 使您在弹出时实际上看不到 textarea ......但它“可见”足以复制。

当您将鼠标悬停在卡片上时,它会调用

TrelloClipboard.set(cardUrl)

...所以剪贴板助手知道Ctrl按下键时要选择什么

惊人的!但是你是如何拥有 Mac OS 的——你“听”那里的 Command 键吗?
2021-03-17 10:35:52
这听起来对键盘用户不利 - 任何时候您尝试复制(或 ctrl+单击以在另一个窗口中打开,或 Ctrl+F 进行搜索,等等),您的焦点都会移动到不相关的某个地方。
2021-03-18 10:35:52
+1。这个答案中有很多整洁的东西。我喜欢你真的分享了源代码。但我认为聪明的是对用于提供 ctrl+c 功能的过程的实际解释。在我看来,利用 ctrl 和 c 不能同时按下的事实,通过在按下 ctrl 时开始准备 c 是非常明智的。我真的很喜欢这种方法。
2021-03-21 10:35:52
如果愿意,请随意使用js2coffee.org将原文翻译成 js。
2021-03-28 10:35:52
值得注意的是,类似的方法也适用于捕获粘贴的内容
2021-04-02 10:35:52

我实际上构建了一个 Chrome 扩展程序,它完全可以做到这一点,并且适用于所有网页。源代码在 GitHub 上

我发现 Trello 的方法存在三个错误,我知道这是因为我自己也遇到过:)

该副本在这些情况下不起作用:

  1. 如果您已经Ctrl按下然后将鼠标悬停在一个链接上并点击C,则该副本将不起作用。
  2. 如果您的光标位于页面中的某个其他文本字段中,则副本不起作用。
  3. 如果您的光标在地址栏中,则副本不起作用。

我通过总是有一个隐藏的跨度来解决 #1,而不是在用户点击Ctrl/时创建一个Cmd

我通过临时清除零长度选择、保存插入符号位置、进行复制和恢复插入符号位置来解决 #2。

我还没有找到 #3 的修复程序:)(有关信息,请查看我的 GitHub 项目中的未解决问题)。

@Pacerier,我认为托马斯提到了收敛进化——“......不同谱系物种中相似特征的独立进化”
2021-03-23 10:35:52
因此,您实际上与 Trello 的做法相同。当这样的事情收敛时甜蜜
2021-03-29 10:35:52
天啊,你可以打开一个关于这个话题的新聊天
2021-03-29 10:35:52
@ThomasAhle,你是什么意思?
2021-04-10 10:35:52

在 GitHub 上raincoat 代码的帮助下,我设法获得了一个运行版本,可以使用纯 JavaScript 访问剪贴板。

function TrelloClipboard() {
    var me = this;

    var utils = {
        nodeName: function (node, name) {
            return !!(node.nodeName.toLowerCase() === name)
        }
    }
    var textareaId = 'simulate-trello-clipboard',
        containerId = textareaId + '-container',
        container, textarea

    var createTextarea = function () {
        container = document.querySelector('#' + containerId)
        if (!container) {
            container = document.createElement('div')
            container.id = containerId
            container.setAttribute('style', [, 'position: fixed;', 'left: 0px;', 'top: 0px;', 'width: 0px;', 'height: 0px;', 'z-index: 100;', 'opacity: 0;'].join(''))
            document.body.appendChild(container)
        }
        container.style.display = 'block'
        textarea = document.createElement('textarea')
        textarea.setAttribute('style', [, 'width: 1px;', 'height: 1px;', 'padding: 0px;'].join(''))
        textarea.id = textareaId
        container.innerHTML = ''
        container.appendChild(textarea)

        textarea.appendChild(document.createTextNode(me.value))
        textarea.focus()
        textarea.select()
    }

    var keyDownMonitor = function (e) {
        var code = e.keyCode || e.which;
        if (!(e.ctrlKey || e.metaKey)) {
            return
        }
        var target = e.target
        if (utils.nodeName(target, 'textarea') || utils.nodeName(target, 'input')) {
            return
        }
        if (window.getSelection && window.getSelection() && window.getSelection().toString()) {
            return
        }
        if (document.selection && document.selection.createRange().text) {
            return
        }
        setTimeout(createTextarea, 0)
    }

    var keyUpMonitor = function (e) {
        var code = e.keyCode || e.which;
        if (e.target.id !== textareaId || code !== 67) {
            return
        }
        container.style.display = 'none'
    }

    document.addEventListener('keydown', keyDownMonitor)
    document.addEventListener('keyup', keyUpMonitor)
}

TrelloClipboard.prototype.setValue = function (value) {
    this.value = value;
}

var clip = new TrelloClipboard();
clip.setValue("test");

查看一个工作示例:http : //jsfiddle.net/AGEf7/

@don41382 它不能在 Safari 上正常工作(至少是 Mac 版本)。在适当的情况下,我的意思是它确实会复制,但是您必须按 cmd+C 两次。
2021-03-16 10:35:52
@VadimIvanov 真的!有人知道为什么吗?
2021-03-25 10:35:52
@don41382 我不知道确切原因,但我找到了解决方案。你有一个小错误,onKeyDown 第一个语句应该是 if (!(e.ctrlKey || e.metaKey)) { return; 这意味着我们需要为按下 metaKey 时的复制准备 textarea(这就是来自 trello 的人的伎俩)。这是来自 trello.com gist.github.com/fustic/10870311的代码
2021-03-28 10:35:52
@VadimIvanov 谢谢。我会在上面修复它。
2021-03-28 10:35:52
由于el.innerText未定义,它在 FF 33.1 中不起作用,因此我将clipboard()函数的最后一行更改clip.setValue(el.innerText || el.textContent);以提高跨浏览器兼容性。链接:jsfiddle.net/AGEf7/31
2021-03-31 10:35:52

Daniel LeCheminant 的代码在将其从 CoffeeScript 转换为 JavaScript ( js2coffee )后对我不起作用它一直在线上轰炸_.defer()

我认为这与 jQuery 延迟有关,所以我将其更改为$.Deferred()现在可以使用了。我在 Internet Explorer 11、Firefox 35 和 Chrome 39 中使用 jQuery 2.1.1 对其进行了测试。用法与 Daniel 的帖子中描述的相同。

var TrelloClipboard;

TrelloClipboard = new ((function () {
    function _Class() {
        this.value = "";
        $(document).keydown((function (_this) {
            return function (e) {
                var _ref, _ref1;
                if (!_this.value || !(e.ctrlKey || e.metaKey)) {
                    return;
                }
                if ($(e.target).is("input:visible,textarea:visible")) {
                    return;
                }
                if (typeof window.getSelection === "function" ? (_ref = window.getSelection()) != null ? _ref.toString() : void 0 : void 0) {
                    return;
                }
                if ((_ref1 = document.selection) != null ? _ref1.createRange().text : void 0) {
                    return;
                }
                return $.Deferred(function () {
                    var $clipboardContainer;
                    $clipboardContainer = $("#clipboard-container");
                    $clipboardContainer.empty().show();
                    return $("<textarea id='clipboard'></textarea>").val(_this.value).appendTo($clipboardContainer).focus().select();
                });
            };
        })(this));

        $(document).keyup(function (e) {
            if ($(e.target).is("#clipboard")) {
                return $("#clipboard-container").empty().hide();
            }
        });
    }

    _Class.prototype.set = function (value) {
        this.value = value;
    };

    return _Class;

})());

当您缩短 URL 时,可以在http://goo.gl上看到非常相似的内容。

有一个只读输入元素,它以编程方式聚焦,工具提示按下CTRL-C复制。

当您点击该快捷方式时,输入内容会有效地进入剪贴板。非常好 :)