近期做需求,有个页面截图下载到本地的功能,这里记录一下使用html2canvas时遇到的问题,简单搭建了一个demo项目(项目地址),用来调试当时遇到的问题。 具体问题如下:
- 图片跨域问题
- 图片清晰度问题
- 图片样式缺失问题
- 长图截取不全问题
使用
- 安装
官网地址: html2canvas.hertzen.com/
npm install --save html2canvas
- 使用
html2canvas接收两个参数,分别是截取的目标dom元素和配置项,配置项非必传。调用html2canvas方法后,会返回一个绘制好的canvas对象,再通过调用canvas对象的toDataURL
方法可以转换成一个图片。
// 具体代码如下:
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配置项中自带了两个跟跨域相关的配置字段,分别是allowTaint
和useCORS
1、allowTaint
由于html2canvas受浏览器同源策略的限制,使用非同源的图片会taint(污染)画布,因此默认allowTaint: false
,不允许污染画布,同时图片也不能画在画布上。
在配置项中设置allowTaint: true
,虽然会污染画布,但可以将跨域的图片渲染到画布上了。
const canvas = await html2canvas(element, { allowTaint: true });
不过我不太推荐这种方式,因为它会导致另外一个问题,被污染的画布因为同源策略而存在安全问题,导致调用html2canvas
后返回的canvas对象
中toDataUrl、getImageData等方法会报错,从而没办法将canvas
画布信息转换成base64格式的字符串了,这不符合我的预期。
所以我将这个配置项pass掉,不用allowTaint,而是使用useCORS。
- 使用allowTaint属性后,报错如下:
2、useCORS
在配置项中设置useCORS:true
,可以解决跨域问题,原理相同。
const canvas = await html2canvas(element, { useCORS: true });
图片我用百度的图标当做网络图片,效果图如下:
- 未设置useCORS
- 设置useCORS
注意:
allowTaint
和useCORS
不要同时使用,如果二者同时设置为true,仍然会认为画布已被污染而不可用。
2. 图片清晰度问题
通过配置项将scale
属性对应的值调整大一点即可,我项目中设置的是4, 在html2canvas
中scale
默认值为浏览器设备像素比
。
const canvas = await html2canvas(element, { scale: 4 });
未设置scale
和设置scale
下载图片到本地后,通过下面这张图可以明显看出来,设置了scale
的图片大小要比没有设置的大很多。
下面两张图是通过调整scale对应值的大小,图片的清晰度对比会更加明显一些。
- 设置
scale: 0.8
- 设置
scale: 4
3. 图片样式缺失问题
此问题的原因在于html2canvas
有一些css样式是不支持的,导致截取出来的图片有部分样式缺失。因此写页面的时候只要自己注意一下就好了。
官方文档上面也有说明支持哪些css属性,不支持哪些css属性。具体详情可以点击传送门。
下面这些css属性是当前不支持的:
- background-blend-mode
- border-image
- box-decoration-break
- box-shadow
- filter
- font-variant-ligatures
- mix-blend-mode
- object-fit
- repeating-linear-gradient()
- writing-mode
- zoom
4. 长图截取不全问题
遇到这个问题的时候,当时查看官方文档没有找到对应的解决方案,后来自己调试样式,发现设置截图元素的高度就可以完美解决了。
- 解决方案:
给要截图的元素设置
height:fit-content
。
效果图:
注意:fit-content有兼容性问题,使用的时候需要注意该属性是否符合你的目标浏览器的兼容性。
给截图元素不设置高度或者设置height: auto
发现都不起作用,后来尝试使用fit-content
这个相对较新的属性后,可以完美解决高度问题。
- 扩展知识
height: fit-content
用于根据元素内容的大小来自动调整元素的高度。它允许你根据内容动态地设置元素的高度,而不是使用固定的像素值或百分比。(这段话来自chatGPT)
另外fit-content
可以和min-height
和max-height
搭配使用。
- 与
min-height
搭配使用:当内容实际高度小于指定的最小高度时,元素的高度为min-height
指定的高度,否则为元素的实际高度。。 - 与
max-height
搭配使用:当内容高度大于指定的最大高度时,元素的高度为max-height
指定的高度,否则为元素的实际高度。
以上是我在做这个功能时遇到的问题,文中若有不对的地方欢迎各位掘友纠正~。