vue使用html2canvas截图下载时,存在svg或者img时截图不全的解决办法

使用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 })
    }
相关推荐
高山我梦口香糖7 分钟前
[react]searchParams转普通对象
开发语言·前端·javascript
m0_748235249 分钟前
前端实现获取后端返回的文件流并下载
前端·状态模式
m0_748240251 小时前
前端如何检测用户登录状态是否过期
前端
black^sugar1 小时前
纯前端实现更新检测
开发语言·前端·javascript
寻找沙漠的人2 小时前
前端知识补充—CSS
前端·css
GISer_Jing2 小时前
2025前端面试热门题目——计算机网络篇
前端·计算机网络·面试
m0_748245522 小时前
吉利前端、AI面试
前端·面试·职场和发展
理想不理想v2 小时前
webpack最基础的配置
前端·webpack·node.js
pubuzhixing2 小时前
开源白板新方案:Plait 同时支持 Angular 和 React 啦!
前端·开源·github
2401_857600952 小时前
SSM 与 Vue 共筑电脑测评系统:精准洞察电脑世界
前端·javascript·vue.js