工作中遇到的问题记录:vue开发头像上传组件,后端提供接口,需求为可相册上传,可相机拍摄上传,文件大小限制为2M

需求点分析

  1. 移动端调用相册/摄像头实现拍照
  2. 图片压缩,当前高像素的相机拍出来的图片都有个4-5m大小,超出了限定的2m
  3. 图片上传

第一点:移动端调用相册/摄像头实现拍照

这就比较简单了,H5的标签足已经能满足我们的要求了,我们要做的只是让它更美观一点

        <div class="yb-headSet-but">
<div class="btn-box">
<input type="file" accept="image/*" @change="uploadImg" class="fromAlbum"> //设置input opacity为0,定位覆盖a标签,a标签设置为设计稿的按钮样式
<a href="" class="weui-but">从相册选一张</a>
</div>
<div class="btn-box">
<a href="" class="weui-but">拍一张照片</a>
<input type="file" accept="image/*" @change="uploadImg" capture="camera" class="fromCamera"> //两者区别:capture="camera" 可直接调用摄像头
</div>
</div>

第二点:图片压缩

利用的是canvas的drawImage 和 toDataURL的api,具体代码是复制的 https://www.jianshu.com/p/4587312d2f44

修改了几个大佬笔误的地方

imgResize(file, callback) {
var _this = this;
var fileReader = new FileReader();
fileReader.onload = function() {
var IMG = new Image();
IMG.src = this.result;
IMG.onload = function() {
var w = this.naturalWidth,
h = this.naturalHeight,
resizeW = 0,
resizeH = 0;
// maxSize 是压缩的设置,设置图片的最大宽度和最大高度,等比缩放,level是报错的质量,数值越小质量越低
var maxSize = {
width: 500,
height: 500,
level: 0.6
};
if (w > maxSize.width || h > maxSize.height) {
var multiple = Math.max(w / maxSize.width, h / maxSize.height);
resizeW = w / multiple;
resizeH = h / multiple;
} else {
// 如果图片尺寸小于最大限制,则不压缩直接上传
return callback(file);
}
var canvas = document.createElement("canvas");
var ctx = canvas.getContext("2d");
if (window.navigator.userAgent.indexOf("iPhone") > 0) {
canvas.width = resizeH;
canvas.height = resizeW;
ctx.rotate(90 * Math.PI / 180);
ctx.drawImage(IMG, 0, -resizeH, resizeW, resizeH);
} else {
canvas.width = resizeW;
canvas.height = resizeH;
ctx.drawImage(IMG, 0, 0, resizeW, resizeH);
}
var base64 = canvas.toDataURL("image/" + _this.fileType, maxSize.level);
_this.convertBlob(window.atob(base64.split(",")[1]), callback);
};
};
fileReader.readAsDataURL(file);
},

图片上传

使用canvas.toDataURL获得的是base64格式,上传时需要转换成Blob二进制格式,使用formData提交

//Blob对象生成
convertBlob(base64, callback) {
var buffer = new ArrayBuffer(base64.length);
var ubuffer = new Uint8Array(buffer);
for (var i = 0; i < base64.length; i++) {
ubuffer[i] = base64.charCodeAt(i);
}
var blob;
try {
blob = new Blob([buffer], { type: "image/" + this.fileType });
} catch (e) {
window.BlobBuilder =
window.BlobBuilder ||
window.WebKitBlobBuilder ||
window.MozBlobBuilder ||
window.MSBlobBuilder;
if (e.name === "TypeError" && window.BlobBuilder) {
var blobBuilder = new BlobBuilder();
blobBuilder.append(buffer);
blob = blobBuilder.getBlob("image/" + this.fileType);
}
}
callback(blob);
},
//提交函数
postImg(file) {
let image = new FormData();
image.append("file", file, this.random_string(12) + '.' + this.fileType);
console.log(image);
setHead(image).then(res => {
this.$indicator.close();
if (res.Success) {
let newPath = res.photoPath +'?t=' + this.random_string(12);
this.$store.commit("UPDATE_BASEINFO", { photoPath: newPath });
this.picValue = newPath;
this.$messagebox.alert(res.Message);
} else {
this.$messagebox.alert(res.Message);
}
});
},

注意注意 这里有个巨大的坑。。。。。

这个坑是这样的,我使用压缩过的Blob格式提交后台时,返回提示文件格式不正确,不属于jpeg、jpg等,这我就纳闷了,我上传的formData里面写明了文件格式了啊;

后来经过简书大佬 一斤代码 老大的帮助,老大测试后说出了问题所在:那就是 FormData.append('file', file),如果只有两个参数,(第三个参数是文件名)的情况下,默认fileName=“blob”,这样上次至后台,如果后台未做其他处理的话,就会出现报错的情况,后来就添加了随机文件名,完整代码如下

