本文主要是介绍vue-clipboard2在vue的created生命周期中直接调用copyText方法报错的原因分析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
vue-clipboard2在vue的created生命周期中直接调用copyText方法报错
先说现象:在created
生命周期中会进入reject
状态(被catch
到),不在生命周期的方法中调用而通过click
事件来调用会正常进入resolved
状态(成功进入then
阶段)。
下面进行相关源码分析:
出错代码:
created() {this.$copyText('asdasdasdas').then(() => {console.log('复制成功');}).catch(err => {console.error('复制出错', err); // 执行到这里了})
},
打开vue-clipboard2
的源码,可以发现底层使用了clipboard
这个库:
Vue.prototype.$copyText = function (text, container) {return new Promise(function (resolve, reject) {var fakeElement = document.createElement('button')// 注意这里,使用了Clipboard的构造方法var clipboard = new Clipboard(fakeElement, {text: function () { return text },action: function () { return 'copy' },container: typeof container === 'object' ? container : document.body})clipboard.on('success', function (e) {clipboard.destroy()resolve(e)})// 注意这里clipboard.on('error', function (e) {clipboard.destroy()reject(e)})if (VueClipboardConfig.appendToBody) document.body.appendChild(fakeElement)fakeElement.click()if (VueClipboardConfig.appendToBody) document.body.removeChild(fakeElement)})
}
在它的package.json
中找到clipboard
的依赖,确保别找错了:
"dependencies": {"clipboard": "^2.0.0"
},
再看一下我们安装的clipboard
的版本:
"version": "2.0.4"
github
上搜一下这个库的源码:https://github.com/zenorocha/clipboard.js
方便查找代码的引用关系,我们去这个网址:https://sourcegraph.com/github.com/zenorocha/clipboard.js@master/-/blob/src/clipboard.js
之前的代码调用了clipboard
的构造方法:
constructor(trigger, options) {super();this.resolveOptions(options);this.listenClick(trigger);}/*** Defines if attributes would be resolved using internal setter functions* or custom functions that were passed in the constructor.* @param {Object} options*/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;}/*** Adds a click event listener to the passed trigger.* @param {String|HTMLElement|HTMLCollection|NodeList} trigger*/listenClick(trigger) {this.listener = listen(trigger, 'click', (e) => this.onClick(e));}
trigger
是vue-clipboard2
传进来的button
实例,listenClick
做的就是给这个button
加上click
事件,再看一下onClick
方法:
onClick(e) {// button实例 delegateTarget是事件委托dom,这里我们走的是currentTargetconst trigger = e.delegateTarget || e.currentTarget;if (this.clipboardAction) {this.clipboardAction = null;}this.clipboardAction = new ClipboardAction({action : this.action(trigger), // 'copy'target : this.target(trigger), // undefinedtext : this.text(trigger), // 'text' => 传入的text参数container : this.container, // 默认为bodytrigger : trigger,emitter : this});}
this.target
方法再初始化时被定义成下面这个函数,因为vue-clipboard2
没有传这个参数
defaultTarget(trigger) {const selector = getAttributeValue('target', trigger); // 返回undefined,因为button没有target这个属性 if (selector) {return document.querySelector(selector);}}
下面再看看ClipboardAction
的构造方法做了什么:
constructor(options) {this.resolveOptions(options);this.initSelection();}/*** Defines base properties passed from constructor.* @param {Object} options*/resolveOptions(options = {}) {this.action = options.action;this.container = options.container;this.emitter = options.emitter;this.target = options.target;this.text = options.text;this.trigger = options.trigger;this.selectedText = '';}/*** Decides which selection strategy is going to be applied based* on the existence of `text` and `target` properties.*/initSelection() {if (this.text) {this.selectFake();}else if (this.target) {this.selectTarget();}}
主要是对传进来的参数进行本地赋值,看到initSelection
方法,进入了第一个分支:
selectFake() {const isRTL = document.documentElement.getAttribute('dir') == 'rtl';// 这个方法做的事情是删除textarea节点,清除container上的click事件// 将fakeHandler,fakeHandlerCallback,fakeElem置为nullthis.removeFake();this.fakeHandlerCallback = () => this.removeFake();this.fakeHandler = this.container.addEventListener('click', this.fakeHandlerCallback) || true;this.fakeElem = document.createElement('textarea');// Prevent zooming on iOSthis.fakeElem.style.fontSize = '12pt';// Reset box modelthis.fakeElem.style.border = '0';this.fakeElem.style.padding = '0';this.fakeElem.style.margin = '0';// Move element out of screen horizontallysthis.fakeElem.style.position = 'absolute';this.fakeElem.style[ isRTL ? 'right' : 'left' ] = '-9999px';// Move element to the same position verticallylet 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();}
创建了一个textarea
的dom
节点,value
为我们传进去的text
。
select
方法为外部依赖,做的事情是帮我们选中textarea
中的文字。接下来调用了copyText
方法,
/*** Executes the copy operation based on the current selection.*/copyText() {let succeeded;try {succeeded = document.execCommand(this.action); // this.action === 'copy'}catch (err) {succeeded = false;}this.handleResult(succeeded);}
可以看到调用了execCommand
方法来执行操作系统的copy
方法,而我们报的错是在handleResult
中emit
出来的,所以我们的$copyText
方法进入了catch
分支。
// vue-clipboard2的监听事件clipboard.on('error', function (e) {clipboard.destroy()reject(e)}) handleResult(succeeded) {this.emitter.emit(succeeded ? 'success' : 'error', {action: this.action,text: this.selectedText,trigger: this.trigger,clearSelection: this.clearSelection.bind(this)});}
也就是说succeeded
变量值为false
,这一点在我们断点调试一下可以发现确实返回了fasle。
为什么呢?先看一下MDN文档对于execCommand
方法的说明:
bool = document.execCommand(aCommandName, aShowDefaultUI, aValueArgument);
// 一个 Boolean ,如果是 false 则表示操作不被支持或未被启用。
// 注意:在调用一个命令前,不要尝试使用返回值去校验浏览器的兼容性
可是我的浏览器是chrome 78
,按理说支持这个方法啊,可是为什么会返回false
呢?
返回false
的原因其实也是浏览器对安全性的考虑,因为copy
这个操作不是由用户操作产生的,而是由代码自执行的,所以默认执行失败。
document.execCommand的特殊性
浏览器处于安全考虑,document.execCommand
这个api
只能在真正的用户操作之后才能被触发。
以下引用自W3C草案:
If an implementation supports ways to execute clipboard commands through scripting, for example by calling the
document.execCommand()
method with the commands “cut”, “copy” and “paste”, the implementation must trigger the corresponding action, which again will dispatch the associated clipboard event.
copy
事件的执行过程:
- If the script-triggered flag is set, then
- If the script-may-access-clipboard flag is unset, then
- Return false from the copy action, terminate this algorithm
- Fire a clipboard event named copy
- If the event was not canceled, then
- Copy the selected contents, if any, to the clipboard. Implementations should create alternate text/html and text/plain clipboard formats when content in a web page is selected.
- Fire a clipboard event named clipboardchange
- Else, if the event was canceled, then
- Call the write content to the clipboard algorithm, passing on the
DataTransferItemList
list items, a clear-was-called flag and a types-to-clear list.- Return true from the copy action
参考链接
Cannot use document.execCommand('copy');
from developer console
execCommand(‘copy’) does not work in Ajax / XHR callback?
W3C:Clipboard API and events
这篇关于vue-clipboard2在vue的created生命周期中直接调用copyText方法报错的原因分析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!