草庐IT

vue+element UI upload 上传组件使用经验分享

前端开发小陈 2023-10-13 原文

用 vue+element UI 框架开发前端项目时,产品功能要求可以上传图像和文件,这就可以用到 el-upload 上传组件。在用此组件实现功能时,从陌生到熟悉,掌握了一些方法,仅以此文做一个总结梳理。 关于 upload 上传 element 的官网就一句话介绍:通过点击或者拖拽上传文件。它的基础代码如下:

<el-upload
  class="upload-demo"
  action="https://jsonplaceholder.typicode.com/posts/"
  :on-preview="handlePreview"
  :on-remove="handleRemove"
  :before-remove="beforeRemove"
  multiple
  :limit="3"
  :on-exceed="handleExceed"
  :file-list="fileList">
  <el-button size="small" type="primary">点击上传</el-button>
  <div slot="tip" class="el-upload__tip">只能上传jpg/png文件,且不超过500kb</div>
</el-upload> 

我们看 element 官网详细的说明,可以发现 el-upload 的触发事件和属性特别多,action, on-priview, on-remove, before-remove, on-exceed 等大概有二十多个,每个场景都不一样,那我们实际项目中真的会用到这么多的事件吗?我的回答是不用,根据自己的需求来做合适的选择。 本文就结合我自己的实际项目需求来进行说明。代码如下:

<el-upload
            action
            list-type="picture-card"
            :limit="2"
            :http-request="uploadPicture"
            :file-list="pictureList"
            :on-change="handleChangePicture"
          >
            <i class="el-icon-plus" />
            <div slot="file" slot-scope="{ file }">
              <el-image
                style="width: 147px; height: 147px"
                :src="file.url"
                fit="cover"
                :preview-src-list="[file.url]"
              />
            </div>
 </el-upload> 

我这段代码要实现的是上传图片的功能,所以 list-type 设置为 picture-card,这样文件列表的类型就是图片了,如果不设置,默认就是 text 文件。如果我不做 limit (用来限制上传文件/图片的数量)限制,那么效果如下:

但是在实际应用时,我只需要上传一张图片,所以我这里用了 limit 来限制,这里你会感觉疑惑,为什么代码中设置的 limit 为 2。那是因为如果设置为 1,就没有办法编辑修改了,这样其实是不太友好的。所以我这里设置为 2,然后结合 file-list 和 on-change 来做新旧图片的覆盖。file-list 是上传的文件列表,是一个数组。给它定义一个数组 pictureList。 on-change 是文件状态改变时的钩子,添加文件、上传成功和上传失败时都会被调用。然后 handleChangePicture 定义如下,当上传文件这个动作执行时,就调用 handleChangePicture 方法,判断 fileList 文件个数如果大于 1 ,就删除第一个文件,这样就实现了图片覆盖的效果。

 /**
     * 覆盖图片
     */
    handleChangePicture(file, fileList) {
      if (fileList.length > 1) {
        fileList.splice(0, 1);
      }
    } 

http-request 是覆盖默认的上传行为,可以自定义上传的实现。我们可以用这个来调用 api 接口,实现与后端的交互,这样就实现图片的上传了。关于图片预览钩子 on-preview,我这里就没有用到,直接用了 el-image 组件自带的 preview-src-list 属性,file 是上传文件默认的字段名,获取文件路径使用 file.url ,点击图片就可以直接预览原图,也比较方便。

同理,上传文件时基本上也是这样应用的,代码如下:区别就是默认 list-type 为文件。

 <el-upload
          ref="upload"
          action
          :on-progress="uploadProcess"
          :http-request="uploadFile"
          :limit="2"
          :file-list="fileList"
          :on-change="handleChangeFile"
          :on-remove="handleRemove"
        >
          <el-button class="updata-button" size="small" round type="success">
            {{
              editForm.file != "" && editForm.file != null
                ? $t("lab.item.ChooseAgain")
                : $t("lab.item.Choose")
            }}
          </el-button>
    </el-upload> 

页面效果如下:

另外这里我还用到了 on-remove 钩子,是文件列表移除文件时的钩子。项目要求是图片必须要有,附档可以为空,所以这里增加了这样一个钩子,在删除文件时调用 handleRemove 方法,清空文件参数,同时按钮文字也根据参数清空而改变,这里做一个 v-if 判断就可以。

