使用html2canvas进行div截图时,存在svg和img的解决办法
写在前面:
1.网上html2canvas资料少,即使你复制拿来用了,也发现各种问题是不是?
2.它的官网也没有好的demo,不好解决我们的问题。
3.我自己研究了两天,自创绝招,搞定了它。
以下不废话:
vue使用html2canvas截图时,存在svg或者img时截图时空白,或者不全
当然再导入canvas,通过img转canvas去替换原来的子节点,再使用html2canvas导出图片。对于组件层级不多的页面是可用的。
javascript
const changeToCanvas = async (element) => {
const imgElems = element.querySelectorAll('img')
let svgelems = [...imgElems]
svgelems.forEach((node) => {
let parentNode = node.parentNode
let canvas = document.createElement('canvas')
canvas.style.zIndex = 999
//处理img转换canvas
if (node.tagName == 'IMG') {
canvas.width = node.width
canvas.height = node.height
canvas.getContext('2d').drawImage(node, 0, 0)
parentNode.removeChild(node)
let index = parentNode.firstChild
if (index != null) {
parentNode.insertBefore(canvas, index)
} else {
parentNode.appendChild(canvas)
}
}
})
}
但是如果你组件层级多时,你会发现这种办法不是很好用,容易出现各种意想不到的问题。
如果你的原元素是svg,那么思路是,遍历div内的svg,把svg转图片,遍历图片转canvas再用html2canvas截图,如果你这么做了成功了,图片没有缺失。那么恭喜你,因为大多数都会存在各种问题。
解决办法如下
第一步,svg或者img先转base64
这里我推荐两个网站,可以将你的svg或img转base64
javascript
https://www.chahuo.com/svg2css.html#google_vignette
https://www.bejson.com/image/imgzoom/#google_vignette
第二步,将转换后的base64设置为新元素的content属性:
如下,用新的div的css的content属性代替原来的svg或者img。这样就不需要canvas,只用html2canvas就能实现截图下载。
javascript
::v-deep(.vxe-icon-square-minus) {
pointer-events: none;
&::before {
content: url('');
background-repeat: no-repeat;
background-position: center;
width: 1em;
height: 1em;
display: inline-block;
}
}
::v-deep(.vxe-icon-square-plus) {
pointer-events: none;
&::before {
content: url('');
background-repeat: no-repeat;
background-position: center;
width: 1em;
height: 1em;
display: inline-block;
}
}
::v-deep(.icon_cri_1) {
&::before {
content: url('');
background-repeat: no-repeat;
background-position: center;
width: 1em;
height: 1em;
margin-right: 5px;
margin-top: 3px;
display: inline-block;
}
}
::v-deep(.icon_cri) {
&::before {
content: url('');
background-repeat: no-repeat;
background-position: center;
width: 1em;
height: 1em;
margin-right: 5px;
display: inline-block;
margin-top: 3px;
}
}
第三步,实现div的截图下载
一:html2canvas npm
javascript
npm install html2canvas@1.4.1
二:引入
javascript
import html2canvas from 'html2canvas'
三:实现
javascript
// 生成图片
const creatImg = async () => {
try {
const setup = {
useCORS: true, // 使用跨域
backgroundColor: null,
allowTaint: true,
logging: true, // 打印日志来检查是否有加载问题
background: '#ffffff', // 一定要添加背景颜色,否则出来的图片,背景全部都是透明的
async: true, // 是否异步解析和呈现元素
scale: 2, // 处理模糊问题
// dpi: 1000 // 处理模糊问题
scrollY: 0, // html2canvas默认绘制视图内的页面,需要把scrollY,scrollX设置为0
scrollX: 0,
ignoreElements: (element) => {//过滤不需要截图的元素
return element.className === 'vxe-tree--btn-wrapper'
}
}
const $image = divImage.value.childNodes[0].childNodes[1]
await html2canvas($image, setup).then((canvasimg) => {
const link = canvasimg.toDataURL('image/png')
const fileName = 'CRI_History_Image_' + getDate()
exportPicture(link, fileName)
})
} catch (e) {
console.log(e)
}
}
// 导出图片
const exportPicture = (link, name) => {
try {
const file = document.createElement('a')
file.style.display = 'none'
file.download = decodeURI(name)
let blob = dataURLtoBlob(link)
let url = URL.createObjectURL(blob)
file.href = url
document.body.appendChild(file)
file.click()
document.body.removeChild(file)
} catch (e) {
console.log(e)
}
}
const dataURLtoBlob = (dataurl) => {
var arr = dataurl.split(',')
var mime = arr[0].match(/:(.*?);/)[1]
var bstr = atob(arr[1])
var n = bstr.length
var u8arr = new Uint8Array(n)
while (n--) {
u8arr[n] = bstr.charCodeAt(n)
}
return new Blob([u8arr], { type: mime })
}