[英]How does Trello access the user's clipboard?
当您将鼠标悬停在Trello 中的卡片上并按Ctrl + C 时,此卡片的 URL 将复制到剪贴板。 他们怎么做到的?
据我所知,没有涉及 Flash 电影。 我已经安装了Flashblock ,并且 Firefox 网络选项卡显示没有加载 Flash 电影。 (这是通常的方法,例如 ZeroClipboard。)
他们是如何实现这种魔力的?
(此时我想我有一个顿悟:你不能在页面上选择文本,所以我假设他们有一个不可见的元素,他们通过 JavaScript 代码创建一个文本选择, Ctrl + C触发浏览器的默认行为,复制那个不可见节点的文本值。)
披露:我编写了 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) ->
在 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键时要选择什么。
我实际上构建了一个 Chrome 扩展程序,它完全可以做到这一点,并且适用于所有网页。 源代码在 GitHub 上。
我发现 Trello 的方法存在三个错误,我知道这是因为我自己也遇到过:)
该副本在这些情况下不起作用:
我通过总是有一个隐藏的跨度来解决 #1,而不是在用户点击Ctrl / Cmd时创建一个。
我通过临时清除零长度选择、保存插入符号位置、进行复制和恢复插入符号位置来解决 #2。
我还没有找到 #3 的修复程序:)(有关信息,请查看我的 GitHub 项目中的未解决问题)。
在 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/
Daniel LeCheminant 的代码在将其从 CoffeeScript 转换为 JavaScript ( js2coffee ) 后对我不起作用。 它一直在_.defer()
行上轰炸。
我认为这与 jQuery deferreds 有关,所以我将其更改为$.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进行复制。
当您点击该快捷方式时,输入内容会有效地进入剪贴板。 真的很好:)
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.