UploadImages.vue 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309
  1. <template>
  2. <div class="upload-container">
  3. <div v-for="(image, index) in uploadedImages" :key="index" class="image-preview">
  4. <el-image
  5. :src="image.url"
  6. :zoom-rate="1.2"
  7. :max-scale="7"
  8. :min-scale="0.2"
  9. :preview-src-list="uploadedImages.map((item) => item.url)"
  10. show-progress
  11. :initial-index="index"
  12. fit="contain"
  13. />
  14. <div class="delete-icon" @click="removeImage(index)">×</div>
  15. </div>
  16. <div
  17. v-show="!isMaximum"
  18. class="upload-box"
  19. :class="{ dragging: isDragging }"
  20. @click="!isMaximum && triggerUpload()"
  21. @drop.prevent="onDrop"
  22. @dragover.prevent
  23. @dragenter.prevent="onDragEnter"
  24. @dragleave.prevent="onDragLeave"
  25. >
  26. <input
  27. ref="fileInput"
  28. type="file"
  29. accept=".jpg,.jpeg,.png"
  30. multiple
  31. @change="handleFileChange"
  32. style="display: none"
  33. :disabled="isMaximum"
  34. />
  35. <div class="upload-icons">
  36. <div class="plus-icon">
  37. <el-icon><Plus /></el-icon>
  38. </div>
  39. <div class="camera-icon">
  40. <el-icon><Camera /></el-icon>
  41. </div>
  42. </div>
  43. </div>
  44. </div>
  45. </template>
  46. <script lang="ts" setup>
  47. import { ref, reactive, computed, watch, onUnmounted } from 'vue';
  48. import { Camera, Plus } from '@element-plus/icons-vue';
  49. import { ElMessage, ElMessageBox } from 'element-plus';
  50. import type { ImageItem } from '@/types/disaster-control';
  51. const props = defineProps({
  52. maxCount: {
  53. type: Number,
  54. default: 9,
  55. },
  56. maxSize: {
  57. type: Number,
  58. default: 10, // 默认10MB
  59. },
  60. acceptTypes: {
  61. type: Array as () => string[],
  62. default: () => ['.jpg', '.jpeg', '.png'],
  63. },
  64. imageList: {
  65. type: Array as () => string[],
  66. },
  67. });
  68. const emit = defineEmits(['uploadSuccess']);
  69. const fileInput = ref<HTMLInputElement | null>(null);
  70. const uploadedImages = reactive<ImageItem[]>([]);
  71. // 计算属性:是否已达到最大上传数量
  72. const isMaximum = computed(() => {
  73. return uploadedImages.length >= props.maxCount;
  74. });
  75. const isDragging = ref(false);
  76. const onDragEnter = () => {
  77. if (!isMaximum.value) {
  78. isDragging.value = true;
  79. }
  80. };
  81. const onDragLeave = () => {
  82. isDragging.value = false;
  83. };
  84. const onDrop = (e: DragEvent) => {
  85. isDragging.value = false;
  86. if (!isMaximum.value && e.dataTransfer?.files) {
  87. handleDrop(e);
  88. }
  89. };
  90. const triggerUpload = () => {
  91. fileInput.value?.click();
  92. };
  93. const handleFileChange = (e: Event) => {
  94. const target = e.target as HTMLInputElement;
  95. if (target.files) {
  96. addFiles(Array.from(target.files));
  97. }
  98. };
  99. const handleDrop = (e: DragEvent) => {
  100. if (e.dataTransfer?.files) {
  101. addFiles(Array.from(e.dataTransfer.files));
  102. }
  103. };
  104. // 检查文件是否已存在
  105. const isFileExist = (file: File): boolean => {
  106. // 通过文件名和文件大小判断是否为同一文件
  107. return uploadedImages.some((item) => item.name === file.name && item.size === file.size);
  108. };
  109. const checkFileValid = (file: File): boolean => {
  110. // 检查文件是否已上传
  111. if (isFileExist(file)) {
  112. ElMessage.warning(`文件"${file.name}"已上传过了`);
  113. return false;
  114. }
  115. // 检查文件类型
  116. const type = file.type.toLowerCase();
  117. const isValidType = type.includes('jpeg') || type.includes('jpg') || type.includes('png');
  118. if (!isValidType) {
  119. ElMessage.error(`文件"${file.name}"格式不正确,请上传jpg、png或jpeg格式的图片`);
  120. return false;
  121. }
  122. // 检查文件大小
  123. const sizeInMB = file.size / (1024 * 1024);
  124. if (sizeInMB > props.maxSize) {
  125. ElMessage.error(`文件"${file.name}"大小超过${props.maxSize}MB限制`);
  126. return false;
  127. }
  128. return true;
  129. };
  130. const addFiles = (files: File[]) => {
  131. // 过滤出有效的图片文件
  132. const validFiles = files.filter(checkFileValid);
  133. // 检查是否超过最大数量限制
  134. const remainingSlots = props.maxCount - uploadedImages.length;
  135. if (validFiles.length > remainingSlots) {
  136. ElMessage.warning(`最多只能上传${props.maxCount}张图片`);
  137. }
  138. const filesToAdd = validFiles.slice(0, remainingSlots);
  139. // 添加到已上传列表
  140. filesToAdd.forEach((file) => {
  141. const url = URL.createObjectURL(file);
  142. uploadedImages.push({
  143. file,
  144. url,
  145. name: file.name,
  146. size: file.size,
  147. });
  148. });
  149. if (fileInput.value!.files!.length > 0) {
  150. // 清空文件输入域
  151. fileInput.value!.value = '';
  152. }
  153. // 触发更新事件
  154. if (filesToAdd.length > 0) {
  155. emitChange();
  156. }
  157. };
  158. const removeImage = (index: number) => {
  159. // 释放对象URL以避免内存泄漏
  160. URL.revokeObjectURL(uploadedImages[index].url);
  161. uploadedImages.splice(index, 1);
  162. ElMessage.success('删除成功');
  163. emitChange();
  164. };
  165. const removeAllImages = () => {
  166. uploadedImages.forEach((item) => {
  167. URL.revokeObjectURL(item.url);
  168. });
  169. uploadedImages.length = 0;
  170. };
  171. const getUploadedImages = () => {
  172. return uploadedImages;
  173. };
  174. defineExpose({
  175. removeAllImages,
  176. getUploadedImages,
  177. });
  178. const emitChange = () => {
  179. const files = uploadedImages.filter((item) => item.file).map((item) => item.file);
  180. emit('uploadSuccess', files);
  181. };
  182. watch(
  183. () => props.imageList,
  184. (newVal) => {
  185. if (!newVal) return;
  186. const urlList = newVal.map((item) => {
  187. return {
  188. url: item,
  189. };
  190. });
  191. uploadedImages.push(...urlList);
  192. },
  193. {
  194. immediate: true,
  195. },
  196. );
  197. onUnmounted(() => {
  198. removeAllImages();
  199. });
  200. </script>
  201. <style lang="scss" scoped>
  202. .upload-container {
  203. display: flex;
  204. flex-wrap: wrap;
  205. gap: 5px;
  206. width: 100%;
  207. }
  208. .upload-box,
  209. .image-preview {
  210. width: 100px;
  211. height: 100px;
  212. border-radius: 4px;
  213. }
  214. .upload-box {
  215. @include flex-center;
  216. border: 1px dashed rgba($text-color, 0.15);
  217. cursor: pointer;
  218. transition: all 0.3s;
  219. &:hover {
  220. border-color: $primary-color;
  221. }
  222. &.dragging {
  223. border-color: $primary-color;
  224. border-width: 2px;
  225. background-color: rgba($primary-color, 0.05);
  226. transform: scale(1.02);
  227. box-shadow: 0 0 8px rgba($primary-color, 0.3);
  228. }
  229. }
  230. .upload-icons {
  231. @include flex-center;
  232. flex-direction: column;
  233. gap: 5px;
  234. color: #b5b8be;
  235. .plus-icon {
  236. @include flex-center;
  237. width: 26px;
  238. height: 26px;
  239. font-size: 20px;
  240. }
  241. .camera-icon {
  242. @include flex-center;
  243. width: 35px;
  244. height: 30px;
  245. font-size: 30px;
  246. }
  247. }
  248. .image-preview {
  249. position: relative;
  250. overflow: hidden;
  251. .el-image {
  252. width: 100%;
  253. height: 100%;
  254. }
  255. .delete-icon {
  256. position: absolute;
  257. top: 0;
  258. right: 0;
  259. width: 20px;
  260. height: 20px;
  261. display: flex;
  262. justify-content: center;
  263. align-items: center;
  264. background-color: rgba(0, 0, 0, 0.5);
  265. color: white;
  266. cursor: pointer;
  267. font-size: 16px;
  268. border-radius: 0 0 0 4px;
  269. }
  270. }
  271. </style>