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 相关文章推荐
JavaScript 滚轮事件使用说明
Mar 07 Javascript
cument.execCommand()用法深入理解
Dec 04 Javascript
基于JavaScript实现 获取鼠标点击位置坐标的方法
Apr 12 Javascript
JavaScript定义类的几种方式总结
Jan 06 Javascript
javascript中Array数组的迭代方法实例分析
Feb 04 Javascript
jQuery实现点击图片翻页展示效果的方法
Feb 16 Javascript
Javascript中With语句用法实例
May 14 Javascript
JavaScript操作class和style样式代码详解
Feb 13 Javascript
JS针对浏览器窗口关闭事件的监听方法集锦
Jun 24 Javascript
dts文件中删除一个node或属性的操作方法
Aug 05 Javascript
Vue组件内部实现一个双向数据绑定的实例代码
Apr 04 Javascript
vue脚手架项目创建步骤详解
Mar 02 Vue.js
微信小程序 实现动态显示和隐藏某个控件
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
MOTOROLA 摩托罗拉 MODEL 66-XI五灯中波收音机
2021/03/02 无线电
php !function_exists(&quot;T7FC56270E7A70FA81A5935B72EACBE29&quot;))代码解密
2011/01/07 PHP
解析Win7 XAMPP apache无法启动的问题
2013/06/26 PHP
php实现在限定区域里自动调整字体大小的类实例
2015/04/02 PHP
PHP实现的链式队列结构示例
2017/09/15 PHP
PHP实现链式操作的三种方法详解
2017/11/16 PHP
javascript 树控件 比较好用
2009/06/11 Javascript
jquery给图片添加鼠标经过时的边框效果
2013/11/12 Javascript
js this函数调用无需再次抓获id,name或标签名
2014/03/03 Javascript
jquery常用方法及使用示例汇总
2014/11/08 Javascript
javascript数据结构之二叉搜索树实现方法
2015/11/25 Javascript
AngularJS手动表单验证
2016/02/01 Javascript
jquery使用Cookie和JSON记录用户最近浏览历史
2016/04/19 Javascript
jQuery四种选择器使用及示例
2016/06/05 Javascript
node.js学习之事件模块Events的使用示例
2017/09/28 Javascript
利用Angular2 + Ionic3开发IOS应用实例教程
2018/01/15 Javascript
如何用webpack4.0撸单页/多页脚手架 (jquery, react, vue, typescript)
2019/06/18 jQuery
关于layui toolbar和template的结合使用方法
2019/09/19 Javascript
javascript实现倒计时效果
2020/02/17 Javascript
Python中的random()方法的使用介绍
2015/05/15 Python
Python实现邮件的批量发送的示例代码
2018/01/23 Python
Python获取CPU、内存使用率以及网络使用状态代码
2018/02/08 Python
pandas 数据归一化以及行删除例程的方法
2018/11/10 Python
用Python读取几十万行文本数据
2018/12/24 Python
Python模拟百度自动输入搜索功能的实例
2019/02/14 Python
python开发之anaconda以及win7下安装gensim的方法
2019/07/05 Python
Python中正则表达式对单个字符,多个字符和匹配边界等使用
2021/01/27 Python
css3设置box-pack和box-align让div里面的元素垂直居中
2014/09/01 HTML / CSS
Dockers美国官方网站:卡其裤、男士服装、鞋及配件
2016/11/22 全球购物
基本款天堂:Everlane
2017/05/13 全球购物
毕业学生推荐信
2013/12/01 职场文书
2014年单位植树节活动方案
2014/03/23 职场文书
中国梦我的梦演讲稿
2014/04/23 职场文书
应急管理培训方案
2014/06/12 职场文书
职业道德模范事迹材料
2014/08/24 职场文书
小公司融资,商业计划书的8切记
2019/07/15 职场文书