今天的目标是封装一个组件,用来把图片上传到cos中。
组件要有两个功能
1. 展示图片(初始显示图片)
2. 修改图片
elementUI里upload组件已经是一个封装好了的组件我们可以对他进行二次封装即可
需求理解:
前端主动发起图片上传使用的是三方的腾讯云上传接口,前端得到一个已经上传完毕的图片地址,然后把这个地址当成一个接口字段 传给我们自己的后端服务
员工详情页
新建公共上传组件
我们的上传功能是基于element上传组件的二次开发,先准备好我们需要的elementUI上传组件,根据我们具体的业务需求选取一个合适的样例代码
在elementUI里复制src/components/UploadImg里代码 具体代码如下(子组件里)
<template>
<div>
<el-upload
class="avatar-uploader"
action="#"
:show-file-list="false"
:on-success="handleAvatarSuccess"
:before-upload="beforeAvatarUpload"
:http-request="upload"
>
<img v-if="imageUrl" :src="imageUrl" class="avatar">
<i v-else class="el-icon-plus avatar-uploader-icon" />
</el-upload>
</div>
</template>
<script>
export default {
data() {
return {
imageUrl: ''
}
},
methods: {
upload(file) {
console.log(file)
},
handleAvatarSuccess(res, file) {
this.imageUrl = URL.createObjectURL(file.raw)
},
beforeAvatarUpload(file) {
const isPNG = file.type === 'image/png'
const isLt2M = file.size / 1024 / 1024 < 2
if (!isPNG) {
this.$message.error('上传头像图片只能是 PNG 格式!')
}
if (!isLt2M) {
this.$message.error('上传头像图片大小不能超过 2MB!')
}
return isPNG && isLt2M
}
}
}
</script>
<style>
.avatar-uploader .el-upload {
border: 1px dashed #d9d9d9;
border-radius: 6px;
cursor: pointer;
position: relative;
overflow: hidden;
}
.avatar-uploader .el-upload:hover {
border-color: #409eff;
}
.avatar-uploader-icon {
font-size: 28px;
color: #8c939d;
width: 178px;
height: 178px;
line-height: 178px;
text-align: center;
}
.avatar {
width: 178px;
height: 178px;
display: block;
}
</style>
自定义上传配置
关键属性::http-request="upload" action="#"
使用自定义行为覆盖默认上传,注意一旦设置自定义上传行为之后,所有的上传操作都需要自己实现,比如数据处理,上传成功之后的后续操作,on-success钩子函数也不会继续触发
<template>
<div>
<!--
show-file-list: 是否显示上传的文件列表
action: '#' 用来指定文件要上传的地址,由于我们需要定制上传动作
这里设为#
:http-request:自定义上传行为(重点)
on-success: 上传成功之后的回调
before-upload: 上传之前的检查
-->
<el-upload
class="avatar-uploader"
action="#"
:show-file-list="false"
:before-upload="beforeAvatarUpload"
:http-request="upload"
>
<img v-if="imageUrl" :src="imageUrl" class="avatar">
<i v-else class="el-icon-plus avatar-uploader-icon" />
</el-upload>
</div>
</template>
<script>
export default {
methods:{
upload(params){
console.log(params)
}
}
}
</script>
注册一个全局组件
我的位置是在src/components/index.js (位置随意)
// 全局注册组件
// 省略其他....
+ import UploadImg from '@/components/UploadImg'
// 1.定义插件(拓展Vue的功能)
const MyPlugin = {
install(Vue) {
// 省略其他....
+ Vue.component(UploadImg.name, UploadImg)
}
}
export default MyPlugin
随机找一个地方测试
封装组件-上传图片到腾讯云
前置基础:
1.在云服务器上的准备:申请cos服务器,配置密钥,设置cors访问
2.在代码层面的准备:
(1) 下载一个官方提供的操作cos服务的包(cos-js-sdk-v5)
(2) 用自己的密钥去实例化cos
(3) 具体做上传
第一步:在项目中安装依赖 :
npm i cos-js-sdk-v5 --save
第二步:实例化cos对象
在src/components/UploadImg
中
// 下面的代码是固定写法
const COS = require('cos-js-sdk-v5')
// 填写自己腾讯云cos中的key和id (密钥)
const cos = new COS({
SecretId: 'xxx', // 身份识别ID
SecretKey: 'xxx' // 身份秘钥
})
上边用到的SecretId和SecretKey在这里找到:
使用cos对象完成上传
主要是用cos.putObject
api来完成上传功能,代码如下:
upload(res) {
if (res.file) {
// 执行上传操作
cos.putObject({
Bucket: 'xxxxxx', /* 存储桶 */
Region: 'xxxx', /* 存储桶所在地域,必须字段 */
Key: res.file.name, /* 文件名 */
StorageClass: 'STANDARD', // 上传模式, 标准模式
Body: res.file // 上传文件对象
}, (err, data) => {
console.log(err || data)
// 上传成功之后
if (data.statusCode === 200) {
this.imageUrl = `https:${data.Location}`
}
})
}
}
上边的Bucket和Region在这里可以找到
用户详情页中使用上传组件-v-model
我们希望达成的目标是:在upload-img上可以使用v-model做双向绑定。如下:
<el-form-item label="员工头像">
<!-- 放置上传图片 -->
<Uploadimg v-model="userInfo.staffPhoto" />
<img>
</el-form-item>
upload-img: 自定义的组件, v-model:双向绑定。
v-model是一个语法糖。
<com v-model="imageUrl" />
上面的写法等价于:
<com :value="imageUrl" @input="val=>imageUrl=val" />
1. 给子组件内部传一个名为value的属性,属性值就是绑定的数据项
2. 在子组件监听一个名为input的事件,回调函数是:将收到的参数值保存到数据项中
父组件:v-model绑定值
<upload-img v-model="userInfo.staffPhoto" />
子组件(图片上传组件)
- 定义props value来接收父组件传入的图片地址
- 直接用value来显示图片(把原来 imageUrl删除了)
- 在上传成功之后,this.$emit('input', 新地址), 抛给父组件,达到更新父组件中v-model绑定数据项
<template>
<div>
<el-upload
class="avatar-uploader"
action="#"
:show-file-list="false"
:on-success="handleAvatarSuccess"
:before-upload="beforeAvatarUpload"
:http-request="upload"
>
<img v-if="value" :src="value" class="avatar">
<i v-else class="el-icon-plus avatar-uploader-icon" />
</el-upload>
</div>
</template>
<script>
// 下面的代码是固定写法
const COS = require('cos-js-sdk-v5')
// 填写自己腾讯云cos中的key和id (密钥)
const cos = new COS({
SecretId: 'AKIDBE7mCMGvzrYbS8bpoaEguuiJitvogOYX', // 身份识别ID
SecretKey: 'zlIaeaL2au1tXxAAvRiGdqpbCalhU7Ys' // 身份秘钥
})
export default {
props: {
value: {
type: String,
default: ''
}
},
data() {
return {
imageUrl: ''
}
},
methods: {
upload(res) {
console.log(res)
cos.putObject(
{
Bucket: 'dd-1312484638' /* 存储桶 */,
Region: 'ap-guangzhou' /* 存储桶所在地域,必须字段 */,
Key: res.file.name /* 文件名 */,
StorageClass: 'STANDARD', // 上传模式, 标准模式
Body: res.file // 上传文件对象
},
(err, data) => {
console.log(err || data)
// 上传成功之后
if (data.statusCode === 200) {
const imageUrl = `https:${data.Location}`
this.$emit('input', imageUrl)
}
}
)
},
handleAvatarSuccess(res, file) {
this.imageUrl = URL.createObjectURL(file.raw)
},
beforeAvatarUpload(file) {
const isPNG = file.type === 'image/png'
const isLt2M = file.size / 1024 / 1024 < 2
if (!isPNG) {
this.$message.error('上传头像图片只能是 PNG 格式!')
}
if (!isLt2M) {
this.$message.error('上传头像图片大小不能超过 2MB!')
}
return isPNG && isLt2M
}
}
}
</script>
<style>
.avatar-uploader .el-upload {
border: 1px dashed #d9d9d9;
border-radius: 6px;
cursor: pointer;
position: relative;
overflow: hidden;
}
.avatar-uploader .el-upload:hover {
border-color: #409eff;
}
.avatar-uploader-icon {
font-size: 28px;
color: #8c939d;
width: 178px;
height: 178px;
line-height: 178px;
text-align: center;
}
.avatar {
width: 178px;
height: 178px;
display: block;
}
</style>
最后小结一下
图片上传组件整个工作过程