利用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 相关文章推荐
推荐10个CSS3 制作的创意下拉菜单效果
Feb 11 HTML / CSS
css3弹性盒模型(Flexbox)详细介绍
Oct 08 HTML / CSS
使用css3实现的tab选项卡代码分享
Dec 09 HTML / CSS
利用CSS3的transition属性实现滑动效果
Aug 05 HTML / CSS
使用CSS3 制作一个material-design 风格登录界面实例
Dec 12 HTML / CSS
css3实现3D文本悬停改变效果的示例代码
Jan 16 HTML / CSS
html5触摸事件判断滑动方向的实现
Jun 05 HTML / CSS
HTML 5 input placeholder 属性如何完美兼任ie
May 12 HTML / CSS
浅谈HTML5新增和废弃的标签
Apr 28 HTML / CSS
五分钟学会HTML5的WebSocket协议
Nov 22 HTML / CSS
萌新的HTML5 入门指南
Nov 06 HTML / CSS
CSS中calc(100%-100px)不加空格不生效
May 07 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
自动把纯文本转换成Web页面的php代码
2009/08/27 PHP
PHP 时间转换Unix时间戳代码
2010/01/22 PHP
PHP输出XML到页面的3种方法详解
2013/06/06 PHP
使用dump函数,给php加断点测试
2013/06/25 PHP
PHP中$_SERVER的详细参数与说明介绍
2013/10/26 PHP
ThinkPHP采用原生query实现关联查询left join实例
2014/12/02 PHP
php命令行(cli)模式下报require 加载路径错误的解决方法
2015/11/23 PHP
中高级PHP程序员应该掌握哪些技术?
2016/09/23 PHP
php curl上传、下载、https登陆实现代码
2017/07/23 PHP
PNGHandler-借助JS让PNG图在IE下实现透明(包括背景图)
2007/08/31 Javascript
用户注册常用javascript代码
2009/08/29 Javascript
IE 条件注释详解总结(附实例代码)
2009/08/29 Javascript
js window.onload 加载多个函数的方法
2009/11/02 Javascript
Javascript中的delete介绍
2012/09/02 Javascript
Jquery validation remote 验证的缓存问题解决方法
2014/03/25 Javascript
javascript 判断两个日期之差的示例代码
2015/09/05 Javascript
手机端实现Bootstrap简单图片轮播效果
2016/10/13 Javascript
ng-options和ng-checked在表单中的高级运用(推荐)
2017/01/21 Javascript
webpack 2.x配置reactjs基本开发环境详解
2017/08/08 Javascript
JavaScript实现图片切换效果
2017/08/12 Javascript
webpack学习笔记之优化缓存、合并、懒加载
2017/08/24 Javascript
使用webpack3.0配置webpack-dev-server教程
2018/05/29 Javascript
vue中如何实现pdf文件预览的方法
2018/07/12 Javascript
JavaScript作用域链实例详解
2019/01/21 Javascript
使用异步controller与jQuery实现卷帘式分页
2019/06/18 jQuery
vue 实现v-for循环回来的数据动态绑定id
2019/11/07 Javascript
Python实例之wxpython中Frame使用方法
2014/06/09 Python
Python插件virtualenv搭建虚拟环境
2017/11/20 Python
浅谈python常用程序算法
2019/03/22 Python
python实现从尾到头打印单链表操作示例
2020/02/22 Python
英国虚拟主机服务商:eUKhost
2016/08/16 全球购物
Notino芬兰:购买香水和化妆品
2019/04/15 全球购物
网络优化专员求职信
2014/05/04 职场文书
优秀应届毕业生自荐书
2014/06/29 职场文书
教师四风问题对照检查材料
2014/09/26 职场文书
绿里奇迹观后感
2015/06/15 职场文书