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

写在前面:

1.网上html2canvas资料少,即使你复制拿来用了,也发现各种问题是不是?
2.它的官网也没有好的demo,不好解决我们的问题。
3.我自己研究了两天,自创魂技,搞定了它。

以下不废话:

vue使用html2canvas截图时,存在svg或者img或者特殊字体时截图时空白,或者不全

如果你的原元素是img,当然再导入canvas,通过img转canvas去替换原来的子节点,再使用html2canvas导出图片。对于组件层级不多的页面是可用的。

 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截图,如果你这么做了成功了,图片没有缺失。那么恭喜你,因为大多数都会存在各种问题。

个人认为:除非你的项目很特殊,根本不需要用canvas去把svg转一下,因为转了以后会出现元素位置问题、删除和新增div节点的延时问题等等。下面自创思路就能完美解决问题。

解决办法如下

第一步,svg或者img先转base64(如果是特殊字体,就把这个字体图形截图转base64)

注意:最好是把转之前的图片尺寸先做好,可以用美图秀秀等把图片尺寸修改一下适配你原来的页面,再转base64,这样就省去了调节css的麻烦。
这里我推荐两个网站,可以将你的svg或img转base64

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就能实现截图下载。

::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设置为0
          scrollX: 0,
          //过滤不需要截图的元素
          ignoreElements: (element) => {
            return element.className === 'vxe-tree--btn-wrapper'
          }
        }
        //找到要截图的div节点,
        //可以用document.getElementById('aaa')等获取要截图的节点,根据实际情况而定
        const $image = divImage.value.childNodes[0].childNodes[1]
        
        //html2canvas 把div转图片
        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)
      }
    }
    
    //图片转Blob类型,避免浏览器报错
    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 })
    }

最近更新

  1. TCP协议是安全的吗?

    2024-06-09 16:50:02       18 阅读
  2. 阿里云服务器执行yum,一直下载docker-ce-stable失败

    2024-06-09 16:50:02       19 阅读
  3. 【Python教程】压缩PDF文件大小

    2024-06-09 16:50:02       18 阅读
  4. 通过文章id递归查询所有评论(xml)

    2024-06-09 16:50:02       20 阅读

热门阅读

  1. 移动端前端开发遇到过的Andorid和IOS的差异记录

    2024-06-09 16:50:02       10 阅读
  2. Audio音频资源播放

    2024-06-09 16:50:02       12 阅读
  3. springboot + easyRules 搭建规则引擎服务

    2024-06-09 16:50:02       10 阅读
  4. AI学习的基础理论路径

    2024-06-09 16:50:02       8 阅读
  5. es6 proxy的作用和用法

    2024-06-09 16:50:02       10 阅读
  6. 简单记录个python国内镜像源

    2024-06-09 16:50:02       9 阅读
  7. Spring Boot配置MySQL数据库连接数

    2024-06-09 16:50:02       11 阅读
  8. vue路由缓存

    2024-06-09 16:50:02       11 阅读
  9. 力扣第185题:部门工资前三高的员工

    2024-06-09 16:50:02       9 阅读