4

具体需求场景如下:

image

选择需要上传的excel文件后,要把文件手动上传到后台服务器,并将导入成功失败的统计结果显示出来。官网也有手动上传的示例,通过 action="url" 传入地址的方式,但在实际项目中请求需要自己配置,下面具体说明实现的方法。

说明:

  • 在上传文件到展示统计结果,我们后端给了两个接口:首先调用文件上传接口,上传成功后,根据后端返回的 mark 再调用统计结果接口,然后展示返回的响应结果。
1、属性设置

.vue文件

<el-row>
    <div class="el-form-item__content">
        <el-upload ref="upload"
            accept=".xls,.xlsx"
            action="#"
            :auto-upload="false"
            :multiple="false"
            :file-list="fileList"
            :before-upload="beforeUpload"
            :http-request="uploadHttpRequest"
            :on-remove="fileRemove"
            :on-change="fileChange">
            <el-button size="small" type="primary">选择文件</el-button>
            <div slot="tip" class="el-upload__tip">一次只能上传一个xls/xlsx文件,且不超过10M</div>
        </el-upload>
    </div>
</el-row>
<el-row>
    <el-button size="small" @click="closeDialog">取 消</el-button>
    <el-button type="primary" size="small" @click="submitUpload">上 传</el-button>
    <el-button type="primary" size="small" @click="downloadRes">下载反馈结果</el-button>
</el-row>
2、处理逻辑

逻辑处理代码如下:

methods: {
    // 上传文件之前的钩子:判断上传文件格式、大小等,若返回false则停止上传
    beforeUpload(file) {
        //文件类型
        const isType = file.type === 'application/vnd.ms-excel'
        const isTypeComputer = file.type === 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet'
        const fileType = isType || isTypeComputer
        if(!fileType) {
            this.$message.error('上传文件只能是xls/xlsx格式!')
        }

        // 文件大小限制为10M
        const fileLimit = file.size / 1024 / 1024 < 10;
        if(!fileLimit) {
            this.$message.error('上传文件大小不超过10M!')
        }
        return fileType && fileLimit
    },
    // 自定义上传方法,param是默认参数,可以取得file文件信息,具体信息如下图
    uploadHttpRequest(param) {
        const formData = new FormData() //FormData对象,添加参数只能通过append('key', value)的形式添加
        formData.append('file', param.file) //添加文件对象
        formData.append('uploadType', this.rulesType)
        const url = `${this.myBaseURL}/operation/ruleImport/importData` //上传地址
        axios.post(url, formData)
            .then( res => {
                const { data: { code, mark } } = res
                if(code === 0) {
                    param.onSuccess()  // 上传成功的文件显示绿色的对勾
                    this.uploadMark = mark
                }
                return this.countData(this.uploadMark) //根据mark值调用统计结果接口,返回一个promise以便进行链式调用
            })
            .then( res => { //链式调用,调用统计结果的响应
                const { data: { code, data } } = res
                if(code === 0) {
                    console.log('统计结果', data)
                }
            })
            .catch( err => {
                console.log('失败', err)
                param.onError() //上传失败的文件会从文件列表中删除
            })
    },
    // 统计结果
    countFile(mark) {
        return new Promise((resolve, reject) => {
            axios
                .get(`/operation/ruleImport/countData?mark=${mark}`)
                .then(res => {
                    resolve(res)
                })
                .catch(error => {
                    reject(error)
                })
        })
    },
    // 点击上传:手动上传到服务器,此时会触发组件的http-request
    submitUpload() {
        this.$refs.upload.submit()
    },
    // 文件发生改变
    fileChange(file, fileList) {
        if (fileList.length > 0) {
            this.fileList = [fileList[fileList.length - 1]] // 展示最后一次选择的文件
        }
    },
    // 移除选择的文件
    fileRemove(file, fileList) {
        if(fileList.length < 1) {
            this.uploadDisabled = true //未选择文件则禁用上传按钮
        }
    },
    // 取消
    closeDialog() {
        this.$refs.upload.clearFiles() //清除上传文件对象
        this.fileList = [] //清空选择的文件列表
        this.$emit('close', false)
    }
}

http-request 的 param 参数,打印结果如图。可通过 param.file 获取当前文件对象。
image

好了,以上就是我手动上传文件的一个实现,但还有一些细节要完善,比如:上传时按钮显示加载状态;上传文件成功后展示列表不允许点击 X 删除等等,大家可根据自己的需求进行调整,谢 :)~~


sugar_coffee
455 声望19 粉丝