<template>
<div class="yb-page">
<div class="yb-page-inner">
<div class="yb-headSetting">
<div class="yb-headSet-img">
<img :src="picValue" alt="" onerror="this.src='https://ixxxxxx.com/we/tmpls/t9000/img/account/blackHead@2x.png'">
</div>
<div class="yb-headSet-but">
<div class="btn-box">
<input type="file" ref="avatarInput" accept="image/*" @change="uploadImg" class="fromAlbum">
<a href="" class="weui-but">从相册选一张</a>
</div>
<div class="btn-box">
<a href="" class="weui-but">拍一张照片</a>
<input type="file" accept="image/*" @change="uploadImg" capture="camera" class="fromCamera">
</div>
</div>
</div>
</div>
</div>
</template>
<script>
import { setHead } from "@/api/user";
export default {
data() {
return {
picValue: this.$store.getters.baseInfo.photoPath,
file: null,
fileType: ''
};
},
methods: {
uploadImg(e) {
if (e.target.value == "") {
return;
}
this.$indicator.open();
let files = e.target.files || e.dataTransfer.files;
if (!files.length) return;
this.file = files[0];
this.fileType = this.file.type.split('/')[1];
this.imgResize(files[0], this.postImg);
},
postImg(file) {
let image = new FormData();
image.append("file", file, this.random_string(12) + '.' + this.fileType);
console.log(image);
setHead(image).then(res => {
this.$indicator.close();
if (res.Success) {
let newPath = res.photoPath +'?t=' + this.random_string(12);
this.$store.commit("UPDATE_BASEINFO", { photoPath: newPath });
this.picValue = newPath;
this.$messagebox.alert(res.Message);
} else {
this.$messagebox.alert(res.Message);
}
});
},
imgResize(file, callback) {
var _this = this;
var fileReader = new FileReader();
fileReader.onload = function() {
var IMG = new Image();
IMG.src = this.result;
IMG.onload = function() {
var w = this.naturalWidth,
h = this.naturalHeight,
resizeW = 0,
resizeH = 0;
// maxSize 是压缩的设置,设置图片的最大宽度和最大高度,等比缩放,level是报错的质量,数值越小质量越低
var maxSize = {
width: 500,
height: 500,
level: 0.6
};
if (w > maxSize.width || h > maxSize.height) {
var multiple = Math.max(w / maxSize.width, h / maxSize.height);
resizeW = w / multiple;
resizeH = h / multiple;
} else {
// 如果图片尺寸小于最大限制,则不压缩直接上传
return callback(file);
}
var canvas = document.createElement("canvas");
var ctx = canvas.getContext("2d");
if (window.navigator.userAgent.indexOf("iPhone") > 0) {
canvas.width = resizeH;
canvas.height = resizeW;
ctx.rotate(90 * Math.PI / 180);
ctx.drawImage(IMG, 0, -resizeH, resizeW, resizeH);
} else {
canvas.width = resizeW;
canvas.height = resizeH;
ctx.drawImage(IMG, 0, 0, resizeW, resizeH);
}
var base64 = canvas.toDataURL("image/" + _this.fileType, maxSize.level);
_this.convertBlob(window.atob(base64.split(",")[1]), callback);
};
};
fileReader.readAsDataURL(file);
},
convertBlob(base64, callback) {
var buffer = new ArrayBuffer(base64.length);
var ubuffer = new Uint8Array(buffer);
for (var i = 0; i < base64.length; i++) {
ubuffer[i] = base64.charCodeAt(i);
}
var blob;
try {
blob = new Blob([buffer], { type: "image/" + this.fileType });
} catch (e) {
window.BlobBuilder =
window.BlobBuilder ||
window.WebKitBlobBuilder ||
window.MozBlobBuilder ||
window.MSBlobBuilder;
if (e.name === "TypeError" && window.BlobBuilder) {
var blobBuilder = new BlobBuilder();
blobBuilder.append(buffer);
blob = blobBuilder.getBlob("image/" + this.fileType);
}
}
callback(blob);
},
random_string(len) {
len = len || 32;
var chars = "ABCDEFGHJKMNPQRSTWXYZabcdefhijkmnprstwxyz2345678";
var maxPos = chars.length;
var pwd = "";
for (var i = 0; i < len; i++) {
pwd += chars.charAt(Math.floor(Math.random() * maxPos));
}
return pwd;
}
}
};
</script>

