利用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 相关文章推荐
纯CSS实现预加载动画效果
Sep 06 HTML / CSS
一款纯css3实现的非常实用的鼠标悬停特效演示
Nov 05 HTML / CSS
css3实现的多级渐变下拉菜单导航效果代码
Aug 31 HTML / CSS
纯HTML5+CSS3制作图片旋转
Jan 12 HTML / CSS
CSS3 文字动画效果
Nov 12 HTML / CSS
CSS3中的弹性布局em运用入门详解 1em等于多少像素
Feb 08 HTML / CSS
HTML5拖放API实现拖放排序的实例代码
May 11 HTML / CSS
详解HTML5 Canvas标签及基本使用
Jan 10 HTML / CSS
uniapp+Html5端实现PC端适配
Jul 15 HTML / CSS
HTML5逐步分析实现拖放功能的方法
Sep 30 HTML / CSS
Html5通过数据流方式播放视频的实现
Apr 27 HTML / CSS
纯CSS实现一个简单步骤条的示例代码
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数据库调用类调用实例(详细注释)
2012/07/12 PHP
使用PHP下载CSS文件中的图片的代码
2013/09/24 PHP
jQuery对象[0]是什么含义?
2010/07/31 Javascript
js网页中的(运行代码)功能实现思路
2013/02/04 Javascript
把字符串按照特定的字母顺序进行排序的js代码
2014/01/28 Javascript
关于JavaScript的变量的数据类型的判断方法
2015/08/14 Javascript
原生JS和jQuery版实现文件上传功能
2016/04/18 Javascript
JavaScript_ECMA5数组新特性详解
2016/06/12 Javascript
一个简单的JavaScript Map实例(分享)
2016/08/03 Javascript
jquery的checkbox,radio,select等方法小结
2016/08/30 Javascript
React中使用async validator进行表单验证的实例代码
2018/08/17 Javascript
微信小程序实现简单跑马灯效果
2020/05/26 Javascript
小程序二次贝塞尔曲线实现购物车商品曲线飞入效果
2019/01/07 Javascript
Vue使用预渲染代替SSR的方法
2020/07/02 Javascript
Nest.js 授权验证的方法示例
2021/02/22 Javascript
Python交换变量
2008/09/06 Python
400多行Python代码实现了一个FTP服务器
2012/05/10 Python
Python读取指定目录下指定后缀文件并保存为docx
2017/04/23 Python
python itchat实现微信自动回复的示例代码
2017/08/14 Python
基于python中pygame模块的Linux下安装过程(详解)
2017/11/09 Python
python提取图像的名字*.jpg到txt文本的方法
2018/05/10 Python
使用Python制作自动推送微信消息提醒的备忘录功能
2018/09/06 Python
Python 3.6打包成EXE可执行程序的实现
2019/10/18 Python
利用OpenCV和Python实现查找图片差异
2019/12/19 Python
使用css3匹配手机屏幕横竖状态
2014/01/27 HTML / CSS
linux下进程间通信的方式
2013/01/23 面试题
自荐信不宜过于夸大
2013/11/06 职场文书
公司庆典邀请函范文
2014/01/13 职场文书
公司业务员岗位职责
2014/03/18 职场文书
海飞丝广告词
2014/03/20 职场文书
小学运动会演讲稿
2014/08/25 职场文书
年检委托书
2014/08/30 职场文书
爱心募捐感谢信
2015/01/22 职场文书
今日说法观后感
2015/06/08 职场文书
开票证明
2015/06/23 职场文书
导游词之井冈山
2019/11/20 职场文书