总结一下使用html2canvas截图时遇到的问题

近期做需求,有个页面截图下载到本地的功能,这里记录一下使用html2canvas时遇到的问题,简单搭建了一个demo项目(项目地址),用来调试当时遇到的问题。 具体问题如下:

  1. 图片跨域问题
  2. 图片清晰度问题
  3. 图片样式缺失问题
  4. 长图截取不全问题

使用

  1. 安装

官网地址: html2canvas.hertzen.com/

js 复制代码
npm install --save html2canvas
  1. 使用

html2canvas接收两个参数,分别是截取的目标dom元素和配置项,配置项非必传。调用html2canvas方法后,会返回一个绘制好的canvas对象,再通过调用canvas对象的toDataURL方法可以转换成一个图片。

js 复制代码
// 具体代码如下:
const canvas = await html2canvas(document.body, options); // options为对应的配置项
const code = canvas.toDataURL(); // 转换为Base64编码的字符串
const aLink = document.createElement('a'); // 创建a标签自动下载
aLink.download = '下载图片';
aLink.href = code;
aLink.click();

遇到的问题

1. 跨域问题

html2canvas配置项中自带了两个跟跨域相关的配置字段,分别是allowTaintuseCORS

1、allowTaint

由于html2canvas受浏览器同源策略的限制,使用非同源的图片会taint(污染)画布,因此默认allowTaint: false,不允许污染画布,同时图片也不能画在画布上。

在配置项中设置allowTaint: true,虽然会污染画布,但可以将跨域的图片渲染到画布上了。

js 复制代码
const canvas = await html2canvas(element, { allowTaint: true });

不过我不太推荐这种方式,因为它会导致另外一个问题,被污染的画布因为同源策略而存在安全问题 ,导致调用html2canvas后返回的canvas对象中toDataUrl、getImageData等方法会报错,从而没办法将canvas画布信息转换成base64格式的字符串了,这不符合我的预期。 所以我将这个配置项pass掉,不用allowTaint,而是使用useCORS。

  • 使用allowTaint属性后,报错如下:

2、useCORS

在配置项中设置useCORS:true,可以解决跨域问题,原理相同。

js 复制代码
const canvas = await html2canvas(element, { useCORS: true });

图片我用百度的图标当做网络图片,效果图如下:

  • 未设置useCORS
  • 设置useCORS

注意: allowTaintuseCORS不要同时使用,如果二者同时设置为true,仍然会认为画布已被污染而不可用。

2. 图片清晰度问题

通过配置项将scale属性对应的值调整大一点即可,我项目中设置的是4, 在html2canvasscale默认值为浏览器设备像素比

js 复制代码
const canvas = await html2canvas(element, { scale: 4 });

未设置scale和设置scale下载图片到本地后,通过下面这张图可以明显看出来,设置了scale的图片大小要比没有设置的大很多。

下面两张图是通过调整scale对应值的大小,图片的清晰度对比会更加明显一些。

  • 设置scale: 0.8
  • 设置scale: 4

3. 图片样式缺失问题

此问题的原因在于html2canvas有一些css样式是不支持的,导致截取出来的图片有部分样式缺失。因此写页面的时候只要自己注意一下就好了。

官方文档上面也有说明支持哪些css属性,不支持哪些css属性。具体详情可以点击传送门

下面这些css属性是当前不支持的:

4. 长图截取不全问题

遇到这个问题的时候,当时查看官方文档没有找到对应的解决方案,后来自己调试样式,发现设置截图元素的高度就可以完美解决了。

  • 解决方案: 给要截图的元素设置height:fit-content

效果图:

注意:fit-content有兼容行问题,使用的时候需要注意该属性是否符合你的目标浏览器的兼容性。

给截图元素不设置高度或者设置height: auto发现都不起作用,后来尝试使用fit-content这个相对较新的属性后,可以完美解决高度问题。

  • 扩展知识

height: fit-content用于根据元素内容的大小来自动调整元素的高度。它允许你根据内容动态地设置元素的高度,而不是使用固定的像素值或百分比。(这段话来自chatGPT)

另外fit-content可以和min-heightmax-height搭配使用。

  1. min-height搭配使用:当内容实际高度小于指定的最小高度时,元素的高度为min-height指定的高度,否则为元素的实际高度。。
  2. max-height搭配使用:当内容高度大于指定的最大高度时,元素的高度为max-height指定的高度,否则为元素的实际高度。

以上是我在做这个功能时遇到的问题,文中若有不对的地方欢迎各位掘友纠正~。

参考文章

  1. html2canvas.hertzen.com/configurati...
  2. blog.fuwenhao.com/post/351.ht...
相关推荐
打瞌睡的朱尤4 分钟前
CSS复习
前端·css
程序员 沐阳5 分钟前
异步编程深潜:事件循环、Promise 与 async/await 的底层真相
javascript
irpywp8 分钟前
Boneyard:基于组件映射的骨架屏方案
前端·ui·github
276695829210 分钟前
zp_stoken 算法风控分析
java·前端·javascript·python·web逆向·boss直聘·zp_stoken
叫我一声阿雷吧12 分钟前
JS 入门通关手册(38):防抖与节流 原理 + 手写 + 实战场景(面试必考)
javascript·性能优化·前端面试·防抖·节流·js手写题
妮妮喔妮21 分钟前
组件的封装
开发语言·前端·javascript
cypking28 分钟前
前端瓦片渲染解决方案(解决大量数据渲染卡顿问题)
前端
李子焱28 分钟前
第三节:开发环境搭建与Trae IDE深度配置
前端·ide·python·node.js·trae ide
王家视频教程图书馆1 小时前
electron 环境搭建
前端·javascript·electron
速易达网络1 小时前
Vue 3 的无人机送餐飞控数字大屏
前端