vue头像上传与文件压缩的更多相关文章

  1. springboot整合vue实现上传下载文件

    https://blog.csdn.net/yhhyhhyhhyhh/article/details/89888953 文章目录 springboot整合vue实现上传下载文件 1上传下载文件api文 ...

  2. vue头像上传

    项目四知识点 默认头像 选择头像 <template> <div class="adatar"> <img :src="adatar?ada ...

  3. springboot 头像上传 文件流保存 文件流返回浏览器查看 区分操作系统 windows 7 or linux

    //我的会员中心 头像上传接口 /*windows 调试*/ @Value("${appImg.location}") private String winPathPic; /*l ...

  4. vue+element+upload实现头像上传

    后台 @RequestMapping("/up") public JSONObject up(@RequestParam("picFile") Multipar ...

  5. vue上传大文件控件

    文件上传是 Web 开发肯定会碰到的问题,而文件夹上传则更加难缠.网上关于文件夹上传的资料多集中在前端,缺少对于后端的关注,然后讲某个后端框架文件上传的文章又不会涉及文件夹.今天研究了一下这个问题,在 ...

  6. AntDesign VUE:上传组件图片/视频宽高、文件大小、image/video/pdf文件类型等限制(Promise、Boolean)

    文件大小限制 - Promise checkFileSize(file, rules) { return new Promise((resolve, reject) => { file.size ...

  7. vue上传大文件的解决方案

    众所皆知,web上传大文件,一直是一个痛.上传文件大小限制,页面响应时间超时.这些都是web开发所必须直面的. 本文给出的解决方案是:前端实现数据流分片长传,后面接收完毕后合并文件的思路. 实现文件夹 ...

  8. 强大的flash头像上传插件(支持旋转、拖拽、剪裁、生成缩略图等)

    今天介绍的这款flash上传头像功能非常强大,支持php,asp,jsp,asp.net 调用 头像剪裁,预览组件插件. 本组件需要安装Flash Player后才可使用,请从http://dl.pc ...

  9. 【Bootstrap-插件使用】Jcrop+fileinput组合实现头像上传功能

    作者:Dreawer链接:https://zhuanlan.zhihu.com/p/24465742来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商业转载请注明出处. 作者:梦游的龙猫(转 ...

随机推荐

  1. 理解一条语句:SELECT difference(sum("value")) FROM "mq_enqueue" WHERE "channel" =~ /ActiveMQ_TEST/ AND $timeFilter GROUP BY time($interval)

    最近使用grafana在查询InfluxDB中,用到了这一条语句 SELECT difference(sum("value")) FROM "mq_enqueue&quo ...

  2. 观察者模式和java委托

    观察者模式与java委托 所谓观察者模式,指的某个状态信息的改变,会影响其他一系列的操作,这时就可以将这些操作抽象化,同时创建一个类统一的管理和执行这些操作.把这些抽象出来的操作称为观察者类,而管理这 ...

  3. csu 1769(数学)

    1769: 想打架吗?算我一个!所有人,都过来!(3) Time Limit: 2 Sec  Memory Limit: 128 MBSubmit: 262  Solved: 76[Submit][S ...

  4. linux中使用vim编译C++程序

    Vi三种模式详解 命令行模式 (command mode/一般模式) 任何时候,不管用户处于何种模式,只要按一下“ESC”键,即可使Vi进入命令行模式:我们在shell环境(提示符为$)下输入启动Vi ...

  5. jmeter------线程组(默认)

    Jmeter中的采样器必须要基于线程组. 一.添加线程组 在测试计划上右键,然后选择,如下图: 二.线程组界面 三.线程组界面配置说明 1.名称:线程组自定义名称: 2.注释:添加的一些备注说明信息, ...

  6. day4 作业计算器

    作业:计算器开发 (1)实现加减乘除及拓号优先级解析: (2)用户输入 1 - 2 * ( (60-30 +(-40/5) * (-9-2*5/-3 + 7 /3*99/4*2998 +10 * 56 ...

  7. electron写俄罗斯方块游戏(Tetris)

    背景 在折腾ES6,突然想起大学时用c语言写过俄罗斯方块,本项目中主要是利用ES6的Class特性进行面向对象编程.项目采用node.js v6.2.0 + electron v1.1.0 进行桌面开 ...

  8. Python并发编程-进程的几个方法

    join()方法 from multiprocessing import Process import time def func(arg1,arg2): print('*'*arg1) time.s ...

  9. Django+Nginx+uwsgi搭建自己的博客(二)

    在上一篇博客中,为大家介绍了Django的一些基本概念以及建立了第一个App——Users,并且在数据库中建立了对应的表. 在这篇博客中,将为大家继续介绍数据库模型的定义和相关操作,以及Users A ...

  10. Spring 常用的注解

    目录 Spring 常用的注解 前言 SpringMVC配置 web配置 @ComponentScan @PropertySource @PropertySources @Value @Control ...