ImageUpload.vue 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262
  1. <template>
  2. <div class="com-image-upload">
  3. <draggable
  4. v-model="imageList"
  5. draggable=".image-item"
  6. @end="end"
  7. class="flex start wrap">
  8. <div
  9. v-for="(item, index) in imageList"
  10. :key="item.uid || index"
  11. class="image-item flex"
  12. :style="{ width: width, height: height }">
  13. <img
  14. :src="item.url"
  15. alt=""
  16. class="" />
  17. <div class="action-area flex center">
  18. <!-- 预览 -->
  19. <span
  20. v-if="!disablePreview"
  21. class="action-icon"
  22. @click="handlePictureCardPreview(item)">
  23. <i class="el-icon-zoom-in"></i>
  24. </span>
  25. <!-- 删除 -->
  26. <span
  27. class="action-icon"
  28. @click="handleRemove(index)">
  29. <i class="el-icon-delete"></i>
  30. </span>
  31. </div>
  32. </div>
  33. <el-progress
  34. v-show="loading"
  35. type="circle"
  36. :percentage="uploadPercent"
  37. :width="Number(width.slice(0, width.length - 2))"
  38. style="margin: 8px"></el-progress>
  39. <div
  40. class="upload-wrap"
  41. :class="{ hide: loading || imageList.length >= max }"
  42. :style="{ width: width, height: height }">
  43. <el-upload
  44. ref="pictureUpload"
  45. :multiple="true"
  46. :limit="max"
  47. action=""
  48. drag
  49. accept=".jpg,.png,.jpeg"
  50. class="custom-upload-item"
  51. list-type="picture-card"
  52. :file-list="imageList"
  53. :show-file-list="false"
  54. :auto-upload="false"
  55. :on-remove="handleRemove"
  56. :on-preview="handlePictureCardPreview"
  57. :on-change="
  58. (file, fileList) => {
  59. handleUpload(file, fileList)
  60. }
  61. ">
  62. <i class="el-icon-plus avatar-uploader-icon"></i>
  63. </el-upload>
  64. </div>
  65. </draggable>
  66. <el-dialog :visible.sync="imageDialogVisible">
  67. <img
  68. width="100%"
  69. :src="imageUrl"
  70. alt="" />
  71. </el-dialog>
  72. </div>
  73. </template>
  74. <script>
  75. import draggable from 'vuedraggable'
  76. // import { common } from '@/api'
  77. export default {
  78. name: 'ImageUpload',
  79. components: {
  80. draggable,
  81. },
  82. props: {
  83. list: {
  84. type: Array,
  85. default: () => [],
  86. },
  87. max: {
  88. type: Number,
  89. default: 16,
  90. },
  91. disablePreview: {
  92. type: Boolean,
  93. default: false,
  94. },
  95. width: {
  96. type: String,
  97. default: '150px',
  98. },
  99. height: {
  100. type: String,
  101. default: '150px',
  102. },
  103. },
  104. data() {
  105. return {
  106. // 组件内部数据.
  107. imageList: [],
  108. loading: false,
  109. uploadPercent: 0,
  110. imageDialogVisible: false,
  111. // 预览大图的url, 每次点击都会更新
  112. imageUrl: '',
  113. }
  114. },
  115. computed: {
  116. fulled() {
  117. return 0
  118. },
  119. },
  120. watch: {
  121. list() {
  122. this.imageList = JSON.parse(JSON.stringify(this.list))
  123. },
  124. },
  125. mounted(){
  126. this.updateList()
  127. },
  128. methods: {
  129. handleUpload(file, fileList) {
  130. this.fileList = []
  131. if (file.status === 'ready') {
  132. this.loading = true
  133. const interval = setInterval(() => {
  134. if (this.uploadPercent >= 99) {
  135. clearInterval(interval)
  136. return
  137. }
  138. this.uploadPercent += 1 // 进度条进度
  139. }, 100)
  140. }
  141. const formData = new FormData()
  142. fileList.forEach((file) => {
  143. formData.append('file', file.raw)
  144. })
  145. formData.append('type', 1)
  146. this.$axios
  147. .post(`/uk-api/user_base/imagesUpload`,formData)
  148. .then((response) => {
  149. if (response.result.code === 200) {
  150. this.imageList.push({
  151. url: response.result.data,
  152. uid: file.uid,
  153. })
  154. this.updateList()
  155. return
  156. }
  157. this.$message({
  158. message: response.result.message,
  159. type: 'warning',
  160. })
  161. })
  162. .catch((error) => {
  163. console.log(error, 'component upload image error')
  164. this.$message.error(error.response.data.msg)
  165. })
  166. .finally(() => {
  167. this.loading = false
  168. // 进度条恢复到初始状态
  169. this.uploadPercent = 0
  170. })
  171. },
  172. handleRemove(index) {
  173. this.imageList.splice(index, 1)
  174. this.updateList()
  175. },
  176. handlePictureCardPreview(file) {
  177. this.imageUrl = file.url
  178. this.imageDialogVisible = true
  179. },
  180. // 每次更新imageList后手动更新父组件的数据, 不能用watch自动更新, 因为同时要watch prop值更新iamgeList, 同时watch会死循环.
  181. // 直接把prop数据绑定到dragable 和 el-upload的话vue和eslint会报错, 也可能造成调试困难
  182. updateList() {
  183. this.$emit('update:list', JSON.parse(JSON.stringify(this.imageList)))
  184. },
  185. end() {
  186. this.updateList()
  187. },
  188. },
  189. }
  190. </script>
  191. <style lang="scss" scoped>
  192. ::v-deep .el-upload {
  193. border-style: solid;
  194. width: 100%;
  195. height: 100%;
  196. }
  197. ::v-deep .el-upload-dragger {
  198. width: 100%;
  199. height: 100%;
  200. border: 0;
  201. display: flex;
  202. align-items: center;
  203. justify-content: center;
  204. }
  205. .com-image-upload {
  206. vertical-align: top;
  207. }
  208. .image-item {
  209. overflow: hidden;
  210. position: relative;
  211. border: 1px solid #c0ccda;
  212. border-radius: 6px;
  213. margin-right: 8px;
  214. margin-bottom: 2px;
  215. img {
  216. width: 100%;
  217. }
  218. &:hover {
  219. .action-area {
  220. display: flex;
  221. }
  222. }
  223. .action-area {
  224. position: absolute;
  225. z-index: 1;
  226. left: 0;
  227. top: 0;
  228. display: none;
  229. width: 100%;
  230. height: 100%;
  231. background-color: rgba(0, 0, 0, 0.7);
  232. color: #fff;
  233. font-size: 32px;
  234. }
  235. .action-icon {
  236. cursor: pointer;
  237. & + .action-icon {
  238. margin-left: 16px;
  239. }
  240. }
  241. }
  242. .upload-wrap {
  243. display: inline-block;
  244. position: relative;
  245. margin-bottom: 2px;
  246. &.hide {
  247. display: none;
  248. }
  249. }
  250. .custom-upload-item {
  251. width: 100%;
  252. height: 100%;
  253. }
  254. </style>