利用canvas实现图片下载功能来实现浏览器兼容问题


Posted in HTML / CSS onMay 31, 2019

前言:项目中需要实现图片下载功能,第一个想到的是使用a标签的download属性来实现,但是在不同浏览器下测试会发现,有的浏览器无效,点击后直接预览图片,所以,上网找到了另外一种兼容不同浏览器的图片下载的方法,那就是利用canvas来处理图片,实现下载;

1.项目中点击事件绑定:

<a href="#" @click.prevent="downloadIamge(imgsrc, name)"><span>{{name}}</span></a>

2.点击事件中操作:

downloadIamge (imgsrc, name) {
      const url = imgsrc
      this.convertUrlToBase64(url).then((base64) => {
        const blob = this.convertBase64UrlToBlob(base64)
        if (getBrowser() === 'IE' || getBrowser() === 'Edge') {
          window.navigator.msSaveBlob(blob, name)
        } else {
          const a = document.createElement('a')
          const body = document.querySelector('body')
          a.download = name || 'image'
          a.href = URL.createObjectURL(blob)
          a.style.display = 'none'
          body.appendChild(a)
          a.click()
          body.removeChild(a)
          window.URL.revokeObjectURL(a.href)
        }
      })
    },

3.this.convertUrlToBase64(url)就是利用canvas和toDataURL把图片转成base64格式并返回

convertUrlToBase64 (url) {
      return new Promise((resolve, reject) => {
        const img = new Image()
        img.crossOrigin = 'Anonymous'
        img.src = url
        img.onload = function () {
          const canvas = document.createElement('canvas')
          canvas.width = img.width
          canvas.height = img.height
          const ctx = canvas.getContext('2d')
          ctx.drawImage(img, 0, 0, img.width, img.height)
          const ext = img.src.substring(img.src.lastIndexOf('.') + 1).toLowerCase()
          const dataURL = canvas.toDataURL('image/' + ext)
          const base64 = {
            dataURL: dataURL,
            type: 'image/' + ext,
            ext: ext
          }
          resolve(base64)
        }
      })
    },

其中:img.crossOrigin = 'Anonymous'是前端对图片的跨域处理;

4.this.convertBase64UrlToBlob(base64)是将图片base64流文件转成blob文件

convertBase64UrlToBlob (base64) {
      const parts = base64.dataURL.split('base64,')
      const contentType = parts[0].split(':')[1]
      const raw = window.atob(parts[1])
      const rawLength = raw.length
      const uInt8Array = new Uint8Array(rawLength)
      for (let i = 0; i < rawLength; i++) {
        uInt8Array[i] = raw.charCodeAt(i)
      }
      return new Blob([uInt8Array], { type: contentType })
    },

5.getBrowser()用来判断浏览器,解决浏览器兼容性问题:

import { getBrowser } from '@/utils/utils'
export function getBrowser () {
  const userAgent = navigator.userAgent
  if (userAgent.indexOf('OPR') > -1) {
    return 'Opera'
  }
  if (userAgent.indexOf('Firefox') > -1) {
    return 'FF'
  }
  if (userAgent.indexOf('Trident') > -1) {
    return 'IE'
  }
  if (userAgent.indexOf('Edge') > -1) {
    return 'Edge'
  }
  if (userAgent.indexOf('Chrome') > -1) {
    return 'Chrome'
  }
  if (userAgent.indexOf('Safari') > -1) {
    return 'Safari'
  }
}

6.如果是IE或者Edge浏览器,可以直接使用window.navigator.msSaveBlob(blob, name)完成下载;

声明:由于ios系统有安全性限制,以上方法在ios上无效;

以上就是记录项目中用到的图片下载,浏览器兼容的问题,涉及到的base64和blob的知识点和原理还不是很清晰,有时间一定要研究一下,整个方法,亲测有效;欢迎测用,与意见反馈。也希望大家多多支持三水点靠木。

