JavaScript 中的复制粘贴功能

发布于 2024-07-22 10:20:12 字数 6803 浏览 18 评论 0

一、基本使用

1.1 API 介绍

复制、剪切、粘贴事件

  • copy 发生复制操作时触发;
  • cut 发生剪切操作时触发;
  • paste 发生粘贴操作时触发

每个事件都有一个 before 事件对应: beforecopybeforecutbeforepaste 。这几个 before 一般不怎么用,所以我们把注意力放在另外三个事件就可以了

触发条件

  • 鼠标右键菜单的 复制粘贴剪切
  • 使用了相应的键盘组合键,比如: command+ccommand+v

使用姿势

copy 为例

document.body.oncopy = e => {
// 监听全局复制 做点什么
};
// 还有这种写法:
document.addEventListener('copy', e => {
// 监听全局复制 做点什么
});

上面是在 document.body 上全局监听的,然而很多人不知道的是,我们还可以为某些 dom 单独添加剪切板事件

// html 结构
<div id="test1"></div>
<div id="test2"></div>

<script>
// 写法一样:
let test1 = document.querySelector('#test1');
test1.oncopy = e => {
// 监听 test1 发生的复制事件 做点什么
// test1 发生的复制事件会触发回调,其他地方不会触发回调
}
</script>

其他事件也是一样的

1.2 clipboardData

clipboardData 对象:用于访问以及修改剪贴板中的数据

不同浏览器,所属的对象不同:在 IE 中这个对象是 window 对象的属性,在 ChromeSafariFirefox 中,这个对象是相应的 event 对象的属性。所以我们在使用的时候,需要做一下如下兼容

document.body.oncopy = e => {
let clipboardData = e.clipboardData || window.clipboardData;
// 获取 clipboardData 对象 + do something
};

对象方法

对象有三个方法: getData()setData()clearData()

getData() 访问剪切板中的数据

getData() 接受一个 text 参数,即要取得的数据的格式

在复制、剪切、粘贴触发的事件的数据

实际上在 chorme 上测试只有 paste 粘贴的时候才能用 getData() 访问到数据,用法如下

// 要粘贴的数据:

document.body.onpaste = e => {
let clipboardData = e.clipboardData || window.clipboardData; // 兼容处理
console.log('要粘贴的数据', clipboardData.getData('text'));
};

被复制/剪切的数据

在复制和剪切中的数据,需要通过 window.getSelection(0).toString() 来访问:

document.body.oncopy = e => {
console.log('被复制的数据:', window.getSelection(0).toString());
};

setData(): 修改剪切板中的数据

第一个参数也是 text ,第二个参数是要放在剪切板中的文本

clearData()

二、应用

2.1 复制大段文本

实现类知乎/掘金复制大段文本添加版权信息

实现很简单:取消默认复制之后,主要是在被复制的内容后面添加信息,然后根据 clipboardDatasetData() 方法将信息写入剪贴板

// 掘金这里不是全局监听,应该只是监听文章的 dom 范围内。
document.body.oncopy = event => {
event.preventDefault(); // 取消默认的复制事件
let textFont,
copyFont = window.getSelection(0).toString(); // 被复制的文字 等下插入
// 防知乎掘金 复制一两个字则不添加版权信息 超过一定长度的文字 就添加版权信息
if (copyFont.length > 10) {
textFont =
copyFont +
'\n' +
'作者:OBKoro1\n' +
'链接:https://juejin.im/user/58714f0e325b123db4a2eb95372/posts\n' +
'来源:掘金\n' +
'著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。';
} else {
textFont = copyFont; // 没超过十个字 则采用被复制的内容。
}
if (event.clipboardData) {
return event.clipboardData.setData('text', textFont); // 将信息写入粘贴板
} else {
// 兼容 IE
return window.clipboardData.setData('text', textFont);
}
};

然后 command+ccommand+v ,输出:

你复制的内容
作者:OBKoro1
链接:https://juejin.im/user/58714f0eb123db4a2eb95372/posts
来源:掘金
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

2.2 防复制功能

  • 禁止复制+剪切
  • 禁止右键,右键某些选项:全选,复制,粘贴等。
  • 禁用文字选择,能选择却不能复制,体验很差。
  • user-selectcss 禁止选择文本
// 禁止右键菜单
document.body.oncontextmenu = e => {
console.log(e, '右键');
return false;
// e.preventDefault();
};
// 禁止文字选择。
document.body.onselectstart = e => {
console.log(e, '文字选择');
return false;
// e.preventDefault();
};
// 禁止复制
document.body.oncopy = e => {
console.log(e, 'copy');
return false;
// e.preventDefault();
}
// 禁止剪切
document.body.oncut = e => {
console.log(e, 'cut');
return false;
// e.preventDefault();
};
// 禁止粘贴
document.body.onpaste = e => {
console.log(e, 'paste');
return false;
// e.preventDefault();
};
/** css 禁止文本选择 这样不会触发 js**/
body {
user-select: none;
-moz-user-select: none;
-webkit-user-select: none;
-ms-user-select: none;
}
  • 使用 e.preventDefault() 也可以禁用,但建议使用 return false 这样就不用去访问 ee 的方法了。
  • 示例中 document.body 全局都禁用了,也可以对 dom (某些区域) 进行禁用

破解防复制

上面的防复制方法通过 js+css 实现的,所以思路就是:禁用 js+取消 user-select 样式。

Chrome 浏览器的话:打开浏览器控制台,按 F1 进入 Setting ,勾选 Disable JavaScript (禁止 js )。

此时如果还不能复制的话,就要去找 user-select 样式,取消这个样式就可以了

2.3 点击复制功能

不能使用 clipboardData

  • IE 中可以用 window.clipboardData.setData('text','内容') 实现
  • 上文提到过,在 IEclipboardDatawindow 的属性
  • 而其他浏览器则是相应的 event 对象的属性,这实际上是一种安全措施,防止未经授权的访问,为了兼容其他浏览器,所以我们不能通过 clipboardData 来实现这种操作

具体做法

  • 创建一个隐藏的 input
  • 点击的时候,将要复制的内容放进 input 框中
  • 选择文本内容 input.select() 。这里只能用 input 或者 textarea 才能选择文本
  • document.execCommand("copy") ,执行浏览器的复制命令
function copyText() {
var text = document.getElementById('text').innerText; // 获取要复制的内容也可以传进来
var input = document.getElementById('input'); // 获取隐藏 input 的 dom
input.value = text; // 修改文本框的内容
input.select(); // 选中文本
document.execCommand('copy'); // 执行浏览器复制命令
alert('复制成功');
}

2.4 第三方库 clipboard

https://github.com/zenorocha/clipboard.js

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据

关于作者

勿挽旧人

暂无简介

0 文章
0 评论
23 人气
更多

推荐作者

謌踐踏愛綪

文章 0 评论 0

开始看清了

文章 0 评论 0

高速公鹿

文章 0 评论 0

alipaysp_PLnULTzf66

文章 0 评论 0

热情消退

文章 0 评论 0

白色月光

文章 0 评论 0

    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文