Jak Trello uzyskuje dostęp do schowka użytkownika?

Gdy najedziesz na kartę w Trello i naciśniesz Ctrl+C , adres URL tej karty jest kopiowany do schowka. Jak oni to robią?

Z tego, co wiem, nie ma żadnego filmu Flash. Mam zainstalowany Flashblock , a zakładka Firefox network nie pokazuje załadowanego filmu Flash. (Jest to zwykła metoda, na przykład przez ZeroClipboard.)

Jak osiągnąć tę magię?

(w tej chwili chyba miałem epiphany: nie można zaznaczać tekstu na stronie, więc zakładam, że mają niewidoczny element, w którym tworzą zaznaczenie tekstu za pomocą kodu JavaScript i Ctrl+C uruchamia domyślne zachowanie przeglądarki, kopiując wartość tekstową niewidocznego węzła.)

Author: Michael, 2013-07-08

5 answers

Ujawnienie: napisałem kod, którego Trello używa ; poniższy kod jest faktycznym kodem źródłowym, którego Trello używa do wykonania sztuczki ze schowka.


W rzeczywistości nie mamy "dostępu do schowka użytkownika" , zamiast tego pomagamy użytkownikowi trochę wybierając coś przydatnego, gdy naciśnie Ctrl+C .

Wygląda na to, że już to rozgryzłeś; wykorzystujemy fakt, że gdy chcesz wcisnąć Ctrl+C , musisz uderzyć w Ctrl klawisz pierwszy. Po naciśnięciu klawisza Ctrl, wskakujemy do obszaru tekstowego zawierającego tekst, który chcemy znaleźć w schowku i zaznaczamy cały tekst w nim, więc zaznaczenie jest ustawione po naciśnięciu klawisza C. (Następnie ukrywamy textarea, gdy pojawi się klawisz Ctrl.)

W szczególności Trello robi to:]}
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) ->

W DOM mamy:

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

CSS do schowka:

#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;
}

... a CSS sprawia, że nie można faktycznie zobaczyć textarea, gdy wyskakuje ... ale jest" widoczny " na tyle, że można go skopiować.

Kiedy najedziesz na kartę, zadzwoni

TrelloClipboard.set(cardUrl)

... wtedy Pomocnik schowka wie, co wybrać po naciśnięciu klawisza Ctrl.

 1552
Author: Daniel LeCheminant,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2021-01-20 08:06:14

Właściwie zbudowałem Rozszerzenie Chrome , które robi dokładnie to, i dla wszystkich stron internetowych. Kod źródłowy jest na GitHub.

Przy podejściu Trello znalazłem trzy błędy, które znam, bo sam się z nimi zmierzyłem:)

Kopia nie działa w tych scenariuszach:

  1. Jeśli masz już Ctrl wciśnięty, a następnie najedź kursorem na link i naciśnij C , Kopia nie działa.
  2. jeśli kursor znajduje się w innym polu tekstowym na stronie, Kopia nie działa.
  3. jeśli kursor znajduje się na pasku adresu, Kopia nie działa.

Rozwiązałem #1 zawsze mając ukrytą rozpiętość, zamiast tworzyć ją, gdy użytkownik naciśnie Ctrl/Cmd .

Rozwiązałem #2, tymczasowo wyczyszczając zaznaczenie o zerowej długości, zapisując pozycję karetki, wykonując kopię i przywracając pozycję karetki.

Nie znalazłem jeszcze poprawki dla #3:) (Aby uzyskać informacje, Sprawdź otwarty problem w moim projekcie GitHub).

 81
Author: Dhruv Vemula,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2013-08-10 07:40:39

Z pomocą kodu raincoat na Githubie , udało mi się uzyskać dostęp do schowka z uruchomioną wersją za pomocą zwykłego 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");

Zobacz działający przykład: http://jsfiddle.net/AGEf7/

 21
Author: Felix,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2021-01-20 08:15:16

Kod Daniela Lecheminanta nie zadziałał u mnie po przekonwertowaniu go z CoffeeScript do JavaScript (js2coffee). / Align= "left" / linear

Założyłem, że to ma coś wspólnego z jQuery deferreds, więc zmieniłem na $.Deferred() i teraz działa. Testowałem go w Internet Explorer 11, Firefox 35 i Chrome 39 z jQuery 2.1.1. Użycie jest takie samo jak opisane w poście Daniela.

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;

})());
 7
Author: TugboatCaptain,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2021-01-20 08:16:53

Coś bardzo podobnego można zobaczyć na http://goo.gl Kiedy skracasz adres URL.

Jest Tylko do odczytu element wejściowy, który jest programowo skupiony, z tooltip naciśnij CTRL-C, aby skopiować.

Po naciśnięciu tego skrótu, zawartość wejściowa skutecznie dostaje się do schowka. Naprawdę ładne :)

 5
Author: Boris Brdarić,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2020-04-19 17:07:20