【Web API系列】使用异步剪贴板API(async clipboard)的图像的编程复制和粘贴

本文主要是介绍【Web API系列】使用异步剪贴板API(async clipboard)的图像的编程复制和粘贴,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在这里插入图片描述

文章目录

  • 前言
  • 一、将数据写入剪切板
    • 1. WriteText()
    • 2. Write()
    • 3. 监听复制事件
  • 二、从剪切板读取数据
    • 1.readText()
    • 2.read()
    • 3. 处理粘贴的文件
    • 4. 监听读剪切板事件
  • 三、申请权限
    • 政策集成
  • 四、功能检测
  • 五、处理多个 MIME 类型


前言

访问系统剪贴板的传统方法是通过 document.execCommand() 进行剪贴板交互。虽然这种剪切和粘贴方法受到广泛支持,但还是有代价的:剪贴板访问是同步的,并且只能对 DOM 执行读写操作。

这对于少量文字来说没什么问题,但在很多情况下,阻止相应网页以进行剪贴板传输会带来糟糕的体验。可能需要耗时的清理或图片解码,才能安全粘贴内容。浏览器可能需要从粘贴的文档加载或内嵌链接的资源。这样会在等待磁盘或网络时阻塞网页。想象一下,向混合中添加权限,要求浏览器在请求剪贴板访问权限时屏蔽网页。同时,针对剪贴板交互的 document.execCommand() 设置的权限较为宽松,并且因浏览器而异。

但是 Chrome 从 104 版开始支持网页自定义格式,可让开发者将任意数据写入剪贴板。

如果你仍然对该技术知之甚少,建议看看过去大佬 Matt Gaunt 的文章是怎么实现怎么实现的。

现在,主流的浏览器已经原生支持这种功能。那就是Async Clipboard: Read and Write Images

我所指的现在是指

浏览器最低版本号是否支持
Chrome86
Edge79
Firefox63不完整
Safari13.1

接下来我将从官方提供的几个示例来介绍这个API。


一、将数据写入剪切板

1. WriteText()

如果只是想要将文本内容复制到剪切板,可以使用writeText()。由于此 API 是异步的,因此writeText()函数会返回一个根据传递的文本是否成功复制来解析或拒绝的 Promise:

async function copyPageUrl() {try {await navigator.clipboard.writeText(location.href);console.log('页面URL已经复制到剪切板');} catch (err) {console.error('复制失败: ', err);}
}

2. Write()

如果你想要把剪切板中写入图片,那就使用Write()方法,但是要注意需要用 blob 格式的图片作为参数,或者fetch()等请求服务器图片,调用blob()方法转换成为合适的格式。也可以将图片绘制到canvas里面,然后调用toBlob()方法。

接下来,将 ClipboardItem 对象数组作为参数传递给 write() 方法。但是一次只能传递一张图片。ClipboardItem 接受一个对象,将图片的 MIME 类型作为键,并使用 blob 作为值。对于从 fetch() 或 canvas.toBlob() 获取的 blob 对象,blob.type 属性会自动包含图片的正确 MIME 类型。

一次复制多张图片等官方后面更新。