有的时候上传文件会限制文件格式,这个时候我们可以使用 accept 属性。参考代码如下:

 <el-upload
        ref="upload"
        class="upload-demo"
        :headers="header"
        :action="changeUploadUrl"
        :on-preview="handlePreview"
        :on-remove="handleRemove"
        :before-upload="beforeUpload"
        :auto-upload="false"
        :on-error="uploadError"
        :on-success="handleFileSuccess"
        accept=".xls, .xlsx"
        name="file"
      >
        <el-button
          slot="trigger"
          size="small"
          type="primary"
        >选取文件</el-button>
        <el-button
          style="margin-left: 10px"
          size="small"
          type="success"
          @click="submitUpload"
        >上传到服务器</el-button>
        <div slot="tip" class="el-upload__tip">
          <span style="color: red">*</span> 只能上传格式正确,文件未损坏的
          xls/xlsx 文件,且文件大小不超过 5M
          <div style="margin-top: 10px">
            可下载
            <el-button type="text" @click="onDownload">
              {{ excelExample.name }}
            </el-button>
            填写内容
          </div>
        </div>
      </el-upload> 

设置 accept 属性为 .xls,xlsx, 就意味着上传文件只能是 Excel 文件。如果是 PDF 文件,就设置为 .pdf, 如果是 Word 文件,就设置 .doc, .docx 等等以此类推,页面效果如下:

当然如果上传文件还有大小要求,就需要用到 before-upload 钩子,在上传文件之前调用,来判断文件大小是否超出限制,超出就中断上传。代码如下:

/**
     * 上传文件之前的钩子,参数为上传的文件,若返回 false 或者返回 Promise 且被 reject,则停止上传。
     * @param {Object} file 上传的文件信息
     * @return {Boolean} 返回 false 则停止上传,true 继续
     */
    beforeUpload(file) {
      // 定义能上传的文件格式
      var ext = file.name.substring(file.name.lastIndexOf('.') + 1);
      const isLtSize = file.size / 1024 / 1024 < 5;
      const extension = ext === 'xls';
      const extension2 = ext === 'xlsx';

      if (!extension && !extension2) {
        this.$notify.warning({
          message: '上传文件只能是 xls/xlsx 格式!'
        });
      }
      if (!isLtSize) {
        this.$notify.warning({ message: '上传文件大小不能超过 5MB!' });
      }
      return isLtSize && (extension || extension2);
    } 

同时也可以判断文件格式是否满足要求,实现一个双保险。

前文我们提到 http-request 可以与服务器进行交互,这个是在文件上传之后就直接调用。但是如果不自动交互的情况,就要对 headers, action, auto-upload 这三个属性进行设置,headers 设置上传的请求头部,action 必选参数,设置上传的地址。auto-upload 设定是否在选取文件后立即进行上传,默认是 true,这里要设置为 false。所以是选择 http-request 还是 headers + action 的方式都是可以的,就根据自己的应用场景来选择。

Upload 上传组件中的其它的钩子和属性,我这里就不做赘述,应用场景都比较容易明白,可以直接参考官网的说明。本文的内容就介绍到这里,希望可以帮助到有同样需求的网友。

