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 })
    }
相关推荐
小曲曲5 分钟前
接口上传视频和oss直传视频到阿里云组件
javascript·阿里云·音视频
学不会•1 小时前
css数据不固定情况下,循环加不同背景颜色
前端·javascript·html
EasyNTS2 小时前
H.264/H.265播放器EasyPlayer.js视频流媒体播放器关于websocket1006的异常断连
javascript·h.265·h.264
活宝小娜4 小时前
vue不刷新浏览器更新页面的方法
前端·javascript·vue.js
程序视点4 小时前
【Vue3新工具】Pinia.js:提升开发效率,更轻量、更高效的状态管理方案!
前端·javascript·vue.js·typescript·vue·ecmascript
coldriversnow4 小时前
在Vue中,vue document.onkeydown 无效
前端·javascript·vue.js
我开心就好o4 小时前
uniapp点左上角返回键, 重复来回跳转的问题 解决方案
前端·javascript·uni-app
开心工作室_kaic5 小时前
ssm161基于web的资源共享平台的共享与开发+jsp(论文+源码)_kaic
java·开发语言·前端
刚刚好ā5 小时前
js作用域超全介绍--全局作用域、局部作用、块级作用域
前端·javascript·vue.js·vue
沉默璇年6 小时前
react中useMemo的使用场景
前端·react.js·前端框架