Trelloはユーザーのクリップボードにどのようにアクセスしますか?


936

Trelloでカードにカーソルを合わせCtrl +C、このカードのURLがクリップボードにコピーされます。彼らはこれをどのように行うのですか?

私の知る限り、関係するFlashムービーはありません。フラッシュブロックを持っていますインストールし、FirefoxのネットワークタブにFlashムービーが読み込まれていません。(これは、ZeroClipboardなどの通常の方法です。)

彼らはどのようにしてこの魔法を達成していますか?

(現時点で私はひらめきを持っていると思います:ページ上のテキストを選択することはできないため、JavaScriptコードを介してテキスト選択を作成する非表示要素があると想定し、Ctrl+ Cはブラウザーのデフォルトの動作をトリガーし、その非表示をコピーしますノードのテキスト値。)


22
ライブDOMを見ると、「clipboard-container」クラスのdivがあります。Ctrlキーを押したままにすると、テキスト領域が表示されます(Ctrlキーを離すと削除されます)。私はあなたの啓示が正しいと思います。カードごとにURLがどこに格納されているのか正確にはわかりません
Ian

@イアン、はい、私は確認できます、それはまさにそれが機能した方法です。掘り下げてくれてありがとう!(URLがどこに保存されているか気になりません。フラッシュボードなしのクリップボードテクノロジーに興味がありました。)
Boldewyn 2013

2
ダニエルのプロフィールを調べたところ、彼はTrello開発者のようです。(Coffeescriptのソースはどこから入手したのだろうと思いました。)それで、彼には不当な利点があります;-)とにかくありがとう!
Boldewyn 2013

1
このテクニックの機知を損なうつもりはありません。かなり賢い方法です。しかし、私は仕方がありませんが、これはせいぜい、十分に公表/文書化されておらず、最悪の場合、かなり不快なユーザーエクスペリエンスだと思います。確かに、それは侵略的に不快ではありません(私が誤ってカードのURLをコピーしたときのことを思い出せないためです)が、Trelloの長いユーザーとして、これが存在することをまったく知りませんでした。
マイケルウェールズ

3
@MichaelWalesこの機能は5日前に追加されました。まだテスト中です。機能している場合は、キーボードショートカットとして記載されています。
Daniel LeCheminant 2013

回答:


1547

開示: Trelloが使用するコードを記述しました。以下のコードは、Trelloがクリップボードトリックを実行するために使用する実際のソースコードです。


実際には「ユーザーのクリップボードにアクセス」するのではなく、ユーザーがCtrl+ を押したときに役立つものを選択することで、ユーザーを少し手助けしますC

あなたはそれを理解したように聞こえます。Ctrl+を押したいC場合は、Ctrl最初にキーを押す必要があるという事実を利用します。ときにCtrlキーが押されたときの選択は、すべてのセットですので、我々は、我々がクリップボードに終わるしたいテキストが含まれているテキストエリアでポップ、そしてその中のすべてのテキストを選択しC、キーを打っています。(その後、Ctrlキーが表示されたときにテキストエリアを非表示にします)

具体的には、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は、キーが押されたときに何を選択するかを認識しています。


3
驚くばかり!しかし、Mac OSをどのように使用していますか?そこで、Commandキーを「聞きますか」?
スマン2013

28
同様の方法が貼り付けられたコンテンツをキャプチャするためにも同様に機能することは注目に値します
Michael Robinson

17
これはキーボードのユーザーにとっては悪いことのように思えます-コピーしようとするたびに(またはCtrl +クリックして別のウィンドウで開くか、Ctrl + Fで検索するなど)、フォーカスは無関係な場所に移動します。
アダムA

2
+1。この答えではたくさんのきちんとしたことが起こっています。私はあなたが実際にソースコードを共有したのが好きです。しかし、私が賢いと思ったのは、ctrl + c機能を提供するために使用されるプロセスの実際の説明でした。私の意見では、ctrlが押されたときにcの準備を開始することによって、ctrlとcを同時に押すことができないという事実を利用することは非常に賢明でした。私はそのアプローチが本当に好きでした。
Travis J

8
必要に応じて、js2coffee.orgを使用してオリジナルをjsに変換してください。
Alexandr Kurilin 2013

79

私は実際これを正確に、すべてのWebページに対して実行するChrome拡張機能を作成しました。ソースコードはGitHubにあります

私はTrelloのアプローチに3つのバグを見つけました。これは私が自分自身に直面したためです。

次のシナリオではコピーは機能しません。

  1. Ctrl押してからリンクにカーソルを合わせてを押すとC、コピーは機能しません。
  2. カーソルがページの他のテキストフィールドにある場合、コピーは機能しません。
  3. カーソルがアドレスバーにある場合、コピーは機能しません。

ユーザーがCtrl/を押したときにスパンを作成するのではなく、常に非表示のスパンを持つことで#1を解決しましたCmd

長さ0の選択を一時的にクリアし、キャレット位置を保存し、コピーを実行してキャレット位置を復元することにより、#2を解決しました。

#3の修正はまだ見つかりません:)(詳細については、GitHubプロジェクトの未解決の問題を確認してください)。


10
つまり、実際にはTrelloと同じ方法でこれを行いました。そのようなものが収束するときの甘い
Thomas Ahle 2013

@ThomasAhle、どういう意味?
Pacerier 2014年

7
@Pacerier、私はトーマスが収束進化に言及したと思います-"...異なる系統の種における同様の機能の独立した進化"
yoniLavi 14

聖なる牛このトピックについての新しいチャットを開くことができます
carkod 2017年

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");

唯一の問題は、このバージョンがChromeでのみ機能することです。Trelloプラットフォームはすべてのブラウザーをサポートします。何が欠けていますか?

VadimIvanovのおかげでSovled。

実際の例を参照してください:http : //jsfiddle.net/AGEf7/


@ don41382 Safari(少なくともMacバージョン)では正しく機能しません。適切な状態では、コピーは行われますが、cmd + Cを2回押す必要があります。
Vadim Ivanov

@VadimIvanov True!なぜ誰かが知っていますか?
Felix 14

1
@ don41382理由は正確にはわかりませんが、解決策を見つけました。マイナーなバグがあります。onKeyDownの最初のステートメントはif(!(e.ctrlKey || e.metaKey)){return;でなければなりません。これは、押されたmetaKeyのコピー用にtextareaを準備する必要があることを意味します(これは、Trelloの人がトリックを作った方法です)。これはtrello.com gist.github.com/fustic/10870311の
Vadim Ivanov

@VadimIvanovありがとう。上で修正します。
Felix

1
el.innerText未定義だったため、FF 33.1では機能しませんでした。そのため、ブラウザーの互換性を高めるために、clipboard()関数の最後の行をに変更しましたclip.setValue(el.innerText || el.textContent);。リンク:jsfiddle.net/AGEf7/31
RevanProdigalKnight

7

Daniel LeCheminantのコードは、CoffeeScriptからJavaScript(js2coffee)に変換した後、機能しませんでした。それは_.defer()線上に爆撃を続けた。

これはjQueryの据え置きと関係があると想定していたので、これをに変更し$.Deferred()、現在は機能しています。Internet Explorer 11、Firefox 35、およびjQuery 2.1.1を搭載したChrome 39でテストしました。使い方はダニエルの投稿で説明されているものと同じです。

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;

})());

5

URLを短くすると、http: //goo.glでよく似たものが見られます。

ツールチップを押しCTRL-Cてコピーすることにより、プログラムでフォーカスされる読み取り専用の入力要素があります。

そのショートカットを押すと、入力コンテンツがクリップボードに効果的に取り込まれます。すごくいい :)

弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.