基于element-ui封装的upload组件

简介: 无论什么项目,大概都少不了图片上传。作为常见的需求,很多地方会使用到,应该单独封装一个上传组件,方便复用。

图片上传

无论什么项目,大概都少不了图片上传。作为常见的需求,很多地方会使用到,应该单独封装一个上传组件,方便复用。

这里使用 vue + element-ui-upload + 七牛云完成上传

前端调用七牛 API

现在主流的七牛云上传方式大概为授权式上传,大概为如下过程:

请求后端接口获取上传凭证 token(后端通过accessKey,secretKey,bucket 生成 token)

请求七牛云的接口地址完成上传

七牛云服务器返回图片地址(返回 hash,key,需要自己拼接下)

如果说后端很好,在服务器端直接调用七牛上传接口,那么前端只需要传一个文件给后端,前端方面会简单许多。

不过我是没有遇到的,而且前端调用上传接口,可控性会更强些

经过测试,上面四个接口都是可用的(https 或者 http),我这里的空间是华南区域,不同区域会有所不同,可以参考

七牛 API,有 3 个参数,token、file、key(可选),其中 key 是文件名,不传会自动生成

首先分析下需求,完成的 upload 组件,要和表单结合起来,意味着要实现双向绑定,调用这个组件的时候,只需要绑定 value(图片url) 属性,组件内部上传完成后通过 $emit('input', url) 改变 value,这样就很方便了

下面介绍下 el-upload 组件:

action 属性是上传的接口地址,直接用上面的七牛云的上传地址

name 字段是文件流的参数字段名,默认值为 file,七牛云上传的文件流参数字段就是 file,所以这里可以忽略

data 属性是需要传递的参数,七牛云的上传地址所需参数包括 file(插件会自动传递)、key、token,key 可以不传(七牛自动生成),我们只需要获取 token,在调用七牛之前将 token 塞到 data 中就可以了

before-upload 属性是上传文件之前的钩子,这里调用后端接口获取到上传需要的 token,塞到 data 中,然后别忘了 resolve,因为获取 token 是异步过程,所以在钩子里面返回 Promise,请求成功后 resolve 进行上传(如果有校验文件大小、文件类型的需求也可以在钩子中完成,提前返回 false 就可以了)

效果:

image.png

image.png

下面是代码:

<template>
  <el-upload
    v-loading="loading"
    class="uploader"
    :class="{'hover-mask': value}"
    action="https://up-z2.qiniup.com"
    :show-file-list="false"
    :data="param"
    accept="image/*"
    :on-success="handleSuccess"
    :before-upload="handlebeforeUpload">
      <img v-if="value" :src="value" class="avatar">
      <i class="el-icon-plus uploader-icon"></i>
  </el-upload>
</template>

<script>
import axios from 'axios'
export default {
  props: {
    value: String,
    required: true
  },
  data() {
    return {
      loading: '',
      param: {
        token: ''
      }
    }
  },
  methods: {
    handleSuccess(res, file) {
      this.loading = false
      // 如果不传 key 参数,就使用七牛自动生成的 hash 值,如果传递了 key 参数,那么就用返回的 key 值
      const { hash } = res
      // 拼接得到图片 url
      const imageUrl = 'your domain prefix' + hash
      // 触发事件 input,父组件会修改绑定的 value 值
      this.$emit('input', imageUrl)
    },
    handlebeforeUpload(file) {
      // 这里做可以做文件校验操作
      const isImg = /^image\/\w+$/i.test(file.type)
      if (!isImg) {
        this.$message.error('只能上传 JPG、PNG、GIF 格式!')
        return false
      }
      return new Promise((resolve, reject) => {
        this.loading = true
        // 获取token
        const tokenUrl = 'http://xxx/upload'
        axios.get(tokenUrl).then(res => {
          const { token } = res.data.data
          this.param.token = token
          resolve(true)
        }).catch(err => {
          this.loading = false
          reject(err)
        })
      })
    }
  }
}
</script>