有关vue+element UI upload 上传组件使用经验分享的更多相关文章

  1. ruby - 如何使用 Nokogiri 的 xpath 和 at_xpath 方法 - 2

    我正在学习如何使用Nokogiri,根据这段代码我遇到了一些问题:require'rubygems'require'mechanize'post_agent=WWW::Mechanize.newpost_page=post_agent.get('http://www.vbulletin.org/forum/showthread.php?t=230708')puts"\nabsolutepathwithtbodygivesnil"putspost_page.parser.xpath('/html/body/div/div/div/div/div/table/tbody/tr/td/div

  2. ruby - 使用 RubyZip 生成 ZIP 文件时设置压缩级别 - 2

    我有一个Ruby程序,它使用rubyzip压缩XML文件的目录树。gem。我的问题是文件开始变得很重,我想提高压缩级别,因为压缩时间不是问题。我在rubyzipdocumentation中找不到一种为创建的ZIP文件指定压缩级别的方法。有人知道如何更改此设置吗?是否有另一个允许指定压缩级别的Ruby库? 最佳答案 这是我通过查看ruby​​zip内部创建的代码。level=Zlib::BEST_COMPRESSIONZip::ZipOutputStream.open(zip_file)do|zip|Dir.glob("**/*")d

  3. ruby - 为什么我可以在 Ruby 中使用 Object#send 访问私有(private)/ protected 方法? - 2

    类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc

  4. ruby-on-rails - 使用 Ruby on Rails 进行自动化测试 - 最佳实践 - 2

    很好奇,就使用ruby​​onrails自动化单元测试而言,你们正在做什么?您是否创建了一个脚本来在cron中运行rake作业并将结果邮寄给您?git中的预提交Hook?只是手动调用?我完全理解测试,但想知道在错误发生之前捕获错误的最佳实践是什么。让我们理所当然地认为测试本身是完美无缺的,并且可以正常工作。下一步是什么以确保他们在正确的时间将可能有害的结果传达给您? 最佳答案 不确定您到底想听什么,但是有几个级别的自动代码库控制:在处理某项功能时,您可以使用类似autotest的内容获得关于哪些有效,哪些无效的即时反馈。要确保您的提

  5. ruby - 在 Ruby 中使用匿名模块 - 2

    假设我做了一个模块如下:m=Module.newdoclassCendend三个问题:除了对m的引用之外,还有什么方法可以访问C和m中的其他内容?我可以在创建匿名模块后为其命名吗(就像我输入“module...”一样)?如何在使用完匿名模块后将其删除,使其定义的常量不再存在? 最佳答案 三个答案:是的,使用ObjectSpace.此代码使c引用你的类(class)C不引用m:c=nilObjectSpace.each_object{|obj|c=objif(Class===objandobj.name=~/::C$/)}当然这取决于

  6. ruby - 使用 ruby​​ 和 savon 的 SOAP 服务 - 2

    我正在尝试使用ruby​​和Savon来使用网络服务。测试服务为http://www.webservicex.net/WS/WSDetails.aspx?WSID=9&CATID=2require'rubygems'require'savon'client=Savon::Client.new"http://www.webservicex.net/stockquote.asmx?WSDL"client.get_quotedo|soap|soap.body={:symbol=>"AAPL"}end返回SOAP异常。检查soap信封,在我看来soap请求没有正确的命名空间。任何人都可以建议我

  7. python - 如何使用 Ruby 或 Python 创建一系列高音调和低音调的蜂鸣声? - 2

    关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。

  8. ruby-on-rails - 'compass watch' 是如何工作的/它是如何与 rails 一起使用的 - 2

    我在我的项目目录中完成了compasscreate.和compassinitrails。几个问题:我已将我的.sass文件放在public/stylesheets中。这是放置它们的正确位置吗?当我运行compasswatch时,它不会自动编译这些.sass文件。我必须手动指定文件:compasswatchpublic/stylesheets/myfile.sass等。如何让它自动运行?文件ie.css、print.css和screen.css已放在stylesheets/compiled。如何在编译后不让它们重新出现的情况下删除它们?我自己编译的.sass文件编译成compiled/t

  9. ruby - 使用 ruby​​ 将 HTML 转换为纯文本并维护结构/格式 - 2

    我想将html转换为纯文本。不过,我不想只删除标签,我想智能地保留尽可能多的格式。为插入换行符标签,检测段落并格式化它们等。输入非常简单,通常是格式良好的html(不是整个文档,只是一堆内容,通常没有anchor或图像)。我可以将几个正则表达式放在一起,让我达到80%,但我认为可能有一些现有的解决方案更智能。 最佳答案 首先,不要尝试为此使用正则表达式。很有可能你会想出一个脆弱/脆弱的解决方案,它会随着HTML的变化而崩溃,或者很难管理和维护。您可以使用Nokogiri快速解析HTML并提取文本:require'nokogiri'h

  10. ruby - 在 64 位 Snow Leopard 上使用 rvm、postgres 9.0、ruby 1.9.2-p136 安装 pg gem 时出现问题 - 2

    我想为Heroku构建一个Rails3应用程序。他们使用Postgres作为他们的数据库,所以我通过MacPorts安装了postgres9.0。现在我需要一个postgresgem并且共识是出于性能原因你想要pggem。但是我对我得到的错误感到非常困惑当我尝试在rvm下通过geminstall安装pg时。我已经非常明确地指定了所有postgres目录的位置可以找到但仍然无法完成安装:$envARCHFLAGS='-archx86_64'geminstallpg--\--with-pg-config=/opt/local/var/db/postgresql90/defaultdb/po

随机推荐