Trello เข้าถึงคลิปบอร์ดของผู้ใช้ได้อย่างไร


936

เมื่อคุณโฮเวอร์เหนือการ์ดในTrelloและกดCtrl+ CURL ของการ์ดนี้จะถูกคัดลอกไปยังคลิปบอร์ด พวกเขาทำเช่นนี้ได้อย่างไร

เท่าที่ฉันสามารถบอกได้ว่าไม่มีภาพยนตร์ Flash เกี่ยวข้อง ฉันติดตั้งFlashblockแล้วและแท็บเครือข่าย Firefox ไม่แสดงภาพยนตร์ Flash ที่โหลด (นั่นเป็นวิธีปกติเช่นโดย ZeroClipboard)

พวกเขาบรรลุความมหัศจรรย์นี้ได้อย่างไร?

(ในขณะนี้ฉันคิดว่าฉันมี epiphany: คุณไม่สามารถเลือกข้อความบนหน้าดังนั้นฉันคิดว่าพวกเขามีองค์ประกอบที่มองไม่เห็นที่พวกเขาสร้างการเลือกข้อความผ่านรหัส JavaScript และCtrl+ Cเรียกพฤติกรรมเริ่มต้นของเบราว์เซอร์คัดลอกที่มองไม่เห็น ค่าข้อความของโหนด)


22
ถ้าคุณดู DOM แบบสดมันจะมี div ที่มี "คลิปบอร์ด - คอนเทนเนอร์" ในคลาส เมื่อคุณกดปุ่ม ctrl คีย์จะถูกเติมด้วย textarea (และจะถูกลบออกเมื่อคุณยกปุ่ม ctrl) ฉันจะถือว่าความศักดิ์สิทธิ์ของคุณถูกต้อง ฉันไม่แน่ใจว่าพวกเขากำลังเก็บ URL ต่อบัตรไว้ตรงไหน
Ian

@ ฉันใช่ฉันสามารถยืนยันได้ว่าเป็นวิธีการทำงาน ขอบคุณที่ขุดมันขึ้นมา! (ฉันไม่สนใจว่าจะเก็บ URL ไว้ที่ไหนฉันสนใจเทคโนโลยีคลิปบอร์ดที่ไม่มีแฟลช)
Boldewyn

2
ฉันค้นหาโปรไฟล์ของ Daniel และดูเหมือนว่าเขาเป็นผู้พัฒนา Trello (ฉันสงสัยว่าที่ที่เขาได้รับแหล่งกาแฟจาก.) ดังนั้นเขาจึงมีข้อได้เปรียบที่ไม่ยุติธรรม ;-) ขอบคุณต่อไป!
Boldewyn

1
ฉันไม่ได้ตั้งใจจะเบี่ยงเบนความสนใจของเทคนิคนี้มันค่อนข้างฉลาด แต่ฉันไม่สามารถช่วยได้ แต่คิดว่านี่เป็นวิธีที่ดีที่สุดเผยแพร่ / จัดทำเอกสารไม่ดีและที่แย่ที่สุดคือประสบการณ์การใช้งานที่สั่นสะเทือน ได้รับมันไม่ได้สั่นสะเทือนอย่างถาวร (เพราะฉันจำไม่ได้ว่ามีครั้งไหนที่ฉันคัดลอก URL การ์ดโดยไม่ตั้งใจ) แต่ในฐานะผู้ใช้ Trello มาเป็นเวลานานฉันก็ไม่รู้ว่ามันมีอยู่จริง
Michael Wales

3
@MichaelWales สถานที่นี้ถูกเพิ่ม 5 วันที่ผ่านมา; เรายังคงทดสอบมันอยู่และหากดูเหมือนว่าใช้งานได้จะมีการบันทึกเป็นแป้นพิมพ์ลัด
Daniel LeCheminant

คำตอบ:


1547

การเปิดเผยข้อมูล: ผมเขียนโค้ดที่ใช้ Trello ; โค้ดด้านล่างนี้เป็นรหัสแหล่งที่มาจริงของ Trello ที่ใช้เพื่อให้บรรลุเคล็ดลับคลิปบอร์ด


เราไม่ได้จริง "การเข้าถึงคลิปบอร์ดของผู้ใช้" แทนเราช่วยให้ผู้ใช้ออกบิตโดยการเลือกสิ่งที่มีประโยชน์เมื่อพวกเขากด+CtrlC