<style scoped lang="scss">
  .uploader {
    width: 130px;
    height: 130px;
    border: 1px dashed #d9d9d9;
    border-radius: 6px;
    cursor: pointer;
    &:hover {
      border-color: #409EFF;
    }
    /deep/ .el-upload {
      position: relative;
      width: 100%;
      height: 100%;
      overflow: hidden;
    }
  }
  .uploader-icon {
    position: absolute;
    top: 0;
    left: 0;
    width: 100%;
    height: 100%;
    line-height: 128px;
    text-align: center;
    font-size: 28px;
    color: #8c939d;
  }
  .avatar + .uploader-icon {
    opacity: 0;
  }
  .avatar {
    width: 128px;
    height: 128px;
    display: block;
    border-radius: 6px;
  }
  .hover-mask:hover .uploader-icon {
    opacity: 1;
    background-color: rgba(0, 0, 0, .2);
    color: #fff;
  }
</style>

如何使用:

  <el-form ref="form" :model="form">
    <el-form-item label="头像" prop="avatar">
      <upload v-model="form.avatar"></upload>
      </el-form-item>
    <el-form-item label="姓名" prop="userName">
      <el-input v-model="form.userName"></el-input>
      </el-form-item>
  </el-form>
    <el-button type="primary" @click="onSubmit">确 定</el-button>
</template>
<script>
import Upload from '@/components/Upload'
export default {
  components: { Upload },
  data () {
    return {
      form: {
        avatar: '',
        userName: ''
      }
    }
  },
  methods: {
    onSubmit() {
      this.$refs.form.validata(valid => {
        if (valid) {
          // 将 this.form 传给后端
        }
      })
    }
  }
}
</script>

实现双向绑定后,收集数据会非常方便,调用后端接口直接传递绑定的值就 ok 了

前端直接调用后端上传接口

如果后端很好(铁哥们),前端只需要传递 file 对象,那就更简单了,在上传前置钩子中就不用获取 token,这部分工作都由后端去处理,我们只需要调用上传接口就可以

  <el-upload
    v-loading="loading"
    class="uploader"
    :class="{'hover-mask': value}"
    action="your upload api"
    :show-file-list="false"
    :on-success="handleSuccess"
    :before-upload="handlebeforeUpload">
      <img v-if="value" :src="value" class="avatar">
        <i class="el-icon-plus uploader-icon"></i>
  </el-upload>
</template>

<script>
import axios from 'axios'
export default {
  props: {
    value: String,
    required: true
  },
  data() {
    return {
      loading: ''
    }
  },
  methods: {
    handleSuccess(res, file) {
      this.loading = false
      const { hash } = res
      const imageUrl = 'your domain prefix' + hash
      this.$emit('input', imageUrl)
    },
    handlebeforeUpload(file) {
      // 不需要操作可以直接 返回 true
      return true
    }
  }
}
</script>

其实,我们在组件上使用v-model的时候,实际上是下面这样:

  :value="form.avatar"
  @input="form.avatar = $event"
></custom-upload>

为了让它正常工作,custom-upload组件内必须:

将其 value 特性绑定到一个名叫 value 的 prop 上(这里就是图片地址)

需要修 value时,将新的值通过自定义的 input 事件抛出(这里就是上传成功后的$emit('input', 图片地址))

所以 v-model是一个语法糖,一种简写形式。如果想要双向绑定,又想自定义,那么可以使用上面的方式:

  :url="form.avatar"
  @update:url="form.avatar = $event"
></custom-upload>

那么在子组件内部就接受url属性,作为图片地址就可以了,在更新url时,也要与绑定的事件名一致$emit('update:url', 图片地址),事件名使用update:propName是vue推荐的风格,目的是提醒开发者这是一个双向绑定的属性。

当然,为了方便起见,vue为这种模式提供一个缩写,即 .sync 修饰符:

<custom-upload :url.sync="form.avatar"></custom-upload>

带有 .sync 修饰符的值不能为表达式(例如:url.sync="domain + form.avatar"是无效的)

多图上传

有时候类似上传资料、凭证这类的需求要求上传多张图片,我们可以再封装一个多图上传的组件

对于 el-upload,多张图片上传注意如下几点:

props 的 value 不再是string,应该是一个数组,数组成员为图片地址['url1', 'url2']

file-list属性为上传的文件列表,我们不能直接把 value 赋值给它,file-list应该是一个数组,例如[{name: 'foo.jpg', url: 'xxx'}, {name: 'bar.jpg', url: 'xxx'}]。这与我们传进来的数据不一样,需要处理一下 value(当然我们使用组件时可以直接传递需要的这种格式,就不用处理了)

show-file-list 设置为 true,当然可以不传,它默认为 true

list-type 可以设置为 'picture-card',图片将以卡片形式显示

