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

Bol*_*wyn 924 javascript clipboard coffeescript trello

当您将鼠标悬停在Trello中的卡片上并按Ctrl+时C,此卡片的URL将被复制到剪贴板.他们如何做到这一点?

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

他们如何实现这种魔力?

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

Dan*_*ant 1537

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


我们实际上并不"访问用户的剪贴板",而是通过在按Ctrl+ 时选择有用的东西来帮助用户C.

听起来你已经弄清楚了; 我们利用这样一个事实:当你想要击中Ctrl+时C,你必须首先击中Ctrl键.当Ctrl按下该键时,我们弹出一个textarea,其中包含我们想要在剪贴板上结束的文本,并选择其中的所有文本,因此当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) ->
Run Code Online (Sandbox Code Playgroud)

在DOM中我们得到了

<div id="clipboard-container"><textarea id="clipboard"></textarea></div>
Run Code Online (Sandbox Code Playgroud)

剪贴板内容的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;
}
Run Code Online (Sandbox Code Playgroud)

...并且CSS使它在弹出时无法真正看到textarea ...但是它"可见"足以复制.

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

TrelloClipboard.set(cardUrl)
Run Code Online (Sandbox Code Playgroud)

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

  • 值得注意的是,类似的方法也可以用于捕获粘贴的内容 (28认同)
  • 这听起来像它会是坏的键盘的用户 - 无论何时您尝试复制(或Ctrl +单击另一个窗口,或按Ctrl + F打开搜索,等等),你的焦点移到什么地方无关. (16认同)
  • 如果愿意的话,请随意使用http://js2coffee.org/将原文翻译成js. (8认同)
  • 真棒!但是你如何拥有Mac OS - 你在那里"听"Command键吗? (3认同)
  • +1.这个答案中有很多简洁的东西.我喜欢你实际上共享源代码.但我认为聪明的是用于提供ctrl + c功能的过程的实际解释.在我看来,当按下ctrl时开始准备c时,利用ctrl和c不能在同一时间按下这一事实是非常聪明的.我真的很喜欢这种方法. (2认同)
  • 我使用[js2coffee.org](http://js2coffee.org)将代码翻译成了js,后面提到了这一点(http://stackoverflow.com/questions/17527870/how-does-trello-access-the -users-clipboard#comment26382952_17528590),它对我不起作用.@Daniel,你能为我们提供一个工作的js版本的代码吗? (2认同)
  • @JamesM.Greene 在coffeescript 中,`set: (@value) -&gt;` 变成了`set: function(value) { this.value = value; }` (2认同)

小智 79

实际上,我构建了一个Chrome扩展程序,可以完成此操作,适用于所有网页.源代码在GitHub上.

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

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

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

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

我通过暂时清除零长度选择,保存插入位置,执行复制和恢复插入位置来解决#2问题.

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

  • 所以你实际上和Trello一样.当这些事情趋同时会很甜蜜 (10认同)
  • @Pacerier,我认为托马斯暗示[Convergent Evolution](https://en.wikipedia.org/wiki/Convergent_evolution) - "......不同谱系物种中相似特征的独立演化" (7认同)

Fel*_*lix 20

在雨衣(链接到GitHub)代码的帮助下,我设法得到一个正在运行的版本,使用纯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");
Run Code Online (Sandbox Code Playgroud)

唯一的问题是,此版本仅适用于Chrome.Trello平台支持所有浏览器.我错过了什么?

谢谢VadimIvanov.

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


Tug*_*ain 7

将Daniel LeCheminant的代码从CoffeeScript转换为JavaScript(js2coffee)后,我的代码对我无效.它一直在轰炸_.defer().

我认为这与jQuery延迟有关,因此我将其更改为$.Deferred()现在正在运行.我使用jQuery 2.1.1在Internet Explorer 11,Firefox 35和Chrome 39中进行了测试.用法与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;

})());
Run Code Online (Sandbox Code Playgroud)

  • 我很确定_.defer()是指[Underscrore.JS](http://underscorejs.org/) (4认同)

Bor*_*rić 5

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

有一个readonly输入元素,以编程方式聚焦,工具提示"按CTRL-C复制".当您点击该快捷方式时,输入内容将有效地进入剪贴板.非常好 :)