@finema/core 1.4.152 → 1.4.154

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (168) hide show
  1. package/README.md +60 -60
  2. package/dist/module.json +1 -5
  3. package/dist/module.mjs +10 -2
  4. package/dist/runtime/components/Alert.vue +48 -48
  5. package/dist/runtime/components/Avatar.vue +27 -27
  6. package/dist/runtime/components/Badge.vue +11 -11
  7. package/dist/runtime/components/Breadcrumb.vue +44 -44
  8. package/dist/runtime/components/Button/Group.vue +37 -37
  9. package/dist/runtime/components/Button/index.vue +75 -75
  10. package/dist/runtime/components/Card.vue +38 -38
  11. package/dist/runtime/components/Core.vue +37 -37
  12. package/dist/runtime/components/Dialog/index.vue +108 -108
  13. package/dist/runtime/components/Dropdown/index.vue +70 -70
  14. package/dist/runtime/components/FlexDeck/Base.vue +143 -143
  15. package/dist/runtime/components/FlexDeck/index.vue +68 -68
  16. package/dist/runtime/components/Form/FieldWrapper.vue +23 -23
  17. package/dist/runtime/components/Form/Fields.vue +230 -230
  18. package/dist/runtime/components/Form/InputCheckbox/index.vue +28 -28
  19. package/dist/runtime/components/Form/InputDateTime/index.vue +60 -60
  20. package/dist/runtime/components/Form/InputDateTimeRange/date_range_time_field.types.d.ts +1 -1
  21. package/dist/runtime/components/Form/InputDateTimeRange/index.vue +83 -83
  22. package/dist/runtime/components/Form/InputNumber/index.vue +27 -27
  23. package/dist/runtime/components/Form/InputRadio/index.vue +27 -27
  24. package/dist/runtime/components/Form/InputSelect/index.vue +45 -45
  25. package/dist/runtime/components/Form/InputSelectMultiple/index.vue +54 -54
  26. package/dist/runtime/components/Form/InputStatic/index.vue +16 -16
  27. package/dist/runtime/components/Form/InputTags/index.vue +145 -145
  28. package/dist/runtime/components/Form/InputText/index.vue +67 -67
  29. package/dist/runtime/components/Form/InputTextarea/index.vue +25 -25
  30. package/dist/runtime/components/Form/InputToggle/index.vue +14 -14
  31. package/dist/runtime/components/Form/InputUploadDropzone/index.vue +206 -206
  32. package/dist/runtime/components/Form/InputUploadDropzone/types.d.ts +1 -1
  33. package/dist/runtime/components/Form/InputUploadDropzoneAuto/index.vue +342 -342
  34. package/dist/runtime/components/Form/InputUploadDropzoneAuto/types.d.ts +1 -1
  35. package/dist/runtime/components/Form/InputUploadDropzoneAutoMultiple/ItemUpload.vue +241 -241
  36. package/dist/runtime/components/Form/InputUploadDropzoneAutoMultiple/ItemView.vue +89 -89
  37. package/dist/runtime/components/Form/InputUploadDropzoneAutoMultiple/index.vue +164 -164
  38. package/dist/runtime/components/Form/InputUploadDropzoneAutoMultiple/types.d.ts +1 -1
  39. package/dist/runtime/components/Form/InputUploadDropzoneImageAutoMultiple/ItemUpload.vue +161 -161
  40. package/dist/runtime/components/Form/InputUploadDropzoneImageAutoMultiple/ItemView.vue +64 -64
  41. package/dist/runtime/components/Form/InputUploadDropzoneImageAutoMultiple/index.vue +172 -172
  42. package/dist/runtime/components/Form/InputUploadDropzoneImageAutoMultiple/types.d.ts +1 -1
  43. package/dist/runtime/components/Form/InputUploadFileClassic/index.vue +95 -95
  44. package/dist/runtime/components/Form/InputUploadFileClassicAuto/index.vue +151 -151
  45. package/dist/runtime/components/Form/InputUploadFileClassicAuto/types.d.ts +1 -1
  46. package/dist/runtime/components/Form/InputUploadImageAuto/index.vue +219 -219
  47. package/dist/runtime/components/Form/InputUploadImageAuto/types.d.ts +1 -1
  48. package/dist/runtime/components/Form/InputWYSIWYG/index.vue +53 -53
  49. package/dist/runtime/components/Form/index.vue +6 -6
  50. package/dist/runtime/components/Icon.vue +23 -23
  51. package/dist/runtime/components/Image.vue +36 -36
  52. package/dist/runtime/components/Loader.vue +27 -27
  53. package/dist/runtime/components/Modal/index.vue +146 -146
  54. package/dist/runtime/components/QRCode.vue +22 -22
  55. package/dist/runtime/components/SimplePagination.vue +96 -96
  56. package/dist/runtime/components/Slideover/index.vue +110 -110
  57. package/dist/runtime/components/Table/Base.vue +139 -139
  58. package/dist/runtime/components/Table/ColumnDate.vue +16 -16
  59. package/dist/runtime/components/Table/ColumnDateTime.vue +18 -18
  60. package/dist/runtime/components/Table/ColumnImage.vue +15 -15
  61. package/dist/runtime/components/Table/ColumnNumber.vue +14 -14
  62. package/dist/runtime/components/Table/ColumnText.vue +25 -25
  63. package/dist/runtime/components/Table/Simple.vue +69 -69
  64. package/dist/runtime/components/Table/index.vue +65 -65
  65. package/dist/runtime/components/Tabs/index.vue +64 -64
  66. package/dist/runtime/components/TeleportSafe.vue +40 -40
  67. package/dist/runtime/composables/loaderList.d.ts +2 -2
  68. package/dist/runtime/composables/{loaderList.js → loaderList.mjs} +2 -2
  69. package/dist/runtime/composables/loaderObject.d.ts +1 -1
  70. package/dist/runtime/composables/{loaderObject.js → loaderObject.mjs} +2 -2
  71. package/dist/runtime/composables/loaderPage.d.ts +1 -1
  72. package/dist/runtime/composables/{loaderPage.js → loaderPage.mjs} +3 -3
  73. package/dist/runtime/composables/useConfig.d.ts +1 -1
  74. package/dist/runtime/composables/useForm.d.ts +1 -1
  75. package/dist/runtime/composables/useTable.d.ts +2 -2
  76. package/dist/runtime/composables/{useTable.js → useTable.mjs} +1 -1
  77. package/dist/runtime/composables/{useUpload.js → useUpload.mjs} +1 -1
  78. package/dist/runtime/helpers/{apiListHelper.js → apiListHelper.mjs} +1 -1
  79. package/dist/runtime/helpers/{apiObjectHelper.js → apiObjectHelper.mjs} +1 -1
  80. package/dist/runtime/helpers/apiPageHelper.d.ts +1 -1
  81. package/dist/runtime/helpers/{apiPageHelper.js → apiPageHelper.mjs} +4 -4
  82. package/dist/runtime/ui.config/index.d.ts +16 -16
  83. package/dist/runtime/ui.config/index.mjs +16 -0
  84. package/dist/runtime/ui.config/{uploadDropzoneImage.js → uploadDropzoneImage.mjs} +1 -1
  85. package/dist/runtime/utils/ArrayHelper.d.ts +1 -1
  86. package/dist/runtime/utils/{ArrayHelper.spec.js → ArrayHelper.spec.mjs} +1 -1
  87. package/dist/runtime/utils/{FileHelper.spec.js → FileHelper.spec.mjs} +1 -1
  88. package/dist/runtime/utils/ObjectHelper.d.ts +2 -2
  89. package/dist/runtime/utils/{ObjectHelper.js → ObjectHelper.mjs} +2 -2
  90. package/dist/runtime/utils/{ObjectHelper.spec.js → ObjectHelper.spec.mjs} +1 -1
  91. package/dist/runtime/utils/ParamHelper.d.ts +1 -1
  92. package/dist/runtime/utils/{ParamHelper.spec.js → ParamHelper.spec.mjs} +1 -1
  93. package/dist/runtime/utils/{StringHelper.spec.js → StringHelper.spec.mjs} +1 -1
  94. package/dist/runtime/utils/{TimeHelper.spec.js → TimeHelper.spec.mjs} +1 -1
  95. package/dist/runtime/utils/{TimeHelper.thai.spec.js → TimeHelper.thai.spec.mjs} +1 -1
  96. package/dist/types.d.mts +16 -1
  97. package/dist/types.d.ts +16 -1
  98. package/package.json +92 -93
  99. package/dist/runtime/ui.config/index.js +0 -16
  100. /package/dist/runtime/components/Dropdown/{types.js → types.mjs} +0 -0
  101. /package/dist/runtime/components/FlexDeck/{types.js → types.mjs} +0 -0
  102. /package/dist/runtime/components/Form/InputCheckbox/{types.js → types.mjs} +0 -0
  103. /package/dist/runtime/components/Form/InputDateTime/{date_time_field.types.js → date_time_field.types.mjs} +0 -0
  104. /package/dist/runtime/components/Form/InputDateTimeRange/{date_range_time_field.types.js → date_range_time_field.types.mjs} +0 -0
  105. /package/dist/runtime/components/Form/InputNumber/{types.js → types.mjs} +0 -0
  106. /package/dist/runtime/components/Form/InputRadio/{types.js → types.mjs} +0 -0
  107. /package/dist/runtime/components/Form/InputSelect/{types.js → types.mjs} +0 -0
  108. /package/dist/runtime/components/Form/InputSelectMultiple/{types.js → types.mjs} +0 -0
  109. /package/dist/runtime/components/Form/InputStatic/{types.js → types.mjs} +0 -0
  110. /package/dist/runtime/components/Form/InputTags/{types.js → types.mjs} +0 -0
  111. /package/dist/runtime/components/Form/InputText/{types.js → types.mjs} +0 -0
  112. /package/dist/runtime/components/Form/InputTextarea/{types.js → types.mjs} +0 -0
  113. /package/dist/runtime/components/Form/InputToggle/{types.js → types.mjs} +0 -0
  114. /package/dist/runtime/components/Form/InputUploadDropzone/{types.js → types.mjs} +0 -0
  115. /package/dist/runtime/components/Form/InputUploadDropzoneAuto/{types.js → types.mjs} +0 -0
  116. /package/dist/runtime/components/Form/InputUploadDropzoneAutoMultiple/{types.js → types.mjs} +0 -0
  117. /package/dist/runtime/components/Form/InputUploadDropzoneImageAutoMultiple/{types.js → types.mjs} +0 -0
  118. /package/dist/runtime/components/Form/InputUploadFileClassic/{types.js → types.mjs} +0 -0
  119. /package/dist/runtime/components/Form/InputUploadFileClassicAuto/{types.js → types.mjs} +0 -0
  120. /package/dist/runtime/components/Form/InputUploadImageAuto/{types.js → types.mjs} +0 -0
  121. /package/dist/runtime/components/Form/InputWYSIWYG/{types.js → types.mjs} +0 -0
  122. /package/dist/runtime/components/Form/{types.js → types.mjs} +0 -0
  123. /package/dist/runtime/components/Table/{types.js → types.mjs} +0 -0
  124. /package/dist/runtime/composables/{useApp.js → useApp.mjs} +0 -0
  125. /package/dist/runtime/composables/{useConfig.js → useConfig.mjs} +0 -0
  126. /package/dist/runtime/composables/{useDialog.js → useDialog.mjs} +0 -0
  127. /package/dist/runtime/composables/{useFlexDeck.js → useFlexDeck.mjs} +0 -0
  128. /package/dist/runtime/composables/{useForm.js → useForm.mjs} +0 -0
  129. /package/dist/runtime/composables/{useNotification.js → useNotification.mjs} +0 -0
  130. /package/dist/runtime/composables/{useWatch.js → useWatch.mjs} +0 -0
  131. /package/dist/runtime/{core.config.js → core.config.mjs} +0 -0
  132. /package/dist/runtime/helpers/{componentHelper.js → componentHelper.mjs} +0 -0
  133. /package/dist/runtime/lib/{Requester.js → Requester.mjs} +0 -0
  134. /package/dist/runtime/{plugin.js → plugin.mjs} +0 -0
  135. /package/dist/runtime/{quill.plugin.js → quill.plugin.mjs} +0 -0
  136. /package/dist/runtime/types/{common.js → common.mjs} +0 -0
  137. /package/dist/runtime/types/{config.js → config.mjs} +0 -0
  138. /package/dist/runtime/types/{lib.js → lib.mjs} +0 -0
  139. /package/dist/runtime/ui.config/{alert.js → alert.mjs} +0 -0
  140. /package/dist/runtime/ui.config/{badge.js → badge.mjs} +0 -0
  141. /package/dist/runtime/ui.config/{breadcrumb.js → breadcrumb.mjs} +0 -0
  142. /package/dist/runtime/ui.config/{button.js → button.mjs} +0 -0
  143. /package/dist/runtime/ui.config/{buttonGroup.js → buttonGroup.mjs} +0 -0
  144. /package/dist/runtime/ui.config/{card.js → card.mjs} +0 -0
  145. /package/dist/runtime/ui.config/{checkbox.js → checkbox.mjs} +0 -0
  146. /package/dist/runtime/ui.config/{formGroup.js → formGroup.mjs} +0 -0
  147. /package/dist/runtime/ui.config/{icon.js → icon.mjs} +0 -0
  148. /package/dist/runtime/ui.config/{input.js → input.mjs} +0 -0
  149. /package/dist/runtime/ui.config/{modal.js → modal.mjs} +0 -0
  150. /package/dist/runtime/ui.config/{notification.js → notification.mjs} +0 -0
  151. /package/dist/runtime/ui.config/{pagination.js → pagination.mjs} +0 -0
  152. /package/dist/runtime/ui.config/{select.js → select.mjs} +0 -0
  153. /package/dist/runtime/ui.config/{selectMenu.js → selectMenu.mjs} +0 -0
  154. /package/dist/runtime/ui.config/{slideover.js → slideover.mjs} +0 -0
  155. /package/dist/runtime/ui.config/{table.js → table.mjs} +0 -0
  156. /package/dist/runtime/ui.config/{tabs.js → tabs.mjs} +0 -0
  157. /package/dist/runtime/ui.config/{tags.js → tags.mjs} +0 -0
  158. /package/dist/runtime/ui.config/{textarea.js → textarea.mjs} +0 -0
  159. /package/dist/runtime/ui.config/{toggle.js → toggle.mjs} +0 -0
  160. /package/dist/runtime/ui.config/{uploadFileDropzone.js → uploadFileDropzone.mjs} +0 -0
  161. /package/dist/runtime/ui.config/{uploadFileInputClassicAuto.js → uploadFileInputClassicAuto.mjs} +0 -0
  162. /package/dist/runtime/ui.config/{uploadImage.js → uploadImage.mjs} +0 -0
  163. /package/dist/runtime/utils/{ArrayHelper.js → ArrayHelper.mjs} +0 -0
  164. /package/dist/runtime/utils/{FileHelper.js → FileHelper.mjs} +0 -0
  165. /package/dist/runtime/utils/{ParamHelper.js → ParamHelper.mjs} +0 -0
  166. /package/dist/runtime/utils/{StringHelper.js → StringHelper.mjs} +0 -0
  167. /package/dist/runtime/utils/{TimeHelper.js → TimeHelper.mjs} +0 -0
  168. /package/dist/runtime/utils/{lodash.js → lodash.mjs} +0 -0
