個人博客里部分內容用到了圖片上傳
那就需要用到vue-element-admin中的upload組件
首先我們需要在需要到圖片上傳的vue文件里引入upload組件 並使用標簽的形式使用該組件 通過v-model綁定image屬性
<template> <div class="createPost-container"> <Upload v-model="image"/> </div> </template> <script> import Upload from '@/components/Upload/SingleImage3' export default { name: 'EssayDetail', components: {Upload}, data() { image:'' }, } </script>
這樣組件就能夠為我們做上傳圖片的功能
這樣的圖片上傳使用的是upload默認的上傳,而它內部做的事情則是將我們上傳的文件轉為image base64編碼
返回的結果可能是
..........................(后面省略1萬個字符)
或者更長的Base64編碼
然后再將這么長的一條編碼存入到數據庫,很顯然這么長的編碼已經超出了預算 雖然數據庫字段啊可以用varchar、text、longtext這樣的類型但早已跟我們預期的不符,如果將這么長的編碼存入到數據庫很顯然影響查詢效率
那唯一的方式就是通過upload這個組件的源碼進行修改
以上面引入的Upload/SingleImage3為例 打開SingleImage3.vue文件
<template> <div class="upload-container"> <el-upload :data="dataObj" :multiple="false" :show-file-list="false" :on-success="handleImageSuccess" class="image-uploader" drag action="https://httpbin.org/post" > <i class="el-icon-upload" /> <div class="el-upload__text"> 將文件拖到此處,或<em>點擊上傳</em> </div> </el-upload> <div class="image-preview image-app-preview"> <div v-show="imageUrl.length>1" class="image-preview-wrapper"> <img :src="imageUrl"> <div class="image-preview-action"> <i class="el-icon-delete" @click="rmImage" /> </div> </div> </div> <div class="image-preview"> <div v-show="imageUrl.length>1" class="image-preview-wrapper"> <img :src="imageUrl"> <div class="image-preview-action"> <i class="el-icon-delete" @click="rmImage" /> </div> </div> </div> </div> </template> 通過element-ui的官方文檔閱讀可知action="https://httpbin.org/post" 這個屬性,這是圖片的上傳地址,並綁定了文件上傳成功時的鈎子 :on-success="handleImageSuccess" 接着查看handleImageSuccess這個方法 export default { name: 'SingleImageUpload3', props: { value: { type: String, default: '' } }, data() { return { tempUrl: '', dataObj: { token: '', key: '' } } }, computed: { imageUrl() { return this.value } }, methods: { rmImage() { this.emitInput('') }, emitInput(val) { this.$emit('input', val) }, handleImageSuccess(file) { this.emitInput(file.files.file) }, beforeUpload() { const _self = this return new Promise((resolve, reject) => { getToken().then(response => { const key = response.data.qiniu_key const token = response.data.qiniu_token _self._data.dataObj.token = token _self._data.dataObj.key = key this.tempUrl = response.data.qiniu_url resolve(true) }).catch(err => { console.log(err) reject(false) }) }) } } }
而關於on-success在elementUI官網中的解釋是
參數 | 說明 | 類型 | 可選值 | 默認值 |
---|---|---|---|---|
on-success | 文件上傳成功時的鈎子 | function(response, file, fileList) | — | — |
on-success綁定的handleImageSuccess函數中只有file 這個參數 ,顯然這個函數可以再多加一個response參數
response是上傳成功的回調,我們可以通過后端實現一個圖片上傳接口把action屬性該為我們自己實現的圖片上傳接口,返回一條json
我后端定義的接口是 /pic/upload 后台返回的數據是json數據格式為,(根據自己情況定義json結構)
{ "success":true, "code":200, "message":"上傳成功", "data":"(上傳成功的圖片地址)" }
並將handleImageSuccess改為
handleImageSuccess(res, file) { this.emitInput(res.data) }
然后該方法調用了emitInput()方法傳入了一個值 這個方法又執行了this.$emit('input', val)
vue中 關於$emit的用法 1、父組件可以使用 props 把數據傳給子組件。 2、子組件可以使用 $emit 觸發父組件的自定義事件。
vm.$emit( event, arg ) //觸發當前實例上的事件
vm.$on( event, fn );//監聽event事件后運行 fn;
具體使用還是多學習Vue吧
最后 handleImageSuccess中this.emitInput(res.data)
這條語句
最終返回的就是上傳圖片的url並回顯為upload 組件上的圖片
步驟
-
修改action屬性為自己實現的接口,定義好返回的json,把圖片存在json里返回給前端,下面的response參數要用到
-
在handleImageSuccess中添加response 將 this.$emit('input', val)修改為 this.emitInput(res.data) res.data為圖片地址
-
完