Vue.js 2.0 移动端拍照压缩图片预览及上传实例


Posted in Javascript onApril 27, 2017

在学习和使用Vue.js 2.0 的过程中遇到不少不一样的地方,本来移动端开发H5应用,准备将mui框架和Vue.js+vue-router+vuex 全家桶结合起来使用,但是在拍照上传的实现过程中遇到了无法调用plus的H5+接口的问题,所以最后拍照上传功能还是使用input file方式里解决的。但是内心还是不甘心的,由于项目进度推进,迭代版本,所以不得不放弃,后续可能我将此功能使用调用H5+接口实现。

首先我来讲我实现这个拍照预览压缩上传的思路,准确的说应该是拍照或选择图片压缩之后预览及上传这个流程。每次拍照或选择一张图片-然后压缩图片-预览上传。上传图片压缩插件是localResizeIMG , 这个插件的使用说明可以去看wiki,基本原理是通过canvas渲染图片,再通过 toDataURL 方法压缩保存为base64字符串(能够编译为jpg格式的图片),压缩效果很不错,ios拍照2MB多压缩下来大概是60-80kb左右,失真不是太严重,但是对我的项目来说图片清晰可见就好,我贴的代码里面也会有使用的演示。

<template>
 <h5 class="content-header">图片列表</h5>
 <div class="image-list">
 <div class="list-default-img" v-show="isPhoto" @click.stop="addPic">
  <img src="./images/icon_photo.png" />
  <span>请选择或者拍照上传照片</span>
  <input type="file" accept="image/jpeg,image/jpg,image/png" capture="camera" @change="onFileChange" style="display: none;">
 </div>
 <ul class="list-ul" v-show="!isPhoto">
  <li class="list-li" v-for="(iu, index) in imgUrls">
  <a class="list-link" @click='previewImage(iu)'>
   <img :src="iu">
  </a>
  <span class="list-img-close" @click='delImage(index)'></span>
  </li>
  <li class="list-li-add">
  <span class="add-img" @click.stop="addPic"></span>
  </li>
 </ul>
 </div>
  <div class="add-preview" v-show="isPreview" @click="closePreview">
 <img :src="previewImg">
 </div>
 
</template>

<script>

 export default {
 data: function () {
  return {
  imgUrls: [],
  urlArr: [],
  isPhoto: true,
  btnTitle: '',
  isModify: false,
  previewImg:'',
  isPreview: false
  }
 },
 watch: {
  imgUrls: 'toggleAddPic'
 },
 methods: {
  toggleAddPic: function() {
  let vm = this;
  if(vm.imgUrls.length >= 1) {
   vm.isPhoto = false;
  } else {
   vm.isPhoto = true;
  }
  },
  addPic: function(e) {
  let vm = this;
  $('input[type=file]').trigger('click');
  return false;
  },
  onFileChange: function(e) {
  var files = e.target.files || e.dataTransfer.files;
  if(!files.length) return;
  this.createImage(files, e);
  },
  createImage: function(file, e) {
  let vm = this;
  lrz(file[0], { width: 480 }).then(function(rst) {
   vm.imgUrls.push(rst.base64);
   return rst;
  }).always(function() {
  // 清空文件上传控件的值
  e.target.value = null;
  });
  },
  delImage: function(index) {
  let vm = this;
  let btnArray = ['取消', '确定'];
  mui.confirm('确定删除该图片?','提示', btnArray, function(e) {
   if (e.index == 1) {
   vm.imgUrls.splice(index, 1);
   }
  })

  },
  previewImage: function(url){
  let vm = this;
  vm.isPreview = true;
  vm.previewImg = url;
  },
  closePreview: function(){
  let vm = this;
  vm.isPreview = false;
  vm.previewImg = "";
  },
  saveImage: function(){
  let vm = this;
  let urlArr = [],
  imgUrls = this.imgUrls;

  for(let i = 0; i < imgUrls.length; i++) {
   if(imgUrls[i].indexOf('file') == -1) {
   urlArr.push(imgUrls[i].split(',')[1]);
   } else {
   urlArr.push(imgUrls[i]);
   }
  }

  //数据传输操作
  }
 }
 }
 
</script>

1.点击拍照或选择图片 addPic

在vue.js中出发拍照和选择图片是频繁操作行为,每次只能拍照或选择一张图片,可以拍多张上传,使用给click事件加上.stop的修饰符,.stop - 调用 event.stopPropagation() ,是为了停止冒泡。accept是为了规定通过文件上传来提交的文件的类型,capture是webApp中捕获到系统默认的设备,camera--照相机;camcorder--摄像机;microphone--录音。

Vue.js 2.0 移动端拍照压缩图片预览及上传实例   