ดูเหมือนคุณจะคิดออก เราใช้ประโยชน์จากข้อเท็จจริงที่ว่าเมื่อคุณต้องการกดCtrl+ Cคุณต้องกดCtrlปุ่มแรก เมื่อCtrlกดปุ่มเราปรากฏใน textarea ที่มีข้อความที่เราต้องการที่จะจบลงบนคลิปบอร์ดและเลือกข้อความทั้งหมดในนั้นเพื่อเลือกจะถูกตั้งค่าเมื่อCกดปุ่ม (จากนั้นเราซ่อน textarea เมื่อ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 ได้อย่างไร - คุณ "ฟัง" ปุ่มคำสั่งที่นั่นหรือไม่
Suman

28
เป็นที่น่าสังเกตว่าวิธีการที่คล้ายกันนั้นใช้งานได้ดีสำหรับการจับเนื้อหาที่วางไว้
Michael Robinson

17
ดูเหมือนว่ามันจะไม่ดีสำหรับผู้ใช้แป้นพิมพ์ - เมื่อใดก็ตามที่คุณพยายามคัดลอก (หรือ ctrl + คลิกเพื่อเปิดในหน้าต่างอื่นหรือ Ctrl + F เพื่อค้นหาและอื่น ๆ ) โฟกัสของคุณจะถูกย้ายไปที่อื่นที่ไม่เกี่ยวข้อง
อดัม A

2
+1 คำตอบนี้มีสิ่งที่เรียบร้อยมากมาย ฉันชอบที่คุณแบ่งปันรหัสต้นฉบับจริง แต่สิ่งที่ฉันคิดว่าฉลาดคือคำอธิบายที่แท้จริงของกระบวนการที่ใช้เพื่อให้ฟังก์ชัน ctrl + c ในความคิดของฉันมันฉลาดมากที่จะใช้ประโยชน์จากความจริงที่ว่า ctrl และ c ไม่สามารถกดในเวลาเดียวกันโดยเริ่มต้นเพื่อเตรียมพร้อมสำหรับ c เมื่อกด ctrl ฉันชอบวิธีการนี้มาก
เทรวิส J

8
อย่าลังเลที่จะใช้js2coffee.orgเพื่อแปลต้นฉบับเป็น js หากต้องการ
Alexandr Kurilin

79

จริง ๆ แล้วฉันสร้างส่วนขยายของ Chromeที่ทำสิ่งนี้และสำหรับหน้าเว็บทั้งหมด รหัสที่มาเป็นบน GitHub

ฉันพบข้อบกพร่องสามข้อด้วยวิธีของ Trello ซึ่งฉันรู้เพราะฉันเผชิญกับตัวเอง :)

สำเนาไม่ทำงานในสถานการณ์เหล่านี้:

  1. หากคุณCtrlกดแล้วจึงวางลิงก์และ Hit Cการคัดลอกจะไม่ทำงาน
  2. หากเคอร์เซอร์ของคุณอยู่ในช่องข้อความอื่น ๆ ในหน้าสำเนาจะไม่ทำงาน
  3. หากเคอร์เซอร์ของคุณอยู่ในแถบที่อยู่สำเนาจะไม่ทำงาน

ฉันจะแก้ไข # 1 โดยมักจะมีช่วงที่ซ่อนอยู่มากกว่าการสร้างหนึ่งเมื่อความนิยมของผู้ใช้/CtrlCmd

ฉันแก้ไข # 2 โดยการล้างการเลือกความยาวเป็นศูนย์ชั่วคราวบันทึกตำแหน่งลูกศรทำการคัดลอกและเรียกคืนตำแหน่งคาเร็ต

ฉันยังไม่พบวิธีแก้ไข # 3 :) (สำหรับข้อมูลให้ตรวจสอบปัญหาเปิดในโครงการ GitHub ของฉัน)


10
ดังนั้นคุณทำแบบนี้เหมือนกับ Trello น่ารักเมื่อสิ่งต่าง ๆ มาบรรจบกัน
โทมัส Ahle

@ThomasAhle คุณหมายถึงอะไร?
Pacerier

7
@ เพียร์ฉันคิดว่าโธมัสพูดถึงConvergent Evolution - "... วิวัฒนาการอิสระของคุณสมบัติที่คล้ายคลึงกันในสายพันธุ์ต่าง ๆ "
yoniLavi

วัวศักดิ์สิทธิ์คุณสามารถเปิดการแชทใหม่เกี่ยวกับหัวข้อนี้
carkod

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

ดูตัวอย่างการทำงาน: http://jsfiddle.net/AGEf7/


@ don41382 มันใช้ไม่ได้กับ Safari (อย่างน้อยเวอร์ชั่น Mac) ภายใต้ความเหมาะสมฉันหมายถึงมันทำสำเนา แต่คุณต้องกด cmd + C สองครั้ง
Vadim Ivanov

@VadimIvanov จริง! ไม่มีใครรู้ว่าทำไม
เฟลิกซ์

1
@ don41382 ฉันไม่รู้เหมือนกันว่าทำไม แต่ฉันพบวิธีแก้ปัญหา คุณมีข้อบกพร่องเล็กน้อย onKeyDown คำสั่งแรกควรจะเป็นถ้า (! (e.ctrlKey || e.metaKey)) {return; } หมายความว่าเราต้องเตรียม textarea สำหรับการคัดลอกบน metaKey ที่ถูกกด (นี่เป็นวิธีที่พวก Trello ทำเคล็ดลับ) นี่คือรหัสจาก trello.com gist.github.com/fustic/10870311
Vadim Ivanov

@VadimIvanov ขอบคุณ ฉันจะแก้ไขด้านบน
เฟลิกซ์

1
มันไม่ทำงานใน FF 33.1 เพราะel.innerTextไม่ได้กำหนดดังนั้นฉันจึงเปลี่ยนบรรทัดสุดท้ายของclipboard()ฟังก์ชั่นเพื่อclip.setValue(el.innerText || el.textContent);ให้เข้ากันได้กับเบราว์เซอร์มากขึ้น link: jsfiddle.net/AGEf7/31
RevanProdigalKnight

7

รหัสของ 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;

})());

5

สิ่งที่คล้ายกันมากสามารถดูได้ที่http://goo.glเมื่อคุณย่อ URL

มีองค์ประกอบอินพุตแบบอ่านอย่างเดียวที่ให้ความสำคัญกับการเขียนโปรแกรมพร้อมด้วยคำแนะนำเครื่องมือกดCTRL-Cเพื่อคัดลอก

เมื่อคุณกดปุ่มลัดนั้นเนื้อหาอินพุตจะเข้าสู่คลิปบอร์ดได้อย่างมีประสิทธิภาพ ดีจริงๆ :)

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.