@@ -1,206 +1,206 @@
1
- <template>
2
- <FieldWrapper v-bind="wrapperProps">
3
- <div
4
- ref="dropzoneRef"
5
- :class="[
6
- ui.base,
7
- {
8
- [ui.disabled]: isDisabled,
9
- [ui.background.default]: !isOverDropZone && !isDisabled,
10
- [ui.background.dragover]: isOverDropZone && !isDisabled,
11
- },
12
- ]"
13
- >
14
- <input
15
- ref="fileInputRef"
16
- type="file"
17
- class="hidden"
18
- :name="name"
19
- :accept="acceptFile"
20
- :disabled="isDisabled"
21
- @change="handleChange"
22
- />
23
- <div :class="[ui.wrapper]">
24
- <div v-if="!selectedFile" :class="[ui.placeholderWrapper]">
25
- <Icon :name="ui.default.uploadIcon" :class="[ui.labelIcon]" />
26
- <div :class="[ui.labelWrapper]">
27
- <p class="text-primary cursor-pointer" @click="handleOpenFile">
28
- {{ selectFileLabel }}
29
- </p>
30
- <p>{{ selectFileSubLabel }}</p>
31
- </div>
32
- <p v-if="placeholder" :class="[ui.placeholder]">{{ placeholder }}</p>
33
- </div>
34
-
35
- <!-- Selected State -->
36
- <div v-if="selectedFile" :class="[ui.onPreview.wrapper]">
37
- <div :class="[ui.onPreview.previewImgWrapper]">
38
- <div v-if="isImage(selectedFile)" class="size-full overflow-hidden">
39
- <img
40
- :src="generateURL(selectedFile)"
41
- :class="[ui.onPreview.previewImgClass]"
42
- alt="image-preview"
43
- />
44
- </div>
45
- <div v-else>
46
- <Icon
47
- :name="ui.onPreview.previewFileIcon || ui.default.filePreviewIcon"
48
- :class="[ui.onPreview.previewFileClass]"
49
- />
50
- </div>
51
- </div>
52
- <div :class="[ui.onPreview.textWrapper]">
53
- <div class="truncate">
54
- <h1 class="truncate font-bold">{{ selectedFile?.name }}</h1>
55
- <p class="truncate font-light text-gray-400">
56
- {{
57
- fileAllocate.isSelectedFileUseMb.value
58
- ? `${fileAllocate.selectedFileSizeMb.value} MB`
59
- : `${fileAllocate.selectedFileSizeKb.value} KB`
60
- }}
61
- </p>
62
- </div>
63
- <div :class="[ui.action.wrapper]">
64
- <Icon
65
- v-if="isImage(selectedFile)"
66
- :name="ui.action.previewIcon"
67
- :class="[ui.action.iconClass]"
68
- @click="() => (isPreviewOpen = true)"
69
- />
70
- <Icon
71
- :name="ui.action.downloadIcon"
72
- :class="[ui.action.iconClass]"
73
- @click="handleDownloadFile"
74
- />
75
- <Icon
76
- :name="ui.action.deleteIcon"
77
- :class="[ui.action.iconClass]"
78
- @click="handleDeleteFile"
79
- />
80
- <Modal v-model="isPreviewOpen" :title="selectedFile?.name">
81
- <img :src="generateURL(selectedFile)" alt="image-preview" />
82
- </Modal>
83
- </div>
84
- </div>
85
- </div>
86
- </div>
87
- </div>
88
- </FieldWrapper>
89
- </template>
90
-
91
- <script lang="ts" setup>
92
- import { useDropZone } from '@vueuse/core'
93
- import { type IUploadDropzoneProps } from './types'
94
- import {
95
- isImage,
96
- generateURL,
97
- checkMaxSize,
98
- checkFileType,
99
- downloadFileFromURL,
100
- useFileAllocate,
101
- } from '#core/helpers/componentHelper'
102
- import FieldWrapper from '#core/components/Form/FieldWrapper.vue'
103
- import { useFieldHOC } from '#core/composables/useForm'
104
- import { computed, ref, toRef, useUI, useUiConfig } from '#imports'
105
- import { uploadFileDropzone } from '#core/ui.config'
106
- import i18next from 'i18next'
107
-
108
- const config = useUiConfig<typeof uploadFileDropzone>(uploadFileDropzone, 'uploadFileDropzone')
109
-
110
- const props = withDefaults(defineProps<IUploadDropzoneProps>(), {
111
- selectFileLabel: 'คลิกเพื่อเลือกไฟล์',
112
- selectFileSubLabel: 'หรือ ลากและวางที่นี่',
113
- })
114
-
115
- const emit = defineEmits(['change', 'delete'])
116
- const isPreviewOpen = ref<boolean>(false)
117
-
118
- const { wrapperProps, handleChange: onChange, setErrors, value } = useFieldHOC<File>(props)
119
-
120
- const { ui } = useUI('uploadFileDropzone', toRef(props, 'ui'), config)
121
-
122
- const fileInputRef = ref<HTMLInputElement>()
123
- const dropzoneRef = ref<HTMLDivElement>()
124
-
125
- const acceptFile = computed(() =>
126
- typeof props.accept === 'string' ? props.accept : props.accept?.join(',')
127
- )
128
-
129
- const selectedFile = computed(() => value?.value)
130
- const fileAllocate = useFileAllocate(selectedFile, props)
131
-
132
- const onDrop = (files: File[] | null) => {
133
- if (props.isDisabled || files?.length === 0 || !files) return
134
-
135
- const file = files[0]
136
- const result = handleCheckFileCondition(file)
137
-
138
- if (result) {
139
- onChange(file)
140
- emit('change', file)
141
- }
142
- }
143
-
144
- const { isOverDropZone } = useDropZone(dropzoneRef as unknown as HTMLElement, {
145
- onDrop,
146
- })
147
-
148
- const handleChange = (e: Event) => {
149
- if (props.isDisabled) return
150
-
151
- const file = (e.target as HTMLInputElement).files?.[0]
152
- const result = handleCheckFileCondition(file)
153
-
154
- if (result && file) {
155
- onChange(file)
156
- value.value = file
157
- emit('change', file)
158
- }
159
- }
160
-
161
- const handleOpenFile = () => {
162
- fileInputRef.value?.click()
163
- }
164
-
165
- const handleDeleteFile = () => {
166
- fileInputRef.value?.value && (fileInputRef.value.value = '')
167
- onChange(undefined)
168
- emit('delete')
169
- }
170
-
171
- const handleDownloadFile = () => {
172
- downloadFileFromURL(generateURL(selectedFile.value), selectedFile.value?.name)
173
- }
174
-
175
- const handleCheckFileCondition = (file: File | undefined): boolean => {
176
- if (!file) return false
177
-
178
- const fileType = checkFileType(file, acceptFile.value ?? '')
179
-
180
- if (!fileType) {
181
- setErrors(i18next.t('custom:invalid_file_type'))
182
-
183
- return false
184
- }
185
-
186
- const maxSize = checkMaxSize(file, fileAllocate.acceptFileSizeKb.value)
187
-
188
- if (!maxSize) {
189
- if (fileAllocate.isAcceptFileUseMb.value) {
190
- setErrors(
191
- i18next.t('custom:invalid_file_size_mb', { size: fileAllocate.acceptFileSizeMb.value })
192
- )
193
- } else {
194
- setErrors(
195
- i18next.t('custom:invalid_file_size_kb', { size: fileAllocate.acceptFileSizeKb.value })
196
- )
197
- }
198
-
199
- return false
200
- }
201
-
202
- setErrors('')
203
-
204
- return true
205
- }
206
- </script>
1
+ <template>
2
+ <FieldWrapper v-bind="wrapperProps">
3
+ <div
4
+ ref="dropzoneRef"
5
+ :class="[
6
+ ui.base,
7
+ {
8
+ [ui.disabled]: isDisabled,
9
+ [ui.background.default]: !isOverDropZone && !isDisabled,
10
+ [ui.background.dragover]: isOverDropZone && !isDisabled,
11
+ },
12
+ ]"
13
+ >
14
+ <input
15
+ ref="fileInputRef"
16
+ type="file"
17
+ class="hidden"
18
+ :name="name"
19
+ :accept="acceptFile"
20
+ :disabled="isDisabled"
21
+ @change="handleChange"
22
+ />
23
+ <div :class="[ui.wrapper]">
24
+ <div v-if="!selectedFile" :class="[ui.placeholderWrapper]">
25
+ <Icon :name="ui.default.uploadIcon" :class="[ui.labelIcon]" />
26
+ <div :class="[ui.labelWrapper]">
27
+ <p class="text-primary cursor-pointer" @click="handleOpenFile">
28
+ {{ selectFileLabel }}
29
+ </p>
30
+ <p>{{ selectFileSubLabel }}</p>
31
+ </div>
32
+ <p v-if="placeholder" :class="[ui.placeholder]">{{ placeholder }}</p>
33
+ </div>
34
+
35
+ <!-- Selected State -->
36
+ <div v-if="selectedFile" :class="[ui.onPreview.wrapper]">
37
+ <div :class="[ui.onPreview.previewImgWrapper]">
38
+ <div v-if="isImage(selectedFile)" class="size-full overflow-hidden">
39
+ <img
40
+ :src="generateURL(selectedFile)"
41
+ :class="[ui.onPreview.previewImgClass]"
42
+ alt="image-preview"
43
+ />
44
+ </div>
45
+ <div v-else>
46
+ <Icon
47
+ :name="ui.onPreview.previewFileIcon || ui.default.filePreviewIcon"
48
+ :class="[ui.onPreview.previewFileClass]"
49
+ />
50
+ </div>
51
+ </div>
52
+ <div :class="[ui.onPreview.textWrapper]">
53
+ <div class="truncate">
54
+ <h1 class="truncate font-bold">{{ selectedFile?.name }}</h1>
55
+ <p class="truncate font-light text-gray-400">
56
+ {{
57
+ fileAllocate.isSelectedFileUseMb.value
58
+ ? `${fileAllocate.selectedFileSizeMb.value} MB`
59
+ : `${fileAllocate.selectedFileSizeKb.value} KB`
60
+ }}
61
+ </p>
62
+ </div>
63
+ <div :class="[ui.action.wrapper]">
64
+ <Icon
65
+ v-if="isImage(selectedFile)"
66
+ :name="ui.action.previewIcon"
67
+ :class="[ui.action.iconClass]"
68
+ @click="() => (isPreviewOpen = true)"
69
+ />
70
+ <Icon
71
+ :name="ui.action.downloadIcon"
72
+ :class="[ui.action.iconClass]"
73
+ @click="handleDownloadFile"
74
+ />
75
+ <Icon
76
+ :name="ui.action.deleteIcon"
77
+ :class="[ui.action.iconClass]"
78
+ @click="handleDeleteFile"
79
+ />
80
+ <Modal v-model="isPreviewOpen" :title="selectedFile?.name">
81
+ <img :src="generateURL(selectedFile)" alt="image-preview" />
82
+ </Modal>
83
+ </div>
84
+ </div>
85
+ </div>
86
+ </div>
87
+ </div>
88
+ </FieldWrapper>
89
+ </template>
90
+
91
+ <script lang="ts" setup>
92
+ import { useDropZone } from '@vueuse/core'
93
+ import { type IUploadDropzoneProps } from './types'
94
+ import {
95
+ isImage,
96
+ generateURL,
97
+ checkMaxSize,
98
+ checkFileType,
99
+ downloadFileFromURL,
100
+ useFileAllocate,
101
+ } from '#core/helpers/componentHelper'
102
+ import FieldWrapper from '#core/components/Form/FieldWrapper.vue'
103
+ import { useFieldHOC } from '#core/composables/useForm'
104
+ import { computed, ref, toRef, useUI, useUiConfig } from '#imports'
105
+ import { uploadFileDropzone } from '#core/ui.config'
106
+ import i18next from 'i18next'
107
+
108
+ const config = useUiConfig<typeof uploadFileDropzone>(uploadFileDropzone, 'uploadFileDropzone')
109
+
110
+ const props = withDefaults(defineProps<IUploadDropzoneProps>(), {
111
+ selectFileLabel: 'คลิกเพื่อเลือกไฟล์',
112
+ selectFileSubLabel: 'หรือ ลากและวางที่นี่',
113
+ })
114
+
115
+ const emit = defineEmits(['change', 'delete'])
116
+ const isPreviewOpen = ref<boolean>(false)
117
+
118
+ const { wrapperProps, handleChange: onChange, setErrors, value } = useFieldHOC<File>(props)
119
+
120
+ const { ui } = useUI('uploadFileDropzone', toRef(props, 'ui'), config)
121
+
122
+ const fileInputRef = ref<HTMLInputElement>()
123
+ const dropzoneRef = ref<HTMLDivElement>()
124
+
125
+ const acceptFile = computed(() =>
126
+ typeof props.accept === 'string' ? props.accept : props.accept?.join(',')
127
+ )
128
+
129
+ const selectedFile = computed(() => value?.value)
130
+ const fileAllocate = useFileAllocate(selectedFile, props)
131
+
132
+ const onDrop = (files: File[] | null) => {
133
+ if (props.isDisabled || files?.length === 0 || !files) return
134
+
135
+ const file = files[0]
136
+ const result = handleCheckFileCondition(file)
137
+
138
+ if (result) {
139
+ onChange(file)
140
+ emit('change', file)
141
+ }
142
+ }
143
+
144
+ const { isOverDropZone } = useDropZone(dropzoneRef as unknown as HTMLElement, {
145
+ onDrop,
146
+ })
147
+
148
+ const handleChange = (e: Event) => {
149
+ if (props.isDisabled) return
150
+
151
+ const file = (e.target as HTMLInputElement).files?.[0]
152
+ const result = handleCheckFileCondition(file)
153
+
154
+ if (result && file) {
155
+ onChange(file)
156
+ value.value = file
157
+ emit('change', file)
158
+ }
159
+ }
160
+
161
+ const handleOpenFile = () => {
162
+ fileInputRef.value?.click()
163
+ }
164
+
165
+ const handleDeleteFile = () => {
166
+ fileInputRef.value?.value && (fileInputRef.value.value = '')
167
+ onChange(undefined)
168
+ emit('delete')
169
+ }
170
+
171
+ const handleDownloadFile = () => {
172
+ downloadFileFromURL(generateURL(selectedFile.value), selectedFile.value?.name)
173
+ }
174
+
175
+ const handleCheckFileCondition = (file: File | undefined): boolean => {
176
+ if (!file) return false
177
+
178
+ const fileType = checkFileType(file, acceptFile.value ?? '')
179
+
180
+ if (!fileType) {
181
+ setErrors(i18next.t('custom:invalid_file_type'))
182
+
183
+ return false
184
+ }
185
+
186
+ const maxSize = checkMaxSize(file, fileAllocate.acceptFileSizeKb.value)
187
+
188
+ if (!maxSize) {
189
+ if (fileAllocate.isAcceptFileUseMb.value) {
190
+ setErrors(
191
+ i18next.t('custom:invalid_file_size_mb', { size: fileAllocate.acceptFileSizeMb.value })
192
+ )
193
+ } else {
194
+ setErrors(
195
+ i18next.t('custom:invalid_file_size_kb', { size: fileAllocate.acceptFileSizeKb.value })
196
+ )
197
+ }
198
+
199
+ return false
200
+ }
201
+
202
+ setErrors('')
203
+
204
+ return true
205
+ }
206
+ </script>
@@ -1,4 +1,4 @@
1
- import { type IFieldProps, type IFormFieldBase, type INPUT_TYPES } from '../types.js';
1
+ import { type IFieldProps, type IFormFieldBase, type INPUT_TYPES } from '../types';
2
2
  export interface IUploadDropzoneProps extends IFieldProps {
3
3
  selectFileLabel?: string;
4
4
  selectFileSubLabel?: string;