on-remove 属性为文件列表移除文件时的钩子,这里需要触发事件,更新 value

on-preview 属性为点击文件列表中已上传的文件时的钩子,可以做图片预览

limit 属性可以指定最大允许上传个数,配合 on-exceed 属性(文件超出个数限制时的钩子)使用,当上传超过指定值后,在这个钩子里面做些提示

效果:

image.png

下面是代码:

<template>
  <div>
    <el-upload
      :action="QINIU_UPLOAD_URL"
      :data="param"
      :file-list="fileList"
      list-type="picture-card"
      :limit="limit"
      :on-exceed="handleUploadExceed"
      :on-preview="handlePictureCardPreview"
      :on-remove="handleRemove"
      :before-upload="handlebeforeUpload"
      :on-success="handleSuccess">
      <i class="el-icon-plus"></i>
    </el-upload>
    <el-dialog :visible.sync="dialogVisible">
      <img width="100%" :src="dialogImageUrl" alt="">
    </el-dialog>
  </div>
</template>

<script>
import axios from 'axios'
export default {
  props: {
    value: {
      type: Array,
      default: () => []
    },
    limit: {
      type: Number,
      default: 4
    }
  },
  data() {
    return {
      dialogImageUrl: '', // 当前预览图片地址
      dialogVisible: false, // 预览弹框 visible
      param: {
        token: ''
      }
    }
  },
  computed: {
    // ['xxx', 'xxx'] 转换为 [{url: 'xxx'}, {url: 'xxx'}]
    fileList() {
      return this.value.map(url => ({ url }))
    }
  },
  methods: {
    handleUploadExceed() {
      this.$message.error(`最多上传${this.limit}张图片`)
    },
    handleRemove(file, fileList) {
      // fileList 为删除后的文件列表
      const value = fileList.map(v => v.url)
      this.$emit('input', value)
    },
    handlePictureCardPreview(file) {
      this.dialogImageUrl = file.url
      this.dialogVisible = true
    },
    handlebeforeUpload(file) {
      return new Promise((resolve, reject) => {
        axios.get('/upload/qiniuToken').then(res => {
          const { token } = res.data
          this.param.token = token
          resolve(true)
        }).catch(err => {
          reject(err)
        })
      })
    },
    handleSuccess(res, file) {
      const { hash } = res
      const imageUrl = this.QINIU_PREFIX + hash
      // 这里如果 this.value.push(imageUrl) 这么写,vue会报出警告,大概意思是value作为props不应该在子组件中被修改
      // 应该根据 value 得到新的值,而不能修改它,this.value.concat(imageUrl)也是可以的,concat方法返回新的数组
      this.$emit('input', [...this.value, imageUrl])
    }
  }
}
</script>

如何使用:

  <el-form ref="form" :model="form">
    <el-form-item label="还款金额" prop="amount">
      <el-input v-model="form.amount"></el-input>
    </el-form-item>
    <el-form-item label="凭证" prop="voucherUrlList">
      <multi-upload v-model="form.voucherUrlList"></multi-upload>
    </el-form-item>
  </el-form>
  <el-button type="primary" @click="onSubmit">确 定</el-button>
</template>
<script>
import MultiUpload from '@/components/MultiUpload'
export default {
  components: { MultiUpload },
  data () {
    return {
      form: {
        amount: '',
        voucherUrlList: []
      }
    }
  },
  methods: {
    onSubmit() {
      this.$refs.form.validata(valid => {
        if (valid) {
          // 将 this.form 传给后端
        }
      })
    }
  }
}
</script>

山水有相逢,来日皆可期,谢谢阅读,我们再会

我手中的金箍棒,上能通天,下能探海

上一篇:接入微信小程序直播

