代码还在优化,后续会进行更新,该组件引用了element-ui的弹出框样式,使用axios库进行ajax请求,向后台发送的参数是formData对象,如果需要的base64可不转formData对象,直接将压缩过后的base64发给后台
该组件上传图片的大致思路:
- 利用H5的FileReader对象将上传的图片转成base64格式
- 利用canvas的drawImage方法重绘上传的图片(注:drawimage是一个异步方法,需要在图片读取成功后在进行重绘,否则可能会压缩不成功)
- 在利用canvas的toDataURL方法将图片压缩
- 在将压缩后的base64编码图片转成blob对象
- 创建一个formData对象,将blob对象append进去
- 图片上传时需设置请求为
{ "Content-Type": "multipart/form-data" }
- 将formData对象上传到后台
html部分
<div class="upload">
<!-- 图片展示 -->
<div class="cha" v-show="(imgUrl)">
<!-- 删除icon -->
<div class="del"><i class="el-icon-delete" @click="delImg"></i></div>
<img :src="imgUrl">
<!-- 放大icon -->
<div class="layer"><i @click="isEnlargeImage = true" class="el-icon-view"></i></div>
</div>
<!-- 图片上传控件 -->
<div class="load" v-show="(!imgUrl)">
<input type="file" name="" id="form" @change="uploadIMG">
</div>
<!-- 图片预览弹框 -->
<el-dialog :visible.sync="isEnlargeImage" size="large" :append-to-body="true" top="8%" width="60%">
<img @click="isEnlargeImage = false" style="width:100%;" :src="imgUrl">
</el-dialog>
</div>
JS代码
<script>
export default {
props: ["uploadUrl"],
data() {
return {
picavalue: "",
imgUrl: null,
isEnlargeImage: false
};
},
methods: {
uploadIMG(e) {
let files = e.target.files || e.dataTransfer.files;
if (!files.length) return;
this.picavalue = files[0];
console.log(this.picavalue.size / 1024);
if (this.picavalue.size / 1024 > 5000) {
this.$message({
message: "图片过大不支持上传",
type: "warning"
});
} else {
this.imgPreview(this.picavalue);
}
},
//获取图片
imgPreview(file, callback) {
let self = this;
//判断支不支持FileReader
if (!file || !window.FileReader) return;
if (/^image/.test(file.type)) {
//创建一个reader
let reader = new FileReader();
//将图片转成base64格式
reader.readAsDataURL(file);
//读取成功后的回调
reader.onloadend = function() {
let result = this.result;
let img = new Image();
img.src = result;
console.log("********未压缩前的图片大小********");
console.log(result.length);
img.onload = function() {
let data = self.compress(img);
self.imgUrl = result;
let blob = self.dataURItoBlob(data);
console.log("*******base64转blob对象******");
console.log(blob);
var formData = new FormData();
formData.append("file", blob);
console.log("********将blob对象转成formData对象********");
console.log(formData.get("file"));
let config = {
headers: { "Content-Type": "multipart/form-data" }
};
// 发送请求;
self.axios.post(self.uploadUrl.url, formData, config).then(res => {
// console.log(res);
// console.log(res.data.data.resultftphost)
// console.log(res.data.data.resulturl)
// this.$emit('')
if (res.data.code == 200) {
self.$emit(
"getImgsrc",
res.data.data.resultftphost,
res.data.data.resulturl
);
} else {
self.$message({
message: "图片上传失败,请重试",
type: "warning"
});
}
});
};
};
}
},
// 压缩图片
compress(img) {
let canvas = document.createElement("canvas");
let ctx = canvas.getContext("2d");
let initSize = img.src.length;
let width = img.width;
let height = img.height;
canvas.width = width;
canvas.height = height;
// 铺底色
ctx.fillStyle = "#fff";
ctx.fillRect(0, 0, canvas.width, canvas.height);
ctx.drawImage(img, 0, 0, width, height);
//进行最小压缩
let ndata = canvas.toDataURL("image/jpeg", 0.1);
// console.log("*******压缩后的图片大小*******");
// console.log(ndata)
// console.log(ndata.length);
return ndata;
},
// base64转成bolb对象
dataURItoBlob(base64Data) {
var byteString;
if (base64Data.split(",")[0].indexOf("base64") >= 0)
byteString = atob(base64Data.split(",")[1]);
else byteString = unescape(base64Data.split(",")[1]);
var mimeString = base64Data
.split(",")[0]
.split(":")[1]
.split(";")[0];
var ia = new Uint8Array(byteString.length);
for (var i = 0; i < byteString.length; i++) {
ia[i] = byteString.charCodeAt(i);
}
return new Blob([ia], { type: mimeString });
},
//删除事件
delImg() {
this.imgUrl = null;
}
}
};
style 部分
.upload {
position: relative;
width: 200px;
height: 220px;
// 图片展示
.cha {
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 100%;
border-radius: 6px;
border: 1px dashed #cccccc;
overflow: hidden;
box-sizing: border-box;
.layer {
position: absolute;
top: 0;
left: 0;
z-index: 2;
width: 100%;
height: 200px;
background-color: rgba(0, 0, 0, 0.3);
text-align: center;
line-height: 200px;
opacity: 0;
color: #ffffff;
border: none;
}
.del {
position: absolute;
bottom: 0;
right: 0;
opacity: 0;
z-index: 2;
border: none;
}
img {
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 200px;
border: none;
}
}
//鼠标放上时显示删除和放大
.cha:hover .layer,
.cha:hover .del {
opacity: 1;
}
//点击上传
.load {
position: absolute;
top: 0;
left: 0;
width: 200px;
height: 200px;
border: 1px dashed #cccccc;
input {
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 100%;
opacity: 0;
}
}
.load::before {
content: "";
position: absolute;
top: 50%;
left: 50%;
transform: translateY(-50%);
width: 1px;
height: 80%;
border-right: 1px solid #cccccc;
}
.load:after {
content: "";
position: absolute;
top: 50%;
left: 50%;
transform: translateX(-50%);
width: 80%;
height: 1px;
border-top: 1px solid #cccccc;
}
}