try {const imgURL = '/images/generic/file.png';const data = await fetch(imgURL);const blob = await data.blob();await navigator.clipboard.write([new ClipboardItem({// 键“blob.type”动态决定blob的文件格式[blob.type]: blob})]);console.log('图片复制完毕');
} catch (err) {console.error(err.name, err.message);
}

或者你可以规定复制的文件格式

try {const imgURL = '/images/generic/file.png';await navigator.clipboard.write([new ClipboardItem({'image/png': fetch(imgURL).then(response => response.blob()),})]);console.log('图片复制完毕');
} catch (err) {console.error(err.name, err.message);
}

3. 监听复制事件

如果用户复制到剪贴板,但未调用 preventDefault(),则 copy 事件会包含一个 clipboardData 属性,其中包含的内容已采用正确的格式。

document.addEventListener("copy", async (e) => {// 阻止事件冒泡e.preventDefault();try {// 准备clipboardItems对象数组let clipboardItems = [];// 将图片放到剪切板clipboardItems.push(new ClipboardItem({[blob.type]: blob,}));await navigator.clipboard.write(clipboardItems);console.log("图片已被复制,文字已省略");} catch (err) {console.error(err.name, err.message);}
});

二、从剪切板读取数据

1.readText()

从剪切板读取文本数据,调用navigator.clipboard.readText()代码如下

async function getClipboardContents() {try {const text = await navigator.clipboard.readText();console.log('粘贴内容: ', text);} catch (err) {console.error('内容读取失败: ', err);}
}

2.read()

从剪贴板读取图片,需要获取 ClipboardItem 对象列表,然后遍历它们。

原话: 每个 ClipboardItem 可以将其内容保存在不同的类型中,因此您需要使用 for…of 循环遍历类型列表。对于每种类型,请使用当前类型作为参数调用 getType() 方法,以获取相应的 blob。与之前一样,此代码未与图片相关联,并且将适用于未来的其他文件类型。

async function getClipboardContents() {try {const clipboardItems = await navigator.clipboard.read();for (const clipboardItem of clipboardItems) {for (const type of clipboardItem.types) {const blob = await clipboardItem.getType(type);console.log(URL.createObjectURL(blob));}}} catch (err) {console.error(err.name, err.message);}
}

3. 处理粘贴的文件

让用户能够使用剪贴板键盘快捷键(例如 ctrl + c 和 ctrl + v)非常有用。Chromium 会在剪贴板上公开只读文件,如下所述。 当用户点击操作系统的默认粘贴快捷方式,或当用户点击浏览器菜单栏中的修改,然后再点击粘贴时,就会触发此事件。

document.addEventListener("paste", async e => {e.preventDefault();if (!e.clipboardData.files.length) {return;}const file = e.clipboardData.files[0];// 加入读取的文件是文本文件// note: 只能读,不能写console.log(await file.text());
});

4. 监听读剪切板事件

处理粘贴的文件其实就是通过监听度剪切板的功能来实现的,看下面用例

document.addEventListener('paste', async (e) => {e.preventDefault();const text = await navigator.clipboard.readText();console.log('粘贴的文本: ', text);
});

三、申请权限

这类原生API对于跨端应用会非常有用,申请权限就是为了这个做准备的。当你需要使用剪切板的功能的时候,必须申请权限才能使用,此处调用代码与其他API基本一致,直接给出代码供大家参考。

const queryOpts = { name: 'clipboard-read', allowWithoutGesture: false };
const permissionStatus = await navigator.permissions.query(queryOpts);
// 结果是 'granted'(已授权), 'denied'(已拒绝) or 'prompt'(询问):
console.log(permissionStatus.state);// 监听权限状态改变
permissionStatus.onchange = () => {console.log(permissionStatus.state);
};

allowWithoutGesture 选项控制是否需要用户手势才能调用剪切或粘贴。

由于浏览器仅允许页面为活动标签页时访问剪贴板,您会发现,如果直接粘贴到浏览器的控制台中,此处的部分示例不会运行,因为开发者工具本身就是活跃标签页。有一种技巧:使用 setTimeout() 延迟对剪贴板的访问,然后在页面内快速点击使其成为焦点,然后再调用函数:

setTimeout(async () => {const text = await navigator.clipboard.readText();console.log(text);
}, 2000);

政策集成

在iframe中使用剪切板API的时候用的,用例如下

<iframesrc="index.html"allow="clipboard-read; clipboard-write"
>
</iframe>

四、功能检测

在使用剪切板API的时候可以提前检测浏览器是否支持这个API,如果不支持则使用过去浏览器的方案,会让你的功能变得更加可靠,提供高用户体验。官方给出样例

document.addEventListener('paste', async (e) => {e.preventDefault();let text;if (navigator.clipboard) {text = await navigator.clipboard.readText();}else {text = e.clipboardData.getData('text/plain');}console.log('从剪切板获取到的文本: ', text);
});

五、处理多个 MIME 类型

对于一次剪切或复制操作,大多数实现都会将多种数据格式放到剪贴板中。这有两个原因:作为应用开发者,您无法知道用户想要将文本或图片复制到的应用的功能,并且许多应用支持将结构化数据粘贴为纯文本。这通常通过修改菜单项向用户显示,对应的名称为粘贴和匹配样式或粘贴(不带格式)。

下面示例中使用的是fetch()得到的数据,这是因为读取本地文件需要获取API权限。

async function copy() {const image = await fetch('kitten.png').then(response => response.blob());const text = new Blob(['Cute sleeping kitten'], {type: 'text/plain'});const item = new ClipboardItem({'text/plain': text,'image/png': image});await navigator.clipboard.write([item]);
}

这篇关于【Web API系列】使用异步剪贴板API(async clipboard)的图像的编程复制和粘贴的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


原文地址:
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.chinasem.cn/article/556712

相关文章

postgresql使用UUID函数的方法

《postgresql使用UUID函数的方法》本文给大家介绍postgresql使用UUID函数的方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录PostgreSQL有两种生成uuid的方法。可以先通过sql查看是否已安装扩展函数,和可以安装的扩展函数

如何使用Lombok进行spring 注入

《如何使用Lombok进行spring注入》本文介绍如何用Lombok简化Spring注入,推荐优先使用setter注入,通过注解自动生成getter/setter及构造器,减少冗余代码,提升开发效... Lombok为了开发环境简化代码,好处不用多说。spring 注入方式为2种,构造器注入和setter

基于Python开发一个图像水印批量添加工具

《基于Python开发一个图像水印批量添加工具》在当今数字化内容爆炸式增长的时代,图像版权保护已成为创作者和企业的核心需求,本方案将详细介绍一个基于PythonPIL库的工业级图像水印解决方案,有需要... 目录一、系统架构设计1.1 整体处理流程1.2 类结构设计(扩展版本)二、核心算法深入解析2.1 自

MySQL中比较运算符的具体使用

《MySQL中比较运算符的具体使用》本文介绍了SQL中常用的符号类型和非符号类型运算符,符号类型运算符包括等于(=)、安全等于(=)、不等于(/!=)、大小比较(,=,,=)等,感兴趣的可以了解一下... 目录符号类型运算符1. 等于运算符=2. 安全等于运算符<=>3. 不等于运算符<>或!=4. 小于运

使用zip4j实现Java中的ZIP文件加密压缩的操作方法

《使用zip4j实现Java中的ZIP文件加密压缩的操作方法》本文介绍如何通过Maven集成zip4j1.3.2库创建带密码保护的ZIP文件,涵盖依赖配置、代码示例及加密原理,确保数据安全性,感兴趣的... 目录1. zip4j库介绍和版本1.1 zip4j库概述1.2 zip4j的版本演变1.3 zip4

Python 字典 (Dictionary)使用详解

《Python字典(Dictionary)使用详解》字典是python中最重要,最常用的数据结构之一,它提供了高效的键值对存储和查找能力,:本文主要介绍Python字典(Dictionary)... 目录字典1.基本特性2.创建字典3.访问元素4.修改字典5.删除元素6.字典遍历7.字典的高级特性默认字典

使用Python构建一个高效的日志处理系统

《使用Python构建一个高效的日志处理系统》这篇文章主要为大家详细讲解了如何使用Python开发一个专业的日志分析工具,能够自动化处理、分析和可视化各类日志文件,大幅提升运维效率,需要的可以了解下... 目录环境准备工具功能概述完整代码实现代码深度解析1. 类设计与初始化2. 日志解析核心逻辑3. 文件处

一文详解如何使用Java获取PDF页面信息

《一文详解如何使用Java获取PDF页面信息》了解PDF页面属性是我们在处理文档、内容提取、打印设置或页面重组等任务时不可或缺的一环,下面我们就来看看如何使用Java语言获取这些信息吧... 目录引言一、安装和引入PDF处理库引入依赖二、获取 PDF 页数三、获取页面尺寸(宽高)四、获取页面旋转角度五、判断

C++中assign函数的使用

《C++中assign函数的使用》在C++标准模板库中,std::list等容器都提供了assign成员函数,它比操作符更灵活,支持多种初始化方式,下面就来介绍一下assign的用法,具有一定的参考价... 目录​1.assign的基本功能​​语法​2. 具体用法示例​​​(1) 填充n个相同值​​(2)

Spring StateMachine实现状态机使用示例详解

《SpringStateMachine实现状态机使用示例详解》本文介绍SpringStateMachine实现状态机的步骤,包括依赖导入、枚举定义、状态转移规则配置、上下文管理及服务调用示例,重点解... 目录什么是状态机使用示例什么是状态机状态机是计算机科学中的​​核心建模工具​​,用于描述对象在其生命