在触发拍照行为的时候绑定change事件 onFileChange 获取file文件对象,然后调用lrz方法压缩图片,在imgUrls数组中添加基于base64格式的图片。

lrz(file[0], { width: 480 }).then(function(rst) {
   vm.imgUrls.push(rst.base64);
   return rst;
  }).always(function() {
  // 清空文件上传控件的值
  e.target.value = null;
  });
lrz(file, [options]);

file: 通过 input:file 得到的文件,或者直接传入图片路径。

[options] :这个参数允许忽略。

  1.     width {Number} 图片最大不超过的宽度,默认为原图宽度,高度不设时会适应宽度;
  2.     height {Number} 同上;
  3.     quality {Number} 图片压缩质量,取值 0 - 1,默认为0.7;
  4.     fieldName {String} 后端接收的字段名,默认:file;

返回结果是一个promise对象,有then()、catch()、always三个方法。

then(rst):

  1.     rst.formData 后端可处理的数据;
  2.     rst.file 压缩后的file对象(默认已经丢在rst.formData有一份了),需要注意的是如果压缩率太低的话,这个会是原始的file对象;
  3.     rst.fileLen 生成后的图片的大小,后端可以通过此值来校验是否传输完整;
  4.     rst.base64 生成后的图片base64,后端可以处理此字符串为图片,也直接用于img.src = base64;
  5.     rst.base64Len 生成后的base64的大小,后端可以通过此值来校验是否传输完整 (如果采用base64上传方式);
  6.     rst.origin 也就是原始的file对象,里面存了一些原始文件的信息,例如大小,日期等;

catch(err) 、always() 。

注意:由于我们可能持续点击拍照上传图片,使用在alway回调函数里面必须清空上传控制的值。

// 清空文件上传控件的值
 e.target.value = null;

2. 点击拍第一张照片和显示预览以及继续拍照的DOM显示  isPhoto

   默认 isPhoto 为true,隐藏继续拍照的DOM显示,toggleAddPic 监听当前选中imgUrls数组长度,转换 isPhoto 的布尔值若有一张及以上的图片设置 isPhoto 为false, 则隐藏点击拍第一张照片DOM,显示继续拍照的DOM;若没有图片,则隐藏继续拍照的DOM,显示击拍第一张照片DOM。

3. 删除已选择的压缩图片 delImage

根据数组对应的下标,在imgUrls中删除对应的图片数据。

delImage: function(index) {
  let vm = this;
  let btnArray = ['取消', '确定'];
  mui.confirm('确定删除该图片?','提示', btnArray, function(e) {
   if (e.index == 1) {
   vm.imgUrls.splice(index, 1);
   }
  })

  }

Vue.js 2.0 移动端拍照压缩图片预览及上传实例

4. 大图预览已经被压缩的图片及关闭大图预览  isPreview previewImage closePreview

在这里大图预览就是将base64格式的图片直接放进预览DOM的img src中放大展示,点击图片自身关闭预览,清空img src资源。

Vue.js 2.0 移动端拍照压缩图片预览及上传实例

5. 对base64图片传输前的处理 saveImage

saveImage: function(){
  let vm = this;
  let urlArr = [],
  imgUrls = this.imgUrls;

  for(let i = 0; i < imgUrls.length; i++) {
   if(imgUrls[i].indexOf('file') == -1) {
   urlArr.push(imgUrls[i].split(',')[1]);
   } else {
   urlArr.push(imgUrls[i]);
   }
  }

  //数据传输操作
  }

我压缩成base64字符串是“data:image/jpeg;base64,~~”的字符串,为了后端好处理,我这里为了将编辑时候后台返回的图片url区别开来,将“data:image/jpeg;base64,"截取掉,只传递给后端逗号后面的base64字符串。

注意:后端接收到我传递的base64字符串数组的时候,发现字符经如果被urlencode后标准的base64中的/、 +会被转成%xx,后端在将base64字符串处理成图片时,需要将特殊字符过滤掉。

