记一次修复移动端html2canvas加载图片失败

起因

事情的起因要从一个优化说起,项目要优化访问速度于是转而启用了阿里服务器。访问域名与静态资源域名不同。这导致html2canvas在请求图片资源时出现跨域的情况。当然官方文档也给出了相应的解决办法:

js 复制代码
html2canvas(selector, {useCORS: true // 跨域请求资源})
.then((canvas) => {
   ...});

当然移动端对于上述的方案也给出了新的问题

是的就是直接报错。这种情况在chrome上不会出现。安卓和ios环境下图片就会加载失败。img标签的 crossOrigin属性、路径加时间戳破除缓存等方法都不奏效。

初步的解决方案

因为看到页面内有些base64的图片可以完好的被转换出来。所以突发奇想既然插件请求不行。不如我就给他把图片都转成base64。 当然是借助canvas:

js 复制代码
let url = ''
const image = new Image()
image.onload = () => {
    const canvas = document.createElement('canvas')        
    const ctx = canvas.getContext('2d')             
    ctx.drawImage( image, 0, 0 )       
    url = canvas.toDataURL('image/png', 1)    }    
    image.crossOrigin = 'anonymous' // 跨域请求资源   
    image.src = xxx //图片地址
}

优化的版本

上述方法成功将图片转为base64格式,但是转出来的图片比例严重变形。是因为canvas有默认宽高。找到原因便很好的就可以解决了。我们可以设置canvas的宽高为图片元素的宽高。同时为了解决清晰度问题还添加了如下代码:

js 复制代码
const el = document.querySelector(selector)
const h = el.clientHeight
const w = el.clientWidth
image.onload = () => {
    const canvas = document.createElement('canvas')
    const ctx = canvas.getContext('2d')
    const ratio =
          (window.devicePixelRatio || 1) /
          (ctx.backingStorePixelRatio ||
            ctx.webkitBackingStorePixelRatio ||
            ctx.mozBackingStorePixelRatio ||
            ctx.msBackingStorePixelRatio ||
            ctx.oBackingStorePixelRatio ||
            ctx.backingStorePixelRatio ||
            1)
            
    canvas.width = w * ratio        
    canvas.height = h * ratio

    ctx.drawImage(image, 0, 0, h * ratio, w * ratio) 
    url = canvas.toDataURL('image/png', 1)
}

新的问题

chrome测试完美,安卓浏览器测试也没问题,当我认为这样问题就完美解决了的时候。在safari那html2canvas甩给我一个新的报错

arduino 复制代码
RangeError: Maximum call stack size exceeded

于是去GitHub上找到了对应的issue,这个类似的问题在mac的浏览器上报过。页面中如果有太多base64图片时,会导致这个错误。我这边将常量SLICE_STACK_SIZE修改为3000可以避免这个报错。

相关推荐
UXbot3 小时前
UI设计工具推荐合集
前端·人工智能·ui
敲敲了个代码3 小时前
如何优化批量图片上传?队列机制+分片处理+断点续传三连击!(附源码)
前端·javascript·学习·职场和发展·node.js
@AfeiyuO3 小时前
Vue 引入全局样式scss
前端·vue·scss
光影少年3 小时前
flex布局和grid布局区别,实现两边固定布局中间自适应
前端·css3·web·ai编程
全栈测试笔记4 小时前
异步函数与异步生成器
linux·服务器·前端·数据库·python
EndingCoder4 小时前
配置 tsconfig.json:高级选项
linux·前端·ubuntu·typescript·json
木风小助理4 小时前
JavaStreamAPI的性能审视,优雅语法背后的隐形成本与优化实践
java·前端·数据库
white-persist5 小时前
轻松抓包微信小程序:Proxifier+Burp Suite教程
前端·网络·安全·网络安全·微信小程序·小程序·notepad++
敲敲了个代码6 小时前
多标签页强提醒不重复打扰:从“弹框轰炸”到“共享待处理队列”的实战
java·前端·javascript·面试·架构
不想上班只想要钱6 小时前
动态类名在 <swiper-slide 的复制项中没有起作用的解决方法
前端·vue.js