ImageUploader.vue 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123
  1. <template>
  2. <div class="image-uploader">
  3. <div class="upload-grid">
  4. <div
  5. v-for="(img, i) in images"
  6. :key="i"
  7. class="upload-item"
  8. >
  9. <van-image :src="img" width="100%" height="100%" fit="cover" radius="6" />
  10. <van-icon
  11. name="cross"
  12. class="remove-btn"
  13. @click="removeImage(i)"
  14. />
  15. </div>
  16. <div
  17. v-if="images.length < max"
  18. class="upload-item upload-add"
  19. @click="showAction = true"
  20. >
  21. <van-icon name="photograph" size="24" color="#999" />
  22. <span class="add-text">拍照</span>
  23. </div>
  24. </div>
  25. <van-action-sheet
  26. v-model:show="showAction"
  27. :actions="actions"
  28. cancel-text="取消"
  29. @select="onSelect"
  30. />
  31. </div>
  32. </template>
  33. <script setup lang="ts">
  34. import { ref, computed } from 'vue'
  35. import { showToast } from 'vant'
  36. import { takePhoto, pickFromAlbum } from '@/utils/camera'
  37. import { uploadImage } from '@/api/upload'
  38. const props = withDefaults(defineProps<{
  39. modelValue: string[]
  40. max?: number
  41. }>(), { max: 9 })
  42. const emit = defineEmits<{ 'update:modelValue': [v: string[]] }>()
  43. const showAction = ref(false)
  44. const uploading = ref(false)
  45. const actions = [
  46. { name: 'camera', description: '拍摄照片' },
  47. { name: 'album', description: '从相册选择' },
  48. ]
  49. const images = computed(() => props.modelValue)
  50. function removeImage(i: number) {
  51. const arr = [...props.modelValue]
  52. arr.splice(i, 1)
  53. emit('update:modelValue', arr)
  54. }
  55. async function onSelect(action: { name: string }) {
  56. showAction.value = false
  57. if (uploading.value) return
  58. uploading.value = true
  59. try {
  60. const files = action.name === 'camera'
  61. ? [await takePhoto()]
  62. : await pickFromAlbum(props.max - props.modelValue.length > 1)
  63. showToast('上传中...')
  64. const urls = await Promise.all(files.map(uploadImage))
  65. emit('update:modelValue', [...props.modelValue, ...urls])
  66. } catch (err: any) {
  67. if (err.message !== '未选择图片') {
  68. showToast(err.message || '操作失败')
  69. }
  70. } finally {
  71. uploading.value = false
  72. }
  73. }
  74. </script>
  75. <style scoped>
  76. .upload-grid {
  77. display: flex;
  78. flex-wrap: wrap;
  79. gap: 8px;
  80. }
  81. .upload-item {
  82. position: relative;
  83. width: calc((100% - 24px) / 4);
  84. aspect-ratio: 1;
  85. }
  86. .upload-add {
  87. display: flex;
  88. flex-direction: column;
  89. align-items: center;
  90. justify-content: center;
  91. border: 1px dashed #ddd;
  92. border-radius: 6px;
  93. background: #fafafa;
  94. cursor: pointer;
  95. gap: 2px;
  96. }
  97. .add-text {
  98. font-size: 11px;
  99. color: #999;
  100. }
  101. .remove-btn {
  102. position: absolute;
  103. top: -6px;
  104. right: -6px;
  105. background: rgba(0,0,0,0.5);
  106. color: #fff;
  107. border-radius: 50%;
  108. padding: 2px;
  109. font-size: 12px;
  110. }
  111. </style>