[HttpPost]
    public ActionResult MUploadImgBase64Str(string base64str)
    {
      try
      {
        var imgData = base64str;
        //过滤特殊字符即可  
        string dummyData = imgData.Trim().Replace("%", "").Replace(",", "").Replace(" ", "+");
        if (dummyData.Length % 4 > 0)
        {
          dummyData = dummyData.PadRight(dummyData.Length + 4 - dummyData.Length % 4, '=');
        }
        byte[] byteArray = Convert.FromBase64String(dummyData);
        using (System.IO.MemoryStream ms = new System.IO.MemoryStream(byteArray))
        {
          var img = System.Drawing.Image.FromStream(ms);

          var path = "~/Content/UploadFiles/mobile/";
          var uploadpath = Server.MapPath(path);
          if (!Directory.Exists(uploadpath))
          {
            Directory.CreateDirectory(uploadpath);
          }
          var saveName = uploadpath + “stoneniqiu” + ".jpg";
          img.Save(saveName);
          return Json(saveName);
        }
      }
      catch (Exception e)
      {
        return Json(e.Message);

      }
    }

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
Jquery+WebService 校验账号是否已被注册的代码
Jul 12 Javascript
基于jQuery的输入框在光标位置插入内容, 并选中
Oct 29 Javascript
鼠标滚轮控制网页横向移动实现思路
Mar 22 Javascript
鼠标拖动实现DIV排序示例代码
Oct 14 Javascript
邮箱下拉自动填充选择示例代码附图
Apr 03 Javascript
Bootstrap carousel轮转图的使用实例详解
May 17 Javascript
去除html代码里面的script正则方法
May 19 Javascript
JS清除字符串中重复值的实现方法
Aug 03 Javascript
jQuery插件FusionCharts绘制的2D帕累托图效果示例【附demo源码】
Mar 28 jQuery
gulp解决跨域的配置文件问题
Jun 08 Javascript
vue axios 给生产环境和发布环境配置不同的接口地址(推荐)
May 08 Javascript
jquery实现点击左右按钮切换图片
Jan 27 jQuery
微信小程序 实现动态显示和隐藏某个控件
Apr 27 #Javascript
JS ES6中setTimeout函数的执行上下文示例
Apr 27 #Javascript
Vue实现购物车功能
Apr 27 #Javascript
js轮播图透明度切换(带上下页和底部圆点切换)
Apr 27 #Javascript
Angular.js中定时器循环的3种方法总结
Apr 27 #Javascript
浅谈js使用in和hasOwnProperty获取对象属性的区别
Apr 27 #Javascript
微信小程序 wx:for的使用实例详解
Apr 27 #Javascript
You might like
PHP的开合式多级菜单程序
2006/10/09 PHP
php中实现进程锁与多进程的方法
2016/09/18 PHP
Yii 框架控制器创建使用及控制器响应操作示例
2019/10/14 PHP
JavaScript中的History历史对象
2008/01/16 Javascript
TextArea 控件的最大长度问题(js json)
2009/12/16 Javascript
JavaScript 学习笔记(十一)
2010/01/19 Javascript
Jquery Validation插件防止重复提交表单的解决方法
2010/03/05 Javascript
cookie 最近浏览记录(中文escape转码)具体实现
2013/06/08 Javascript
js焦点文字滚动效果代码分享
2015/08/25 Javascript
jquery实现简单的全选和反选功能
2016/01/02 Javascript
javascript鼠标右键菜单自定义效果
2020/12/08 Javascript
javascript实现起伏的水波背景效果
2016/05/16 Javascript
js canvas实现擦除动画
2016/07/16 Javascript
JavaScript 随机验证码的生成实例代码
2016/09/22 Javascript
浅谈javascript中的事件冒泡和事件捕获
2016/12/28 Javascript
JS+html5 canvas实现的简单绘制折线图效果示例
2017/03/13 Javascript
js实现登录与注册界面
2017/11/01 Javascript
nodejs项目windows下开机自启动的方法
2017/11/22 NodeJs
微信小程序实现单选选项卡切换效果
2020/06/19 Javascript
解决layui中onchange失效以及form动态渲染失效的问题
2019/09/27 Javascript
Vue.js暴露方法给WebView的使用操作
2020/09/07 Javascript
JavaScript Blob对象原理及用法详解
2020/10/14 Javascript
python随机在一张图像上截取任意大小图片的方法
2019/01/24 Python
python 利用turtle库绘制笑脸和哭脸的例子
2019/11/23 Python
Python处理mysql特殊字符的问题
2020/03/02 Python
基于python 等频分箱qcut问题的解决
2020/03/03 Python
python 函数嵌套及多函数共同运行知识点讲解
2020/03/03 Python
pandas数据分组groupby()和统计函数agg()的使用
2021/03/04 Python
如何打印出当前源文件的文件名以及源文件的当前行号
2015/04/05 面试题
农救科工作职责
2013/11/27 职场文书
法学专业本科生自荐信范文
2013/12/17 职场文书
员工培训协议书
2014/09/15 职场文书
2015年电信员工工作总结
2015/05/26 职场文书
学习型家庭事迹材料(2016精选版)
2016/02/29 职场文书
送给自己的励志语句:要安静的优秀,悄无声息的坚强
2019/11/26 职场文书
详解运行Python的神器Jupyter Notebook
2021/06/03 Python