使用html2canvas進行div截圖時,存在svg和img的解決辦法
- 寫在前面:
- vue使用html2canvas截圖時,存在svg或者img時截圖時空白,或者不全
- 解決辦法如下
- 第一步,svg或者img先轉base64
- 第二步,將轉換后的base64設置為新元素的content屬性:
- 第三步,實現div的截圖下載
寫在前面:
1.網上html2canvas資料少,即使你復制拿來用了,也發現各種問題是不是?
2.它的官網也沒有好的demo,不好解決我們的問題。
3.我自己研究了兩天,自創絕招,搞定了它。
以下不廢話:
vue使用html2canvas截圖時,存在svg或者img時截圖時空白,或者不全
當然再導入canvas,通過img轉canvas去替換原來的子節點,再使用html2canvas導出圖片。對于組件層級不多的頁面是可用的。
const changeToCanvas = async (element) => {const imgElems = element.querySelectorAll('img')let svgelems = [...imgElems]svgelems.forEach((node) => {let parentNode = node.parentNodelet canvas = document.createElement('canvas')canvas.style.zIndex = 999//處理img轉換canvasif (node.tagName == 'IMG') {canvas.width = node.widthcanvas.height = node.heightcanvas.getContext('2d').drawImage(node, 0, 0)parentNode.removeChild(node)let index = parentNode.firstChildif (index != null) {parentNode.insertBefore(canvas, index)} else {parentNode.appendChild(canvas)}}})}
但是如果你組件層級多時,你會發現這種辦法不是很好用,容易出現各種意想不到的問題。
如果你的原元素是svg,那么思路是,遍歷div內的svg,把svg轉圖片,遍歷圖片轉canvas再用html2canvas截圖,如果你這么做了成功了,圖片沒有缺失。那么恭喜你,因為大多數都會存在各種問題。
解決辦法如下
第一步,svg或者img先轉base64
這里我推薦兩個網站,可以將你的svg或img轉base64
https://www.chahuo.com/svg2css.html#google_vignettehttps://www.bejson.com/image/imgzoom/#google_vignette
第二步,將轉換后的base64設置為新元素的content屬性:
如下,用新的div的css的content屬性代替原來的svg或者img。這樣就不需要canvas,只用html2canvas就能實現截圖下載。
::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
npm install html2canvas@1.4.1
二:引入
import html2canvas from 'html2canvas'
三:實現
// 生成圖片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設置為0scrollX: 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 = urldocument.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.lengthvar u8arr = new Uint8Array(n)while (n--) {u8arr[n] = bstr.charCodeAt(n)}return new Blob([u8arr], { type: mime })}