Trello でカードの上にカーソルを置き、を押すと Ctrl+CこのカードのURLがクリップボードにコピーされます。彼らはどのようにこれをしますか?
私が言える限りでは、Flashムービーは関与していません。 Flashblock をインストールしましたが、FirefoxのネットワークタブにFlashムービーがロードされていないと表示されます。 (これは、例えばZeroClipboardによる通常の方法です。)
彼らはどのようにしてこの魔法を達成しますか?
(現時点で私は私はエピファニーを持っていたと思います:あなたはページ上のテキストを選択することができません。 Ctrl+C ブラウザのデフォルトの動作を引き起こし、その見えないノードのテキスト値をコピーします。)
ディスクロージャー: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 キーが押されています。
私は実際に構築しました Chrome拡張機能 これはまさにこれを、そしてすべてのWebページに対して行います。ソースコードは GitHub上 です。
私はTrelloのアプローチに3つのバグを見つけました。それらは私が自分自身に直面したからです。
コピーはこれらのシナリオでは動作しません。
ユーザーがヒットしたときにスパンを作成するのではなく、常に非表示のスパンを持つことで#1を解決しました Ctrl/Cmd。
#2を解決するには、長さゼロの選択を一時的にクリアし、キャレット位置を保存し、コピーを実行して、キャレット位置を復元します。
私はまだ#3の修正を見つけることができませんでした:)(詳細については、私のGitHubプロジェクトの未解決の問題を確認してください)。
Raincoatの( 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/ /
CoffeeScriptからJavaScript( js2coffee )に変換した後、Daniel LeCheminantのコードが機能しませんでした。それは_.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;
})());
URLを短くすると、 http://goo.gl に非常によく似たものが表示されます。
プログラム的にフォーカスされる読み取り専用のinput要素があり、「コピーするにはCtrl + Cキーを押す」というツールチップがあります。このショートカットを押すと、入力内容が事実上クリップボードに入ります。すごくいい :)