index.vue 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267
  1. <template>
  2. <el-dialog
  3. :title="title"
  4. :visible.sync="dialogFormVisible"
  5. width="909px"
  6. :before-close="handleClose"
  7. :close-on-click-modal="false"
  8. >
  9. <div class="upload">
  10. <el-alert
  11. :closable="false"
  12. :title="`支持jpg、jpeg、png格式,单次可最多选择${limit}张图片,每张不可大于${size}M,如果大于${size}M会自动为您过滤`"
  13. type="info"
  14. ></el-alert>
  15. <br />
  16. <el-upload
  17. ref="upload"
  18. class="upload-content"
  19. :name="name"
  20. :data="data"
  21. :action="action"
  22. :headers="headers"
  23. :on-change="handleChange"
  24. :on-preview="handlePreview"
  25. :on-remove="handleRemove"
  26. :on-exceed="handleExceed"
  27. :on-success="handleSuccess"
  28. :on-progress="handleProgress"
  29. :on-error="handleError"
  30. :file-list="fileList"
  31. :multiple="true"
  32. :auto-upload="false"
  33. accept="image/png, image/jpeg"
  34. :limit="limit"
  35. list-type="picture-card"
  36. :close-on-click-modal="false"
  37. >
  38. <i slot="trigger" class="el-icon-plus"></i>
  39. <el-dialog
  40. title="查看大图"
  41. append-to-body
  42. :visible.sync="dialogVisible"
  43. >
  44. <div style="padding-bottom: 20px !important">
  45. <img width="100%" :src="dialogImageUrl" alt="" />
  46. </div>
  47. </el-dialog>
  48. </el-upload>
  49. </div>
  50. <div
  51. slot="footer"
  52. class="dialog-footer"
  53. style="position: relative; padding-right: 15px; text-align: right"
  54. >
  55. <div
  56. v-if="show"
  57. style="position: absolute; top: 10px; left: 15px; color: #999"
  58. >
  59. 正在上传中... 当前上传成功数:{{ imgSuccessNum }}张 当前上传失败数:{{
  60. imgErrorNum
  61. }}张
  62. </div>
  63. <el-button type="primary" @click="handleClose">关闭</el-button>
  64. <el-button
  65. style="margin-left: 10px"
  66. size="small"
  67. type="success"
  68. :loading="loading"
  69. @click="submitUpload"
  70. >
  71. 开始上传
  72. </el-button>
  73. </div>
  74. </el-dialog>
  75. </template>
  76. <script>
  77. import { tokenName } from "@/config/settings";
  78. export default {
  79. name: "VabUpload",
  80. props: {
  81. url: {
  82. type: String,
  83. default: "/upload",
  84. required: true,
  85. },
  86. name: {
  87. type: String,
  88. default: "file",
  89. required: true,
  90. },
  91. limit: {
  92. type: Number,
  93. default: 50,
  94. required: true,
  95. },
  96. size: {
  97. type: Number,
  98. default: 1,
  99. required: true,
  100. },
  101. },
  102. data() {
  103. return {
  104. show: false,
  105. loading: false,
  106. dialogVisible: false,
  107. dialogImageUrl: "",
  108. action: "",
  109. headers: {},
  110. fileList: [],
  111. picture: "picture",
  112. imgNum: 0,
  113. imgSuccessNum: 0,
  114. imgErrorNum: 0,
  115. typeList: null,
  116. title: "上传",
  117. dialogFormVisible: false,
  118. data: {},
  119. };
  120. },
  121. computed: {
  122. percentage() {
  123. if (this.allImgNum == 0) return 0;
  124. return this.$baseLodash.round(this.imgNum / this.allImgNum, 2) * 100;
  125. },
  126. },
  127. created() {
  128. if ("development" === process.env.NODE_ENV) {
  129. this.api = process.env.VUE_APP_BASE_API;
  130. } else {
  131. this.api = `${window.location.protocol}//${window.location.host}`;
  132. }
  133. this.action = this.api + this.url;
  134. this.headers[tokenName] = this.$baseAccessToken();
  135. },
  136. methods: {
  137. submitUpload() {
  138. this.$refs.upload.submit();
  139. },
  140. handleProgress(event, file, fileList) {
  141. this.loading = true;
  142. this.show = true;
  143. },
  144. handleChange(file, fileList) {
  145. if (file.size > 1048576 * this.size) {
  146. fileList.map((item, index) => {
  147. if (item === file) {
  148. fileList.splice(index, 1);
  149. }
  150. });
  151. this.fileList = fileList;
  152. } else {
  153. this.allImgNum = fileList.length;
  154. }
  155. },
  156. handleSuccess(response, file, fileList) {
  157. this.imgNum = this.imgNum + 1;
  158. this.imgSuccessNum = this.imgSuccessNum + 1;
  159. if (fileList.length === this.imgNum) {
  160. setTimeout(() => {
  161. this.$emit("fetch-datas");
  162. this.$baseMessage(
  163. `上传完成! 共上传${fileList.length}张图片`,
  164. "success"
  165. );
  166. }, 1000);
  167. }
  168. setTimeout(() => {
  169. this.loading = false;
  170. this.show = false;
  171. }, 1000);
  172. },
  173. handleError(err, file, fileList) {
  174. this.imgNum = this.imgNum + 1;
  175. this.imgErrorNum = this.imgErrorNum + 1;
  176. this.$baseMessage(
  177. `文件[${file.raw.name}]上传失败,文件大小为${this.$baseLodash.round(
  178. file.raw.size / 1024,
  179. 0
  180. )}KB`,
  181. "error"
  182. );
  183. setTimeout(() => {
  184. this.loading = false;
  185. this.show = false;
  186. }, 1000);
  187. },
  188. handleRemove(file, fileList) {
  189. this.imgNum = this.imgNum - 1;
  190. this.allNum = this.allNum - 1;
  191. },
  192. handlePreview(file) {
  193. this.dialogImageUrl = file.url;
  194. this.dialogVisible = true;
  195. },
  196. handleExceed(files, fileList) {
  197. this.$baseMessage(
  198. `当前限制选择 ${this.limit} 个文件,本次选择了
  199. ${files.length}
  200. 个文件`,
  201. "error"
  202. );
  203. },
  204. handleShow(data) {
  205. this.title = "上传";
  206. this.data = data;
  207. this.dialogFormVisible = true;
  208. },
  209. handleClose() {
  210. this.fileList = [];
  211. this.picture = "picture";
  212. this.allImgNum = 0;
  213. this.imgNum = 0;
  214. this.imgSuccessNum = 0;
  215. this.imgErrorNum = 0;
  216. if ("development" === process.env.NODE_ENV) {
  217. this.api = process.env.VUE_APP_BASE_API;
  218. } else {
  219. this.api = `${window.location.protocol}//${window.location.host}`;
  220. }
  221. this.action = this.api + this.url;
  222. this.headers[tokenName] = this.$baseAccessToken();
  223. this.dialogFormVisible = false;
  224. },
  225. },
  226. };
  227. </script>
  228. <style lang="scss" scoped>
  229. .upload {
  230. height: 600px;
  231. .upload-content {
  232. .el-upload__tip {
  233. display: block;
  234. height: 30px;
  235. line-height: 30px;
  236. }
  237. ::v-deep {
  238. .el-upload--picture-card {
  239. width: 128px;
  240. height: 128px;
  241. margin: 3px 8px 8px 8px;
  242. border: 2px dashed #c0ccda;
  243. }
  244. .el-upload-list--picture {
  245. margin-bottom: 20px;
  246. }
  247. .el-upload-list--picture-card {
  248. .el-upload-list__item {
  249. width: 128px;
  250. height: 128px;
  251. margin: 3px 8px 8px 8px;
  252. }
  253. }
  254. }
  255. }
  256. }
  257. </style>