乐趣区

关于javascript:clipboardjs开源的复制粘贴库

一、clipboard.js 简介

clipboard.js 是一个用于将文本复制到剪贴板的 JS 库。没有应用 Flash,没有应用任何框架,开启 gzipped 压缩后仅仅只有 3kb

(图片起源:https://clipboardjs.com/#exam…)

那么为什么会有 clipboard.js 这个库呢?因为作者 zenorocha 认为:

将文本复制到剪贴板应该不难。它不须要几十个步骤来配置,也不须要加载数百 KB 的文件。最最重要的是,它不应该依赖于 Flash 或其余任何框架。

该库依赖于 Selection 和 execCommand API,简直所有的浏览器都反对 Selection API,然而 execCommand API 却存在肯定的兼容性问题:

(图片起源:https://caniuse.com/?search=e…)

(图片起源:https://caniuse.com/?search=e…)

当然对于较老的浏览器,clipboard.js 也能够优雅地降级。好的,当初咱们来看一下如何应用 clipboard.js。

二、clipboard.js 应用

在应用 clipboard.js 之前,你能够通过 NPM 或 CDN 的形式来装置它:

NPM

`npm install clipboard –save
`

CDN

`<script src=”https://cdn.jsdelivr.net/npm/clipboard@2.0.6/dist/clipboard.min.js”></script>
`

clipboard.js 应用起来很简略,个别只有 3 个步骤:

1. 定义一些标记

`<input id=”foo” type=”text” value=” 大家好,我是阿宝哥 ”>
<button class=”btn” data-clipboard-action=”copy” data-clipboard-target=”#foo”> 复制 </button>
`

2. 引入 clipboard.js

`<script src=”https://cdn.jsdelivr.net/npm/clipboard@2.0.6/dist/clipboard.min.js”></script>
`

3. 实例化 clipboard

`<script> var clipboard = new ClipboardJS(‘.btn’);
  clipboard.on(‘success’, function(e) {
    console.log(e);
  });
    
  clipboard.on(‘error’, function(e) {
    console.log(e);
  });</script>
`

以上代码胜利运行之后,当你点击“复制”按钮时,输入框中的文字会被选中,同时输入框中的文字将会被复制到剪贴板中,对应的成果如下图所示:

除了 input 元素之外,复制的指标还能够是 div 或 textarea 元素。在以上示例中,咱们复制的指标是通过 data-* 属性 来指定。此外,咱们也能够在实例化 clipboard 对象时,设置复制的指标:

`// https://github.com/zenorocha/…
let clipboard = new ClipboardJS(‘.btn’, {
  target: function() {
    return document.querySelector(‘div’);
  }
});
`

如果须要设置复制的文本,咱们也能够在实例化 clipboard 对象时,设置复制的文本:

`// https://github.com/zenorocha/…
let clipboard = new ClipboardJS(‘.btn’, {
  text: function() {
    return ‘to be or not to be’;
  }
});
`

对于 clipboard.js 的应用,阿宝哥就介绍到这里,感兴趣的小伙伴能够查看 Github 上 clipboard.js 的应用示例。

因为 clipboard.js 底层依赖于 Selection 和 execCommand API,所以在剖析 clipboard.js 源码前,咱们先来理解一下 Selection 和 execCommand API。

三、Selection 与 execCommand API

3.1 Selection API

Selection 对象示意用户抉择的文本范畴或插入符号的以后地位。它代表页面中的文本选区,可能横跨多个元素。文本选区由用户拖拽鼠标通过文字而产生。如果要获取用于查看或批改的 Selection 对象,能够调用 window.getSelection 办法。

Selection 对象所对应的是用户所抉择的 ranges(区域),俗称 拖蓝。默认状况下,该函数只针对一个区域,咱们能够这样应用这个函数:

`let selection = window.getSelection();
let range = selection.getRangeAt(0);
`

以上示例演示了如何获取选区中的第一个区域,其实除了获取选区中的区域之外,咱们还能够通过 createRange API 创立一个新的区域,而后将该区域增加到选区中:

`<div> 大家好,我是 阿宝哥 。欢送关注 全栈修仙之路</div>
<script> let strongs = document.getElementsByTagName(“strong”);
   let s = window.getSelection();
   if (s.rangeCount > 0) s.removeAllRanges(); // 从选区中移除所有区域
   for (let i = 0; i < strongs.length; i++) {
     let range = document.createRange(); // 创立 range 区域
     range.selectNode(strongs[i]); // 让 range 区域蕴含指定节点及其内容
     s.addRange(range); // 将创立的区域增加到选区中
   }</script>
`

以上代码用于选中页面中所有的 strong 元素,但须要留神的是,目前只有应用 Gecko 渲染引擎的浏览器,比方 Firefox 浏览器实现了多个区域。

在某些场景下,你可能须要获取选中区域中的文本。针对这种场景,你能够通过调用 Selection 对象的 toString 办法来获取被选中区域中的纯文本。

3.2 execCommand API

document.execCommand API 容许运行命令来操作网页中的内容,罕用的命令有 bold、italic、copy、cut、delete、insertHTML、insertImage、insertText 和 undo 等。上面咱们来看一下该 API 的语法:

`bool = document.execCommand(aCommandName, aShowDefaultUI, aValueArgument)
`

相干的参数阐明如下:

  • aCommandName:字符串类型,用于表示命令的名称;
  • aShowDefaultUI:布尔类型,用于示意是否展现用户界面,个别为 false;
  • aValueArgument:额定参数,一些命令(比方 insertImage)须要额定的参数(提供插入图片的 URL),默认为 null。

调用 document.execCommand 办法后,该办法会返回一个布尔值。如果是 false 的话,示意操作不被反对或未被启用。对于 clipboard.js 这个库来说,它会通过 document.execCommand API 来执行 copy 和 cut 命令,从而实现把内容复制到剪贴板。

那么当初问题来了,咱们有没有方法判断以后浏览器是否反对 copy 和 cut 命令呢?答案是有的,即应用浏览器提供的 API —— Document.queryCommandSupported,该办法容许咱们确定以后的浏览器是否反对指定的编辑命令。

clipboard.js 这个库的作者,也思考到了这种需要,所以提供了一个动态的 isSupported 办法,用于检测以后的浏览器是否反对指定的命令:

`// src/clipboard.js
static isSupported(action = [‘copy’, ‘cut’]) {
  const actions = (typeof action === ‘string’) ? [action] : action;
  let support = !!document.queryCommandSupported;
  actions.forEach((action) => {
    support = support && !!document.queryCommandSupported(action);
  });
  return support;
}
`

Document.queryCommandSupported 兼容性较好,大家能够放心使用,具体的兼容性如下图所示:

(图片起源:https://caniuse.com/?search=q…)

介绍完 Selection、execCommand 和 queryCommandSupported API,接下来咱们开始剖析 clipboard.js 的源码。

如果你想理解浏览源码的思路与技巧,能够浏览 应用这些思路与技巧,我读懂了多个优良的开源我的项目 这篇文章。

四、clipboard.js 源码解析

4.1 Clipboard 类

看源码的时候,阿宝哥习惯从最简略的用法动手,这样能够疾速地理解外部的执行流程。上面咱们来回顾一下后面的示例:

`<!– 定义一些标记 –>
<input id=”foo” type=”text” value=” 大家好,我是阿宝哥 ”>
<button class=”btn” data-clipboard-action=”copy” data-clipboard-target=”#foo”> 复制 </button>
<!– 实例化 clipboard –>
<script> let clipboard = new ClipboardJS(‘.btn’);
  clipboard.on(‘success’, function(e) {
    console.log(e);
  });
    
  clipboard.on(‘error’, function(e) {
    console.log(e);
  });</script>
`

通过观察以上的代码,咱们能够疾速地找到切入点 —— new ClipboardJS('.btn')。在 clipboard.js 我的项目内的 webpack.config 配置文件中,咱们能够找到 ClipboardJS 的定义:

`module.exports = {
  entry: ‘./src/clipboard.js’,
  mode: ‘production’,
  output: {
    filename: production ? ‘clipboard.min.js’ : ‘clipboard.js’,
    path: path.resolve(__dirname, ‘dist’),
    library: ‘ClipboardJS’,
    globalObject: ‘this’,
    libraryExport: ‘default’,
    libraryTarget: ‘umd’
  },
  // 省略其余配置信息
}
`

基于以上的配置信息,咱们进一步找到了 ClipboardJS 指向的构造函数:

`import Emitter from ‘tiny-emitter’;
import listen from ‘good-listener’;
class Clipboard extends Emitter {
  constructor(trigger, options) {
    super();
    this.resolveOptions(options);
    this.listenClick(trigger);
  }
}
`

在示例中,咱们并没有设置 Clipboard 的配置信息,所以咱们先不必关怀 this.resolveOptions(options) 的解决逻辑。顾名思义 listenClick 办法是用来监听 click 事件,该办法的具体实现如下:

`listenClick(trigger) {
  this.listener = listen(trigger, ‘click’, (e) => this.onClick(e));
}
`

在 listenClick 办法外部,会通过一个第三方库 good-listener 来增加事件处理器。当指标触发 click 事件时,就会执行对应的事件处理器,该处理器外部会进一步调用 this.onClick 办法,该办法的实现如下:

`// src/clipboard.js
onClick(e) {
  const trigger = e.delegateTarget || e.currentTarget;
  // 为每次点击事件,创立一个新的 ClipboardAction 对象
  if (this.clipboardAction) {
    this.clipboardAction = null;
  }
  this.clipboardAction = new ClipboardAction({
    action    : this.action(trigger),
    target    : this.target(trigger),
    text      : this.text(trigger),
    container : this.container,
    trigger   : trigger,
    emitter   : this
  });
}
`

在 onClick 办法外部,会应用事件触发指标来创立 ClipboardAction 对象。当你点击本示例 复制 按钮时,创立的 ClipboardAction 对象如下所示:

置信看完上图,大家对创立 ClipboardAction 对象时,所应用到的办法都有理解。那么 this.actionthis.target 和 this.text 这几个办法是在哪里定义的呢?通过浏览源码,咱们发现在 resolveOptions 办法外部会初始化上述 3 个办法:

`// src/clipboard.js
resolveOptions(options = {}) {
  this.action = (typeof options.action === ‘function’) 
    ? options.action :  this.defaultAction;
  this.target = (typeof options.target === ‘function’) 
    ? options.target : this.defaultTarget;
  this.text = (typeof options.text === ‘function’)
    ? options.text : this.defaultText;
  this.container = (typeof options.container === ‘object’)   
    ? options.container : document.body;
}
`

在 resolveOptions 办法外部,如果用户自定义了处理函数,则会优先应用用户自定义的函数,否则将应用 clipboard.js 中对应的默认处理函数。因为咱们在调用 Clipboard 构造函数时,并未设置 options 参数,所以将应用默认的处理函数:

由上图可知在 defaultActiondefaultTarget 和 defaultText 办法外部都会调用 getAttributeValue 办法来获取事件触发对象上自定义属性,而对应的 getAttributeValue 办法也很简略,具体代码如下:

“// src/clipboard.js
function getAttributeValue(suffix, element) {
  const attribute = data-clipboard-${suffix};
  if (!element.hasAttribute(attribute)) {
    return;
  }
  return element.getAttribute(attribute);
}

介绍完 Clipboard 类,接下来咱们来重点剖析一下 ClipboardAction 类,该类会蕴含具体的复制逻辑。

4.2 ClipboardAction 类

在 clipboard.js 我的项目中,ClipboardAction 类被定义在 src/clipboard-action.js 文件内:

`// src/clipboard-action.js
class ClipboardAction {
  constructor(options) {
    this.resolveOptions(options);
    this.initSelection();
  }
}
`

与 Clipboard 类的构造函数一样,ClipboardAction 类的构造函数会优先解析 options 配置对象,而后调用 initSelection 办法,来初始化选区。在 initSelection 办法中会依据 text 和 target 属性来抉择不同的抉择策略:

`initSelection() {
  if (this.text) {
    this.selectFake();
  } else if (this.target) {
    this.selectTarget();
  }
}
`

对于后面的示例,咱们是通过 data-* 属性 来指定复制的指标,即 data-clipboard-target="#foo",相应的代码如下:

`<input id=”foo” type=”text” value=” 大家好,我是阿宝哥 ”>
<button class=”btn” data-clipboard-action=”copy” data-clipboard-target=”#foo”> 复制 </button>
`

所以接下来咱们先来剖析含有 target 属性的情景,如果含有 target 属性,则会进入 else if 分支,而后调用 this.selectTarget 办法:

`// src/clipboard-action.js
selectTarget() {
  this.selectedText = select(this.target);
  this.copyText();
}
`

在 selectTarget 办法外部,会调用 select 函数获取已选中的文本,该函数是来自 clipboard.js 作者开发的另一个 npm 包,对应的代码如下:

`// https://github.com/zenorocha/…
function select(element) {
  var selectedText;
  if (element.nodeName === ‘SELECT’) {
    element.focus();
    selectedText = element.value;
  }
  else if (element.nodeName === ‘INPUT’ || element.nodeName === ‘TEXTAREA’) {
    var isReadOnly = element.hasAttribute(‘readonly’);
    if (!isReadOnly) {
      element.setAttribute(‘readonly’, ”);
    }
    element.select();
    element.setSelectionRange(0, element.value.length);
    if (!isReadOnly) {
      element.removeAttribute(‘readonly’);
    } 
      selectedText = element.value;
    }
  else {
    // 省略相干代码
  }
  return selectedText;
}
`

因为在以上示例中,咱们复制的指标是 input 元素,所以咱们先来剖析该分支的代码。在该分支中,应用了 HTMLInputElement 对象的 select 和 setSelectionRange 办法:

  • select:用于选中一个 <textarea> 元素或者一个带有 text 字段的 <input> 元素里的所有内容。
  • setSelectionRange:用于设定 <input> 或 <textarea> 元素中以后选中文本的起始和完结地位。

在获取选中的文本之后,selectTarget 办法会持续调用 copyText 办法来复制文本:

`copyText() {
  let succeeded;
  try {
    succeeded = document.execCommand(this.action);
  } catch (err) {
    succeeded = false;
  }
  this.handleResult(succeeded);
}
`

后面阿宝哥曾经简略介绍了 execCommand API,copyText 办法外部就是应用这个 API 来复制文本。在实现复制之后,copyText 办法会调用 this.handleResult 办法来派发复制的状态信息:

`handleResult(succeeded) {
  this.emitter.emit(succeeded ? ‘success’ : ‘error’, {
    action: this.action,
    text: this.selectedText,
    trigger: this.trigger,
    clearSelection: this.clearSelection.bind(this)
  });
}
`

看到这里有些小伙伴可能会问 this.emitter 对象是来自哪里的?其实 this.emitter 对象也就是 Clipboard 实例:

`// src/clipboard.js
class Clipboard extends Emitter {
  onClick(e) {
    const trigger = e.delegateTarget || e.currentTarget;
    // 省略局部代码
    this.clipboardAction = new ClipboardAction({
      // 省略局部属性
      trigger   : trigger,
      emitter   : this // Clipboard 实例
    });
  }
}
`

而对于 handleResult 办法派发的事件,咱们能够通过 clipboard 实例来监听对应的事件,具体的代码如下:

`let clipboard = new ClipboardJS(‘.btn’);
clipboard.on(‘success’, function(e) {
  console.log(e);
});
    
clipboard.on(‘error’, function(e) {
  console.log(e);
});
`

在持续介绍另一个分支的解决逻辑之前,阿宝哥用一张图来总结一下上述示例的执行流程:

上面咱们来介绍另一个分支,即含有 text 属性的情景,对应的应用示例如下:

`// https://github.com/zenorocha/…
let clipboard = new ClipboardJS(‘.btn’, {
  text: function() {
    return ‘ 大家好,我是阿宝哥 ’;
  }
});
`

当用户在创立 clipboard 对象时,设置了 text 属性,则会执行 if 分支的逻辑,即调用 this.selectFake 办法:

`// src/clipboard-action.js
class ClipboardAction {
  constructor(options) {
    this.resolveOptions(options);
    this.initSelection();
  }
  
  initSelection() {
    if (this.text) {
      this.selectFake();
    } else if (this.target) {
      this.selectTarget();
    }
  }
}
`

在 selectFake 办法外部,它会先创立一个假的 textarea 元素并设置该元素的相干款式和定位信息,并应用 this.text 的值来设置 textarea 元素的内容,而后应用后面介绍的 select 函数来获取已抉择的文本,最初通过 copyText 把文本拷贝到剪贴板:

“// src/clipboard-action.js
selectFake() {
  const isRTL = document.documentElement.getAttribute(‘dir’) == ‘rtl’;
  this.removeFake(); // 移除事件监听并移除之前创立的 fakeElem
  this.fakeHandlerCallback = () => this.removeFake();
  this.fakeHandler = this.container.addEventListener(‘click’, this.fakeHandlerCallback) || true;
  this.fakeElem = document.createElement(‘textarea’);
  // Prevent zooming on iOS
  this.fakeElem.style.fontSize = ’12pt’;
  // Reset box model
  this.fakeElem.style.border = ‘0’;
  this.fakeElem.style.padding = ‘0’;
  this.fakeElem.style.margin = ‘0’;
  // Move element out of screen horizontally
  this.fakeElem.style.position = ‘absolute’;
  this.fakeElem.style[isRTL ? ‘right’ : ‘left’] = ‘-9999px’;
  // Move element to the same position vertically
  let yPosition = window.pageYOffset || document.documentElement.scrollTop;
  this.fakeElem.style.top = ${yPosition}px;
  this.fakeElem.setAttribute(‘readonly’, ”);
  this.fakeElem.value = this.text;
  this.container.appendChild(this.fakeElem);
  this.selectedText = select(this.fakeElem);
  this.copyText();
}

为了让大家可能更直观理解 selectFake 办法执行后的页面成果,阿宝哥截了一张理论的效果图:

其实 clipboard.js 除了反对拷贝 input 或 textarea 元素的内容之外,它还反对拷贝其它 HTML 元素的内容,比方 div 元素:

`<div> 大家好,我是阿宝哥 </div>
<button class=”btn” data-clipboard-action=”copy” data-clipboard-target=”div”>Copy</button>
`

针对这种情景,在 clipboard.js 外部仍会利用后面介绍的 select 函数来选中指标元素并获取需拷贝的内容,具体的代码如下所示:

`function select(element) {
  var selectedText;
  if (element.nodeName === ‘SELECT’) {
      element.focus();
      selectedText = element.value;
  }
  else if (element.nodeName === ‘INPUT’ || element.nodeName === ‘TEXTAREA’) {
      // 省略相干代码
  }
  else {
     if (element.hasAttribute(‘contenteditable’)) {
        element.focus();
     }
     var selection = window.getSelection(); // 创立选取
     var range = document.createRange(); // 新建区域
     range.selectNodeContents(element); // 使新建的区域蕴含 element 节点的内容
     selection.removeAllRanges(); // 移除选取中的所有区域
     selection.addRange(range); // 往选区中增加新建的区域
     selectedText = selection.toString(); // 获取已选中的文本
    }
    return selectedText;
}
`

在取得要拷贝的文本之后,clipboard.js 会持续调用 copyText 办法把对应的文本拷贝到剪贴板。到这里 clipboard.js 的外围源码,咱们差不多都剖析完了,心愿浏览本文后,大家不仅理解了 clipboard.js 背地的工作原理,同时也学会了如何利用事件派发器来实现音讯通信 及 Selection 和 execCommand API 等相干的常识。

五、参考资源

  • MDN Selection
  • MDN execCommand
  • MDN queryCommandSupported
  • MDN selectNodeContents
退出移动版