HTML / CSS 相关文章推荐
纯css3制作网站后台管理面板
Dec 30 HTML / CSS
举例详解CSS3中的Transition
Jul 15 HTML / CSS
CSS3制作漂亮的照片墙的实现代码
Jun 08 HTML / CSS
详解Canvas事件绑定
Jun 27 HTML / CSS
浅谈three.js中的needsUpdate的应用
Nov 12 HTML / CSS
详解HTML5中的Communication API基本使用方法
Jan 29 HTML / CSS
HTML5实现桌面通知 提示功能
Oct 11 HTML / CSS
HTML5页面直接调用百度地图API获取当前位置直接导航目的地的实现代码
Mar 02 HTML / CSS
使用canvas压缩图片上传的方法示例
Feb 07 HTML / CSS
Canvas波浪花环的示例代码
Aug 21 HTML / CSS
移动端HTML5 input常见问题(小结)
Sep 28 HTML / CSS
CSS3中Animation实现简单的手指点击动画的示例
Jul 15 HTML / CSS
HTML5印章绘制电子签章图片(中文英文椭圆章、中文英文椭圆印章)
Jun 03 #HTML / CSS
详解利用canvas实现环形进度条的方法
Jun 12 #HTML / CSS
Html5 实现微信分享及自定义内容的流程
Aug 20 #HTML / CSS
前端canvas动画如何转成mp4视频的方法
Jun 17 #HTML / CSS
详解FireFox下Canvas使用图像合成绘制SVG的Bug
Jul 10 #HTML / CSS
canvas实现有递增动画的环形进度条的实现方法
Jul 10 #HTML / CSS
Html5新增标签与样式及让元素水平垂直居中
Jul 11 #HTML / CSS
You might like
PHP面向对象程序设计之对象克隆clone和魔术方法__clone()用法分析
2019/06/12 PHP
thinkPHP和onethink微信支付插件分享
2019/08/11 PHP
PHP高并发和大流量解决方案整理
2019/12/24 PHP
JScript中的&quot;this&quot;关键字使用方式补充材料
2007/03/08 Javascript
jquery下组织javascript代码(js函数化)
2010/08/25 Javascript
jQuery实现的导航条切换可显示隐藏
2014/10/22 Javascript
js获取iframe中的window对象的实现方法
2016/05/20 Javascript
javascript设计模式之module(模块)模式
2016/08/19 Javascript
Bootstrap源码解读网格系统(3)
2016/12/22 Javascript
微信小程序 Template详解及简单实例
2017/01/05 Javascript
BootStrapValidator初使用教程详解
2017/02/10 Javascript
vue父组件向子组件(props)传递数据的方法
2018/01/02 Javascript
vue-cli axios请求方式及跨域处理问题
2018/03/28 Javascript
vue子组件改变父组件传递的prop值通过sync实现数据双向绑定(DEMO)
2020/02/01 Javascript
基于Vue+Webpack拆分路由文件实现管理
2020/11/16 Javascript
[08:08]DOTA2-DPC中国联赛2月28日Recap集锦
2021/03/11 DOTA
Python操作MongoDB数据库PyMongo库使用方法
2015/04/27 Python
mac 安装python网络请求包requests方法
2018/06/13 Python
mac下pycharm设置python版本的图文教程
2018/06/13 Python
python实现图片横向和纵向拼接
2020/03/05 Python
Python基于类路径字符串获取静态属性
2020/03/12 Python
python编写一个会算账的脚本的示例代码
2020/06/02 Python
基于Django集成CAS实现流程详解
2020/11/28 Python
canvas生成带二维码海报的踩坑记录
2019/09/11 HTML / CSS
德国户外商店:eXXpozed
2020/07/25 全球购物
"序列点" 是什么
2016/07/29 面试题
企业项目策划书
2014/01/11 职场文书
自主招生自荐信指南
2014/02/04 职场文书
网络编辑求职信
2014/04/30 职场文书
祖国在我心中演讲稿200字
2014/08/28 职场文书
党政领导班子群众路线对照检查材料
2014/10/26 职场文书
房屋分割离婚协议书范本
2014/12/01 职场文书
社区安全温馨提示语
2015/07/14 职场文书
导游词书写之黄山
2019/08/06 职场文书
解析在浏览器地址栏输入一个URL后发生了什么
2021/06/21 Servers
ObjectMapper 如何忽略字段大小写
2021/06/29 Java/Android