相关文章
|
18天前
「Mac畅玩鸿蒙与硬件46」UI互动应用篇23 - 自定义天气预报组件
本篇将带你实现一个自定义天气预报组件。用户可以通过选择不同城市来获取相应的天气信息,页面会显示当前城市的天气图标、温度及天气描述。这一功能适合用于动态展示天气信息的小型应用。
124 38
|
9天前
|
人工智能 开发框架 JavaScript
LowCodeEngine:阿里开源的企业级低代码开发平台,提供预制的 UI 组件和模板,覆盖完整的研发周期
LowCodeEngine 是阿里巴巴开源的低代码开发框架,旨在通过拖拽、配置等简单操作,帮助开发者快速构建复杂的系统页面,提升开发效率和质量。
52 4
LowCodeEngine:阿里开源的企业级低代码开发平台,提供预制的 UI 组件和模板,覆盖完整的研发周期
|
2月前
|
UED
「Mac畅玩鸿蒙与硬件31」UI互动应用篇8 - 自定义评分星级组件
本篇将带你实现一个自定义评分星级组件,用户可以通过点击星星进行评分,并实时显示评分结果。为了让界面更具吸引力,我们还将添加一只小猫图片作为评分的背景装饰。
80 6
|
2月前
|
前端开发 开发者
「Mac畅玩鸿蒙与硬件21」鸿蒙UI组件篇11 - Canvas 组件的静态进阶应用
在鸿蒙应用开发中,Canvas 组件不仅用于基础绘图,还提供了处理复杂路径和渐变效果的多种手段,帮助开发者实现精美的静态图形。本篇将介绍如何在 Canvas 中绘制复杂路径、创建渐变填充效果。
64 7
「Mac畅玩鸿蒙与硬件21」鸿蒙UI组件篇11 - Canvas 组件的静态进阶应用
|
2月前
|
前端开发 开发者
「Mac畅玩鸿蒙与硬件22」鸿蒙UI组件篇12 - Canvas 组件的动态进阶应用
在鸿蒙应用中,Canvas 组件可以实现丰富的动态效果,适合用于动画和实时更新的场景。本篇将介绍如何在 Canvas 中实现动画循环、动态进度条、旋转和缩放动画,以及性能优化策略。
67 6
|
2月前
|
前端开发 开发者
「Mac畅玩鸿蒙与硬件23」鸿蒙UI组件篇13 - 自定义组件的创建与使用
自定义组件可以帮助开发者实现复用性强、逻辑清晰的界面模块。通过自定义组件,鸿蒙应用能够提高代码的可维护性,并简化复杂布局的构建。本篇将介绍如何创建自定义组件,如何向组件传递数据,以及如何在不同页面间复用这些组件。
59 5
|
1月前
|
XML 搜索推荐 前端开发
安卓开发中的自定义视图:打造个性化UI组件
在安卓应用开发中,自定义视图是一种强大的工具,它允许开发者创造独一无二的用户界面元素,从而提升应用的外观和用户体验。本文将通过一个简单的自定义视图示例,引导你了解如何在安卓项目中实现自定义组件,并探讨其背后的技术原理。我们将从基础的View类讲起,逐步深入到绘图、事件处理以及性能优化等方面。无论你是初学者还是有经验的开发者,这篇文章都将为你提供有价值的见解和技巧。
|
2月前
|
搜索推荐 Android开发 开发者
探索安卓开发中的自定义视图:打造个性化UI组件
【10月更文挑战第39天】在安卓开发的世界中,自定义视图是实现独特界面设计的关键。本文将引导你理解自定义视图的概念、创建流程,以及如何通过它们增强应用的用户体验。我们将从基础出发,逐步深入,最终让你能够自信地设计和实现专属的UI组件。
|
3月前
|
开发框架 JavaScript 前端开发
鸿蒙NEXT开发声明式UI是咋回事?
【10月更文挑战第15天】鸿蒙NEXT的声明式UI基于ArkTS,提供高效简洁的开发体验。ArkTS扩展了TypeScript,支持声明式UI描述、自定义组件及状态管理。ArkUI框架则提供了丰富的组件、布局计算和动画能力。开发者仅需关注数据变化,UI将自动更新,简化了开发流程。此外,其前后端分层设计与编译时优化确保了高性能运行,利于生态发展。通过组件创建、状态管理和渲染控制等方式,开发者能快速构建高质量的鸿蒙应用。
163 3
|
14天前
|
移动开发 前端开发 Java
Java最新图形化界面开发技术——JavaFx教程(含UI控件用法介绍、属性绑定、事件监听、FXML)
JavaFX是Java的下一代图形用户界面工具包。JavaFX是一组图形和媒体API,我们可以用它们来创建和部署富客户端应用程序。 JavaFX允许开发人员快速构建丰富的跨平台应用程序,允许开发人员在单个编程接口中组合图形,动画和UI控件。本文详细介绍了JavaFx的常见用法,相信读完本教程你一定有所收获!
Java最新图形化界面开发技术——JavaFx教程(含UI控件用法介绍、属性绑定、事件监听、FXML)