UploadImages.vue 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304
  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. // 触发更新事件
  150. if (filesToAdd.length > 0) {
  151. emitChange();
  152. }
  153. };
  154. const removeImage = (index: number) => {
  155. // 释放对象URL以避免内存泄漏
  156. URL.revokeObjectURL(uploadedImages[index].url);
  157. uploadedImages.splice(index, 1);
  158. ElMessage.success('删除成功');
  159. emitChange();
  160. };
  161. const removeAllImages = () => {
  162. uploadedImages.forEach((item) => {
  163. URL.revokeObjectURL(item.url);
  164. });
  165. uploadedImages.length = 0;
  166. };
  167. const getUploadedImages = () => {
  168. return uploadedImages;
  169. };
  170. defineExpose({
  171. removeAllImages,
  172. getUploadedImages,
  173. });
  174. const emitChange = () => {
  175. const files = uploadedImages.filter((item) => item.file).map((item) => item.file);
  176. emit('uploadSuccess', files);
  177. };
  178. watch(
  179. () => props.imageList,
  180. (newVal) => {
  181. if (!newVal) return;
  182. const urlList = newVal.map((item) => {
  183. return {
  184. url: item,
  185. };
  186. });
  187. uploadedImages.push(...urlList);
  188. },
  189. {
  190. immediate: true,
  191. },
  192. );
  193. onUnmounted(() => {
  194. removeAllImages();
  195. });
  196. </script>
  197. <style lang="scss" scoped>
  198. .upload-container {
  199. display: flex;
  200. flex-wrap: wrap;
  201. gap: 5px;
  202. width: 100%;
  203. }
  204. .upload-box,
  205. .image-preview {
  206. width: 100px;
  207. height: 100px;
  208. border-radius: 4px;
  209. }
  210. .upload-box {
  211. @include flex-center;
  212. border: 1px dashed rgba($text-color, 0.15);
  213. cursor: pointer;
  214. transition: all 0.3s;
  215. &:hover {
  216. border-color: $primary-color;
  217. }
  218. &.dragging {
  219. border-color: $primary-color;
  220. border-width: 2px;
  221. background-color: rgba($primary-color, 0.05);
  222. transform: scale(1.02);
  223. box-shadow: 0 0 8px rgba($primary-color, 0.3);
  224. }
  225. }
  226. .upload-icons {
  227. @include flex-center;
  228. flex-direction: column;
  229. gap: 5px;
  230. color: #b5b8be;
  231. .plus-icon {
  232. @include flex-center;
  233. width: 26px;
  234. height: 26px;
  235. font-size: 20px;
  236. }
  237. .camera-icon {
  238. @include flex-center;
  239. width: 35px;
  240. height: 30px;
  241. font-size: 30px;
  242. }
  243. }
  244. .image-preview {
  245. position: relative;
  246. overflow: hidden;
  247. .el-image {
  248. width: 100%;
  249. height: 100%;
  250. }
  251. .delete-icon {
  252. position: absolute;
  253. top: 0;
  254. right: 0;
  255. width: 20px;
  256. height: 20px;
  257. display: flex;
  258. justify-content: center;
  259. align-items: center;
  260. background-color: rgba(0, 0, 0, 0.5);
  261. color: white;
  262. cursor: pointer;
  263. font-size: 16px;
  264. border-radius: 0 0 0 4px;
  265. }
  266. }
  267. </style>