adminforth 2.4.0-next.3 → 2.4.0-next.300

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 (206) hide show
  1. package/commands/callTsProxy.js +14 -4
  2. package/commands/cli.js +12 -4
  3. package/commands/createApp/templates/api.ts.hbs +10 -0
  4. package/commands/createApp/templates/custom/tsconfig.json.hbs +2 -3
  5. package/commands/createApp/templates/index.ts.hbs +13 -5
  6. package/commands/createApp/templates/package.json.hbs +1 -1
  7. package/commands/createApp/utils.js +45 -7
  8. package/commands/createCustomComponent/configLoader.js +17 -4
  9. package/commands/createCustomComponent/configUpdater.js +25 -21
  10. package/commands/createCustomComponent/fileGenerator.js +1 -1
  11. package/commands/createCustomComponent/main.js +15 -8
  12. package/commands/createCustomComponent/templates/customCrud/beforeActionButtons.vue.hbs +38 -0
  13. package/commands/createCustomComponent/templates/customCrud/saveButton.vue.hbs +28 -0
  14. package/commands/createCustomComponent/templates/login/beforeLogin.vue.hbs +18 -0
  15. package/commands/createPlugin/templates/custom/tsconfig.json.hbs +2 -5
  16. package/commands/createPlugin/templates/package.json.hbs +1 -1
  17. package/commands/generateModels.js +30 -22
  18. package/dist/auth.d.ts +9 -1
  19. package/dist/auth.d.ts.map +1 -1
  20. package/dist/auth.js +21 -2
  21. package/dist/auth.js.map +1 -1
  22. package/dist/dataConnectors/baseConnector.d.ts +1 -1
  23. package/dist/dataConnectors/baseConnector.d.ts.map +1 -1
  24. package/dist/dataConnectors/baseConnector.js +69 -17
  25. package/dist/dataConnectors/baseConnector.js.map +1 -1
  26. package/dist/dataConnectors/clickhouse.d.ts.map +1 -1
  27. package/dist/dataConnectors/clickhouse.js +15 -0
  28. package/dist/dataConnectors/clickhouse.js.map +1 -1
  29. package/dist/dataConnectors/mongo.d.ts.map +1 -1
  30. package/dist/dataConnectors/mongo.js +50 -15
  31. package/dist/dataConnectors/mongo.js.map +1 -1
  32. package/dist/dataConnectors/mysql.d.ts.map +1 -1
  33. package/dist/dataConnectors/mysql.js +11 -0
  34. package/dist/dataConnectors/mysql.js.map +1 -1
  35. package/dist/dataConnectors/postgres.d.ts.map +1 -1
  36. package/dist/dataConnectors/postgres.js +43 -14
  37. package/dist/dataConnectors/postgres.js.map +1 -1
  38. package/dist/dataConnectors/sqlite.d.ts.map +1 -1
  39. package/dist/dataConnectors/sqlite.js +11 -0
  40. package/dist/dataConnectors/sqlite.js.map +1 -1
  41. package/dist/index.d.ts +12 -2
  42. package/dist/index.d.ts.map +1 -1
  43. package/dist/index.js +42 -21
  44. package/dist/index.js.map +1 -1
  45. package/dist/modules/codeInjector.d.ts +2 -0
  46. package/dist/modules/codeInjector.d.ts.map +1 -1
  47. package/dist/modules/codeInjector.js +67 -12
  48. package/dist/modules/codeInjector.js.map +1 -1
  49. package/dist/modules/configValidator.d.ts +6 -0
  50. package/dist/modules/configValidator.d.ts.map +1 -1
  51. package/dist/modules/configValidator.js +203 -25
  52. package/dist/modules/configValidator.js.map +1 -1
  53. package/dist/modules/restApi.d.ts +1 -1
  54. package/dist/modules/restApi.d.ts.map +1 -1
  55. package/dist/modules/restApi.js +172 -31
  56. package/dist/modules/restApi.js.map +1 -1
  57. package/dist/modules/styles.d.ts +499 -13
  58. package/dist/modules/styles.d.ts.map +1 -1
  59. package/dist/modules/styles.js +555 -31
  60. package/dist/modules/styles.js.map +1 -1
  61. package/dist/modules/utils.d.ts +7 -15
  62. package/dist/modules/utils.d.ts.map +1 -1
  63. package/dist/modules/utils.js +45 -68
  64. package/dist/modules/utils.js.map +1 -1
  65. package/dist/servers/express.d.ts +5 -0
  66. package/dist/servers/express.d.ts.map +1 -1
  67. package/dist/servers/express.js +40 -1
  68. package/dist/servers/express.js.map +1 -1
  69. package/dist/spa/index.html +1 -1
  70. package/dist/spa/package-lock.json +1208 -708
  71. package/dist/spa/package.json +34 -34
  72. package/dist/spa/src/App.vue +58 -173
  73. package/dist/spa/src/adminforth.ts +42 -18
  74. package/dist/spa/src/afcl/AreaChart.vue +0 -1
  75. package/dist/spa/src/afcl/BarChart.vue +2 -2
  76. package/dist/spa/src/afcl/Button.vue +6 -6
  77. package/dist/spa/src/afcl/ButtonGroup.vue +91 -0
  78. package/dist/spa/src/afcl/Card.vue +25 -0
  79. package/dist/spa/src/afcl/Checkbox.vue +21 -13
  80. package/dist/spa/src/afcl/CountryFlag.vue +4 -1
  81. package/dist/spa/src/{components/CustomDatePicker.vue → afcl/DatePicker.vue} +95 -9
  82. package/dist/spa/src/afcl/Dialog.vue +47 -27
  83. package/dist/spa/src/afcl/Dropzone.vue +103 -44
  84. package/dist/spa/src/afcl/Input.vue +18 -8
  85. package/dist/spa/src/afcl/JsonViewer.vue +25 -0
  86. package/dist/spa/src/afcl/Link.vue +1 -1
  87. package/dist/spa/src/afcl/LinkButton.vue +3 -3
  88. package/dist/spa/src/afcl/PieChart.vue +5 -5
  89. package/dist/spa/src/afcl/ProgressBar.vue +7 -7
  90. package/dist/spa/src/afcl/Select.vue +82 -34
  91. package/dist/spa/src/afcl/Skeleton.vue +6 -6
  92. package/dist/spa/src/afcl/Table.vue +309 -73
  93. package/dist/spa/src/afcl/Textarea.vue +31 -0
  94. package/dist/spa/src/afcl/Toggle.vue +32 -0
  95. package/dist/spa/src/afcl/Tooltip.vue +28 -18
  96. package/dist/spa/src/afcl/VerticalTabs.vue +16 -7
  97. package/dist/spa/src/afcl/index.ts +6 -3
  98. package/dist/spa/src/components/AcceptModal.vue +48 -14
  99. package/dist/spa/src/components/Breadcrumbs.vue +5 -5
  100. package/dist/spa/src/components/CallActionWrapper.vue +15 -0
  101. package/dist/spa/src/components/ColumnValueInput.vue +38 -18
  102. package/dist/spa/src/components/ColumnValueInputWrapper.vue +4 -3
  103. package/dist/spa/src/components/CustomDateRangePicker.vue +9 -8
  104. package/dist/spa/src/components/CustomRangePicker.vue +37 -21
  105. package/dist/spa/src/components/ErrorMessage.vue +21 -0
  106. package/dist/spa/src/components/Filters.vue +194 -131
  107. package/dist/spa/src/components/GroupsTable.vue +9 -8
  108. package/dist/spa/src/components/MenuLink.vue +90 -23
  109. package/dist/spa/src/components/ResourceForm.vue +94 -51
  110. package/dist/spa/src/components/ResourceListTable.vue +120 -90
  111. package/dist/spa/src/components/ResourceListTableVirtual.vue +118 -84
  112. package/dist/spa/src/components/ShowTable.vue +21 -15
  113. package/dist/spa/src/components/Sidebar.vue +470 -0
  114. package/dist/spa/src/components/SingleSkeletLoader.vue +6 -6
  115. package/dist/spa/src/components/SkeleteLoader.vue +3 -3
  116. package/dist/spa/src/components/ThreeDotsMenu.vue +84 -15
  117. package/dist/spa/src/components/Toast.vue +40 -29
  118. package/dist/spa/src/components/UserMenuSettingsButton.vue +69 -0
  119. package/dist/spa/src/components/ValueRenderer.vue +44 -17
  120. package/dist/spa/src/controls/BoolToggle.vue +34 -0
  121. package/dist/spa/src/i18n.ts +5 -3
  122. package/dist/spa/src/main.ts +1 -1
  123. package/dist/spa/src/renderers/CompactField.vue +1 -1
  124. package/dist/spa/src/renderers/CompactUUID.vue +1 -1
  125. package/dist/spa/src/router/index.ts +8 -0
  126. package/dist/spa/src/shims-vue.d.ts +5 -0
  127. package/dist/spa/src/spa_types/core.ts +13 -1
  128. package/dist/spa/src/stores/core.ts +13 -1
  129. package/dist/spa/src/stores/filters.ts +33 -2
  130. package/dist/spa/src/stores/modal.ts +6 -1
  131. package/dist/spa/src/stores/toast.ts +22 -3
  132. package/dist/spa/src/types/Back.ts +164 -23
  133. package/dist/spa/src/types/Common.ts +92 -32
  134. package/dist/spa/src/types/FrontendAPI.ts +31 -5
  135. package/dist/spa/src/types/adapters/CaptchaAdapter.ts +34 -0
  136. package/dist/spa/src/types/adapters/CompletionAdapter.ts +25 -0
  137. package/dist/spa/src/types/adapters/EmailAdapter.ts +27 -0
  138. package/dist/spa/src/types/adapters/ImageGenerationAdapter.ts +50 -0
  139. package/dist/spa/src/types/adapters/ImageVisionAdapter.ts +30 -0
  140. package/dist/spa/src/types/adapters/KeyValueAdapter.ts +16 -0
  141. package/dist/spa/src/types/adapters/OAuth2Adapter.ts +34 -0
  142. package/dist/spa/src/types/adapters/StorageAdapter.ts +73 -0
  143. package/dist/spa/src/types/adapters/index.ts +8 -0
  144. package/dist/spa/src/utils.ts +291 -11
  145. package/dist/spa/src/views/CreateView.vue +63 -21
  146. package/dist/spa/src/views/EditView.vue +55 -22
  147. package/dist/spa/src/views/ListView.vue +144 -87
  148. package/dist/spa/src/views/LoginView.vue +60 -55
  149. package/dist/spa/src/views/ResourceParent.vue +2 -2
  150. package/dist/spa/src/views/SettingsView.vue +121 -0
  151. package/dist/spa/src/views/ShowView.vue +83 -53
  152. package/dist/spa/src/websocket.ts +6 -1
  153. package/dist/spa/tsconfig.app.json +1 -1
  154. package/dist/spa/vite.config.ts +45 -2
  155. package/dist/types/Back.d.ts +147 -14
  156. package/dist/types/Back.d.ts.map +1 -1
  157. package/dist/types/Back.js +15 -0
  158. package/dist/types/Back.js.map +1 -1
  159. package/dist/types/Common.d.ts +107 -29
  160. package/dist/types/Common.d.ts.map +1 -1
  161. package/dist/types/Common.js.map +1 -1
  162. package/dist/types/FrontendAPI.d.ts +31 -3
  163. package/dist/types/FrontendAPI.d.ts.map +1 -1
  164. package/dist/types/FrontendAPI.js.map +1 -1
  165. package/dist/types/adapters/CaptchaAdapter.d.ts +30 -0
  166. package/dist/types/adapters/CaptchaAdapter.d.ts.map +1 -0
  167. package/dist/types/adapters/CaptchaAdapter.js +5 -0
  168. package/dist/types/adapters/CaptchaAdapter.js.map +1 -0
  169. package/dist/types/adapters/CompletionAdapter.d.ts +20 -0
  170. package/dist/types/adapters/CompletionAdapter.d.ts.map +1 -0
  171. package/dist/types/adapters/CompletionAdapter.js +2 -0
  172. package/dist/types/adapters/CompletionAdapter.js.map +1 -0
  173. package/dist/types/adapters/EmailAdapter.d.ts +20 -0
  174. package/dist/types/adapters/EmailAdapter.d.ts.map +1 -0
  175. package/dist/types/adapters/EmailAdapter.js +2 -0
  176. package/dist/types/adapters/EmailAdapter.js.map +1 -0
  177. package/dist/types/adapters/ImageGenerationAdapter.d.ts +37 -0
  178. package/dist/types/adapters/ImageGenerationAdapter.d.ts.map +1 -0
  179. package/dist/types/adapters/ImageGenerationAdapter.js +2 -0
  180. package/dist/types/adapters/ImageGenerationAdapter.js.map +1 -0
  181. package/dist/types/adapters/ImageVisionAdapter.d.ts +25 -0
  182. package/dist/types/adapters/ImageVisionAdapter.d.ts.map +1 -0
  183. package/dist/types/adapters/ImageVisionAdapter.js +2 -0
  184. package/dist/types/adapters/ImageVisionAdapter.js.map +1 -0
  185. package/dist/types/adapters/KeyValueAdapter.d.ts +10 -0
  186. package/dist/types/adapters/KeyValueAdapter.d.ts.map +1 -0
  187. package/dist/types/adapters/KeyValueAdapter.js +2 -0
  188. package/dist/types/adapters/KeyValueAdapter.js.map +1 -0
  189. package/dist/types/adapters/OAuth2Adapter.d.ts +32 -0
  190. package/dist/types/adapters/OAuth2Adapter.d.ts.map +1 -0
  191. package/dist/types/adapters/OAuth2Adapter.js +2 -0
  192. package/dist/types/adapters/OAuth2Adapter.js.map +1 -0
  193. package/dist/types/adapters/StorageAdapter.d.ts +63 -0
  194. package/dist/types/adapters/StorageAdapter.d.ts.map +1 -0
  195. package/dist/types/adapters/StorageAdapter.js +2 -0
  196. package/dist/types/adapters/StorageAdapter.js.map +1 -0
  197. package/dist/types/adapters/index.d.ts +9 -0
  198. package/dist/types/adapters/index.d.ts.map +1 -0
  199. package/dist/types/adapters/index.js +2 -0
  200. package/dist/types/adapters/index.js.map +1 -0
  201. package/package.json +4 -2
  202. package/dist/spa/src/types/Adapters.ts +0 -213
  203. package/dist/types/Adapters.d.ts +0 -168
  204. package/dist/types/Adapters.d.ts.map +0 -1
  205. package/dist/types/Adapters.js +0 -2
  206. package/dist/types/Adapters.js.map +0 -1
@@ -0,0 +1,91 @@
1
+ <template>
2
+ <div v-if="isInitFinished" class="inline-flex rounded-md shadow-xs" role="group">
3
+ <button v-for="button in buttons" :key="`${button}-button-controll`"
4
+ :disabled="slotProps[button].disabled"
5
+ class="inline-flex items-center text-sm font-medium border-t border-b border-r border-lightButtonGroupBorder focus:z-10 focus:ring-2 dark:border-darkButtonGroupBorder disabled:opacity-50 disabled:cursor-not-allowed"
6
+ :class="[
7
+ buttonsStyles[button] === 'rounded' ? 'border rounded-lg'
8
+ : buttonsStyles[button] === 'rounded-left' ? 'border rounded-s-lg'
9
+ : buttonsStyles[button] === 'rounded-right' ? 'border rounded-e-lg border-l-lightButtonGroupBackground focus:border-l-lightButtonGroupBorder dark:border-l-darkButtonGroupBackground dark:focus:border-l-darkButtonGroupBorder'
10
+ : buttonsStyles[button] === 'no-rounded' ? 'border border-l-lightButtonGroupBackground focus:border-l-lightButtonGroupBorder dark:border-l-darkButtonGroupBackground dark:focus:border-l-darkButtonGroupBorder'
11
+ : buttonsStyles[button] === 'rounded-left-with-right-border' ? 'border rounded-s-lg' : '',
12
+ (button === activeButton || props.solidColor === true) ? 'bg-lightButtonGroupActiveBackground text-lightButtonGroupActiveText focus:ring-lightButtonGroupActiveFocusRing dark:bg-darkButtonGroupActiveBackground dark:text-darkButtonGroupActiveText dark:focus:ring-darkButtonGroupActiveFocusRing'
13
+ :'text-lightButtonGroupText bg-lightButtonGroupBackground focus:ring-lightButtonGroupFocusRing hover:bg-lightButtonGroupBackgroundHover hover:text-lightButtonGroupTextHover dark:bg-darkButtonGroupBackground dark:text-darkButtonGroupText dark:hover:text-darkButtonGroupTextHover dark:hover:bg-darkButtonGroupBackgroundHover dark:focus:ring-darkButtonGroupFocusRing'
14
+ ]"
15
+ @click="setActiveButton(button)"
16
+ >
17
+ <slot :name="`button:${button}`"></slot>
18
+ </button>
19
+ </div>
20
+ </template>
21
+
22
+ <script lang="ts" setup>
23
+ import { onMounted, useSlots, reactive, ref, type Ref } from 'vue';
24
+
25
+ const buttons : Ref<string[]> = ref([]);
26
+ const buttonsStyles : Ref<Record<string, string>> = ref({});
27
+ const activeButton = ref('');
28
+ const slotProps = reactive<Record<string, any>>({});
29
+ const isInitFinished = ref(false);
30
+
31
+ const emits = defineEmits(['update:modelValue']);
32
+
33
+ const props = defineProps<{
34
+ solidColor?: boolean;
35
+ modelValue?: string;
36
+ }>();
37
+
38
+ onMounted(() => {
39
+ const slots = useSlots();
40
+ buttons.value = Object.keys(slots).reduce((tbs: string[], tb: string) => {
41
+ if (tb.startsWith('button:')) {
42
+ tbs.push(tb.replace('button:', ''));
43
+ }
44
+ return tbs;
45
+ }, []);
46
+ if (buttons.value.length > 0) {
47
+ activeButton.value = buttons.value[0];
48
+ }
49
+ for (const button of buttons.value) {
50
+ const temp = {
51
+ [button]: getButtonsClasses(button)
52
+ }
53
+ buttonsStyles.value = { ...buttonsStyles.value, ...temp };
54
+ const slot = slots[`button:${button}`];
55
+ if (slot && slot()[0]?.props) {
56
+ slotProps[button] = slot()[0].props;
57
+ } else {
58
+ slotProps[button] = {};
59
+ }
60
+ if (slotProps[button]?.disabled === undefined) {
61
+ slotProps[button].disabled = false;
62
+ }
63
+ }
64
+ isInitFinished.value = true;
65
+ });
66
+
67
+
68
+ function setActiveButton(button: string) {
69
+ if (buttons.value.includes(button)) {
70
+ activeButton.value = button;
71
+ emits('update:modelValue', button);
72
+ }
73
+ }
74
+
75
+ function getButtonsClasses(button: string) {
76
+ const index = buttons.value.indexOf(button);
77
+ const lenght = buttons.value.length;
78
+ if ( lenght === 1 ) {
79
+ return 'rounded'
80
+ } else if ( lenght === 2 && index === 0 ) {
81
+ return 'rounded-left-with-right-border'
82
+ } else if ( index === 0 ) {
83
+ return 'rounded-left'
84
+ } else if ( index === lenght - 1 ) {
85
+ return 'rounded-right'
86
+ } else {
87
+ return 'no-rounded'
88
+ }
89
+ }
90
+
91
+ </script>
@@ -0,0 +1,25 @@
1
+ <template>
2
+ <a href="#"
3
+ class="block max-w-sm bg-lightCardBackground border border-lightCardBorder rounded-lg shadow-sm hover:bg-lightCardBackgroundHover dark:bg-darkCardBackground dark:border-darkCardBorder dark:hover:bg-darkCardBackgroundHover"
4
+ :class="[
5
+ props.size === 'sm' ? 'p-2' : props.size === 'md' ? 'p-4' : 'p-6',
6
+ props.hideBorder ? 'border-0' : ''
7
+ ]"
8
+ >
9
+
10
+ <h5 class="font-bold tracking-tight text-lightCardTitle dark:text-darkCardTitle" :class="[props.size === 'sm' ? 'text-base' : props.size === 'md' ? 'text-lg mb-1' : 'text-xl mb-2']">{{ props.title }}</h5>
11
+ <p class="font-normal text-lightCardDescription dark:text-darkCardDescription" :class="[props.size === 'sm' ? 'text-sm' : props.size === 'md' ? 'text-base' : 'text-lg']">{{ props.description }}</p>
12
+ <slot></slot>
13
+ </a>
14
+ </template>
15
+
16
+
17
+ <script setup lang="ts">
18
+ const props = defineProps<{
19
+ title?: string;
20
+ description?: string;
21
+ size?: 'sm' | 'md' | 'lg';
22
+ hideBorder?: boolean;
23
+ }>();
24
+
25
+ </script>
@@ -1,21 +1,29 @@
1
1
  <template>
2
- <div class="flex items-center h-5" :class="{'opacity-50' : props.disabled}">
3
- <input :id="id"
4
- ref="rememberInput"
5
- type="checkbox"
6
- :checked="props.modelValue"
7
- :disabled="props.disabled"
8
- @change="$emit('update:modelValue', $event.target.checked)"
9
- class="w-4 h-4 border border-gray-300 rounded bg-gray-50 focus:ring-3 focus:ring-lightPrimary cursor-pointer
10
- focus:ring-opacity-50 dark:bg-gray-700 dark:border-gray-600 dark:focus:ring-blue-600 dark:ring-offset-gray-800 dark:focus:ring-offset-gray-800 checked:bg-lightPrimary checked:dark:bg-darkPrimary" />
2
+ <div class="afcl-checkbox flex items-center relative h-5" :class="{'opacity-50' : props.disabled}">
3
+ <div class="flex items-center justify-center">
4
+ <input :id="id"
5
+ ref="rememberInput"
6
+ type="checkbox"
7
+ :checked="props.modelValue"
8
+ :disabled="props.disabled"
9
+ @change="$emit('update:modelValue', ($event.target as HTMLInputElement).checked)"
10
+ class="peer appearance-none min-w-4 min-h-4 bg-lightCheckboxBgUnchecked border border-lightCheckboxBorderColor rounded-sm checked:bg-lightCheckboxBgChecked
11
+ focus:ring-lightFocusRing dark:focus:ring-darkFocusRing dark:focus:ring-darkFocusRing
12
+ focus:ring-2 dark:bg-darkCheckboxBgUnchecked dark:border-darkCheckboxBorderColor dark:checked:bg-darkCheckboxBgChecked cursor-pointer"
13
+ >
14
+ <div class="pointer-events-none absolute text-lightCheckboxIconColor dark:text-darkCheckboxIconColor leading-none peer-checked:block hidden">
15
+ <IconCheckOutline class="w-[1.1rem] h-[1.1rem]" />
16
+ </div>
17
+ </div>
18
+ <label :for="id" class="ms-2 text-sm font-medium text-lightTextLabel dark:text-darkTextLabel">
19
+ <slot></slot>
20
+ </label>
21
+
11
22
  </div>
12
- <label :for="id" class="ms-2 text-sm font-medium text-gray-900 dark:text-gray-300">
13
- <slot></slot>
14
- </label>
15
23
  </template>
16
24
 
17
25
  <script setup lang="ts">
18
-
26
+ import { IconCheckOutline } from '@iconify-prerendered/vue-flowbite';
19
27
  const props = defineProps({
20
28
  modelValue: Boolean,
21
29
  disabled: Boolean,
@@ -1,6 +1,8 @@
1
1
  <template>
2
- <component v-if="getFlagComponent(countryCode)" class="flag-icon rounded-sm" :is="getFlagComponent(countryCode)" />
2
+ <div class="aspect-[4/3] w-8 h-6 flex items-center justify-center">
3
+ <component v-if="getFlagComponent(countryCode)" :is="getFlagComponent(countryCode)" class="flag-icon h-full object-contain rounded-sm"/>
3
4
  <span v-else-if="countryCode">{{ countryCode }}</span>
5
+ </div>
4
6
  </template>
5
7
 
6
8
  <script setup lang="ts">
@@ -22,5 +24,6 @@ const getFlagComponent = (countryCode: string) => {
22
24
  box-shadow: inset -0.3px -0.3px 0.3px 0px rgba(0 0 0 / 0.2),
23
25
  inset 0.3px 0.3px 0.3px 0px rgba(255 255 255 / 0.2),
24
26
  0px 0px 3px #00000030;
27
+ width: auto;
25
28
  }
26
29
  </style>
@@ -2,15 +2,15 @@
2
2
  <div>
3
3
  <div class="grid w-40 gap-4 mb-2">
4
4
  <div>
5
- <label v-if="label" for="start-time" class="block mb-2 text-sm font-medium text-gray-900 dark:text-white">{{ label }}</label>
5
+ <label v-if="label" for="start-time" class="block mb-2 text-sm font-medium text-lightDatePickerButtonText dark:text-darkDatePickerButtonText">{{ label }}</label>
6
6
 
7
7
  <div class="relative" :class="{hidden: column.type === 'time'}">
8
8
  <div class="absolute inset-y-0 end-0 top-0 flex items-center pe-3.5 pointer-events-none">
9
- <IconCalendar class="w-4 h-4 text-gray-500 dark:text-gray-400"/>
9
+ <IconCalendar class="w-4 h-4 text-lightDatePickerIcon dark:text-darkDatePickerIcon"/>
10
10
  </div>
11
11
 
12
12
  <input ref="datepickerStartEl" type="text"
13
- class="bg-gray-50 border border-gray-300 text-gray-900 text-sm rounded-lg block w-full p-2.5 dark:bg-gray-700 dark:border-gray-600 dark:placeholder-gray-400 dark:text-white focus:ring-lightPrimary focus:border-lightPrimary dark:focus:ring-darkPrimary dark:focus:border-darkPrimary"
13
+ class="af-pick-date-button bg-lightDatePickerButtonBackground border border-lightDatePickerButtonBorder text-lightDatePickerButtonText text-sm rounded-lg block w-full p-2.5 dark:bg-darkDatePickerButtonBackground dark:border-darkDatePickerButtonBorder dark:placeholder-darkInputPlaceholderText dark:text-darkDatePickerButtonText focus:ring-lightInputFocusRing focus:border-lightInputFocusBorder dark:focus:ring-darkInputFocusRing dark:focus:border-darkInputFocusBorder"
14
14
  :placeholder="$t('Select date')" :disabled="readonly" />
15
15
 
16
16
  </div>
@@ -22,18 +22,17 @@
22
22
  <div>
23
23
  <div class="relative">
24
24
  <div class="absolute inset-y-0 end-0 top-0 flex items-center pe-3.5 pointer-events-none">
25
- <IconTime class="w-4 h-4 text-gray-500 dark:text-gray-400 bg-white dark:bg-gray-700"/>
25
+ <IconTime class="w-4 h-4 text-lightDatePickerIcon dark:text-darkDatePickerIcon bg-lightDatePickerButtonBackground dark:bg-darkDatePickerButtonBackground"/>
26
26
  </div>
27
-
28
- <input v-model="startTime" type="time" id="start-time" onfocus="this.showPicker()" onclick="this.showPicker()" step="1"
29
- class="bg-gray-50 border border-gray-300 text-gray-900 text-sm rounded-lg block w-full p-2.5 dark:bg-gray-700 dark:border-gray-600 dark:placeholder-gray-400 dark:text-white focus:ring-lightPrimary focus:border-lightPrimary dark:focus:ring-darkPrimary dark:focus:border-darkPrimary"
27
+ <input v-model="startTime" type="time" id="start-time" step="1"
28
+ class="af-pick-time-button bg-lightDatePickerButtonBackground border border-lightDatePickerButtonBorder text-lightDatePickerButtonText text-sm rounded-lg block w-full p-2.5 dark:bg-darkInputBackground dark:border-darkInputBorder dark:placeholder-darkInputPlaceholderText dark:text-darkDatePickerButtonText focus:ring-lightInputFocusRing focus:border-lightInputFocusBorder dark:focus:ring-darkInputFocusRing dark:focus:border-darkInputFocusBorder"
30
29
  value="00:00" :disabled="readonly" required/>
31
30
  </div>
32
31
  </div>
33
32
  </div>
34
33
 
35
34
  <button type="button"
36
- class="text-lightPrimary dark:text-darkPrimary text-base font-medium hover:underline p-0 inline-flex items-center mb-2"
35
+ class="text-lightDatePickerExpandText dark:text-darkDatePickerExpandText text-base font-medium hover:underline p-0 inline-flex items-center mb-2"
37
36
  :class="{hidden: column.type !== 'datetime'}"
38
37
  @click="toggleTimeInputs">{{ showTimeInputs ? $t('Hide time') : $t('Show time') }}
39
38
  <svg class="w-8 h-8 ms-0.5" :class="{'rotate-180': showTimeInputs}" aria-hidden="true"
@@ -197,4 +196,91 @@ function focus() {
197
196
  defineExpose({
198
197
  focus,
199
198
  });
200
- </script>
199
+ </script>
200
+
201
+ <style lang="css" scoped>
202
+
203
+ :global(.datepicker-controls button svg),
204
+ :global(.datepicker-controls button) {
205
+ @apply text-lightDatePickerCalendarText;
206
+ }
207
+
208
+ :global( .datepicker-dropdown ),
209
+ :global( .datepicker-picker ) {
210
+ @apply !bg-lightDatePickerCalendarBackground
211
+ }
212
+
213
+ :global(.datepicker-controls button:hover) {
214
+ @apply bg-lightDatePickerCalendarArrowButtonBackgroundHover;
215
+ }
216
+
217
+ :global(.datepicker-controls button) {
218
+ @apply ring-lightDatePickerCalendarArrowButtonFocusRing bg-lightDatePickerCalendarArrowButtonBackground;
219
+ }
220
+
221
+ :global(.datepicker-controls button:focus) {
222
+ @apply ring-lightDatePickerCalendarArrowButtonFocusRing;
223
+ }
224
+
225
+ :global(.days-of-week span) {
226
+ @apply text-lightDatePickerCalendarDaysOfWeekText;
227
+ }
228
+
229
+ :global(.datepicker-grid span) {
230
+ @apply text-lightDatePickerCalendarDateButtonText;
231
+ }
232
+
233
+ :global(.datepicker-grid span:hover:not(.selected)) {
234
+ @apply bg-lightDatePickerCalendarDateButtonBackgroundHover;
235
+ }
236
+
237
+ :global(.datepicker-grid .selected ) {
238
+ @apply text-lightDatePickerCalendarDateActiveButtonText bg-lightDatePickerCalendarDateActiveButtonBackground;
239
+ }
240
+
241
+
242
+
243
+
244
+
245
+
246
+
247
+
248
+ :global(.dark .datepicker-controls button svg),
249
+ :global(.dark .datepicker-controls button) {
250
+ @apply text-darkDatePickerCalendarMainText;
251
+ }
252
+
253
+ :global(.dark .datepicker-dropdown),
254
+ :global(.dark .datepicker-picker) {
255
+ @apply !bg-darkDatePickerCalendarBackground;
256
+ }
257
+
258
+ :global(.dark .datepicker-controls button:hover) {
259
+ @apply bg-darkDatePickerCalendarArrowButtonBackgroundHover;
260
+ }
261
+
262
+ :global(.dark .datepicker-controls button) {
263
+ @apply ring-darkDatePickerCalendarArrowButtonFocusRing bg-darkDatePickerCalendarArrowButtonBackground;
264
+ }
265
+
266
+ :global(.dark .datepicker-controls button) {
267
+ @apply ring-darkDatePickerCalendarArrowButtonFocusRing;
268
+ }
269
+
270
+ :global(.dark .days-of-week span) {
271
+ @apply text-darkDatePickerCalendarDaysOfWeekText;
272
+ }
273
+
274
+ :global(.dark .datepicker-grid span) {
275
+ @apply text-darkDatePickerCalendarDateButtonText;
276
+ }
277
+
278
+ :global(.dark .datepicker-grid span:hover:not(.selected)) {
279
+ @apply bg-darkDatePickerCalendarDateButtonBackgroundHover;
280
+ }
281
+
282
+ :global(.dark .datepicker-grid .selected) {
283
+ @apply text-darkDatePickerCalendarDateActiveButtonText
284
+ bg-darkDatePickerCalendarDateActiveButtonBackground;
285
+ }
286
+ </style>
@@ -6,22 +6,22 @@
6
6
  <slot name="trigger"></slot>
7
7
  </div>
8
8
  <Teleport to="body">
9
- <div ref="modalEl" tabindex="-1" aria-hidden="true" class="hidden overflow-y-auto overflow-x-hidden fixed top-0 right-0 left-0 z-50 justify-center items-center w-full md:inset-0 h-full max-h-full">
10
- <div v-bind="$attrs" class="relative p-4 max-w-2xl max-h-full" :class="$attrs.class?.includes('w-') ? '' : 'w-full'">
9
+ <div ref="modalEl" tabindex="-1" aria-hidden="true" class="[scrollbar-gutter:stable] hidden overflow-y-auto overflow-x-hidden fixed top-0 right-0 left-0 z-50 justify-center items-center w-full md:inset-0 h-full max-h-full">
10
+ <div v-bind="$attrs" class="relative p-4 max-w-2xl max-h-full" :class="($attrs.class as string)?.includes('w-') ? '' : 'w-full'">
11
11
  <!-- Modal content -->
12
- <div class="relative bg-white rounded-lg shadow-sm dark:bg-gray-700">
12
+ <div class="relative bg-lightDialogBackgorund rounded-lg shadow-sm dark:bg-darkDialogBackgorund">
13
13
  <!-- Modal header -->
14
14
  <div
15
15
  v-if="header"
16
- class="flex items-center justify-between p-4 md:p-5 border-b rounded-t dark:border-gray-600 border-gray-200"
16
+ class="flex items-center justify-between p-4 md:p-5 border-b rounded-t dark:border-darkDialogBreakLine border-lightDialogBreakLine"
17
17
  >
18
- <h3 class="text-xl font-semibold text-gray-900 dark:text-white">
18
+ <h3 class="text-xl font-semibold text-lightDialogHeaderText dark:text-darkDialogHeaderText">
19
19
  {{ header }}
20
20
  </h3>
21
21
  <button
22
22
  v-if="headerCloseButton"
23
23
  type="button"
24
- class="text-gray-400 bg-transparent hover:bg-gray-200 hover:text-gray-900 rounded-lg text-sm w-8 h-8 ms-auto inline-flex justify-center items-center dark:hover:bg-gray-600 dark:hover:text-white"
24
+ class="text-lightDialogCloseButton bg-transparent hover:bg-lightDialogCloseButtonHoverBackground hover:text-lightDialogCloseButtonHover rounded-lg text-sm w-8 h-8 ms-auto inline-flex justify-center items-center dark:text-darkDialogCloseButton dark:hover:bg-darkDialogCloseButtonHoverBackground dark:hover:text-darkDialogCloseButtonHover"
25
25
  @click="modal?.hide()"
26
26
  >
27
27
  <svg class="w-3 h-3" aria-hidden="true" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 14 14">
@@ -31,13 +31,13 @@
31
31
  </button>
32
32
  </div>
33
33
  <!-- Modal body -->
34
- <div class="p-4 md:p-5 space-y-4 text-gray-700 dark:text-gray-400">
34
+ <div class="p-4 md:p-5 space-y-4 text-lightDialogBodyText dark:text-darkDialogBodyText">
35
35
  <slot></slot>
36
36
  </div>
37
37
  <!-- Modal footer -->
38
38
  <div
39
39
  v-if="buttons.length"
40
- class="flex items-center p-4 md:p-5 border-t border-gray-200 rounded-b dark:border-gray-600"
40
+ class="flex items-center p-4 md:p-5 border-t border-lightDialogBreakLine rounded-b dark:border-darkDialogBreakLine"
41
41
  >
42
42
  <Button
43
43
  v-for="(button, buttonIndex) in buttons"
@@ -63,24 +63,33 @@ import { Modal } from 'flowbite';
63
63
  const modalEl = ref(null);
64
64
  const modal: Ref<Modal|null> = ref(null);
65
65
 
66
- const props = defineProps({
67
- header: {
68
- type: String,
69
- default: '',
70
- },
71
- headerCloseButton: {
72
- type: Boolean,
73
- default: true,
74
- },
75
- buttons: {
76
- type: Array,
77
- default: () => [{ label: 'Close', onclick: (dialog) => dialog.hide(), type: '' }],
78
- },
79
- clickToCloseOutside: {
80
- type: Boolean,
81
- default: true,
82
- },
83
- });
66
+ interface DialogButton {
67
+ label: string
68
+ onclick: (dialog: any) => void
69
+ options?: Record<string, any>
70
+ }
71
+
72
+ interface DialogProps {
73
+ header?: string
74
+ headerCloseButton?: boolean
75
+ buttons?: DialogButton[]
76
+ clickToCloseOutside?: boolean
77
+ beforeCloseFunction?: (() => void | Promise<void>) | null
78
+ beforeOpenFunction?: (() => void | Promise<void>) | null
79
+ closable?: boolean
80
+ }
81
+
82
+ const props = withDefaults(defineProps<DialogProps>(), {
83
+ header: '',
84
+ headerCloseButton: true,
85
+ buttons: () => [
86
+ { label: 'Close', onclick: (dialog: any) => dialog.hide(), type: '' },
87
+ ],
88
+ clickToCloseOutside: true,
89
+ beforeCloseFunction: null,
90
+ beforeOpenFunction: null,
91
+ closable: true,
92
+ })
84
93
 
85
94
  onMounted(async () => {
86
95
  //await one tick when all is mounted
@@ -88,8 +97,19 @@ onMounted(async () => {
88
97
  modal.value = new Modal(
89
98
  modalEl.value,
90
99
  {
100
+ closable: props.closable,
91
101
  backdrop: props.clickToCloseOutside ? 'dynamic' : 'static',
92
- },
102
+ onHide: async () => {
103
+ if (props.beforeCloseFunction) {
104
+ await props.beforeCloseFunction();
105
+ }
106
+ },
107
+ onShow: async () => {
108
+ if (props.beforeOpenFunction) {
109
+ await props.beforeOpenFunction();
110
+ }
111
+ },
112
+ }
93
113
  );
94
114
  })
95
115
 
@@ -3,48 +3,91 @@
3
3
  <form class="flex items-center justify-center w-full"
4
4
  @dragover.prevent="dragging = true"
5
5
  @dragleave.prevent="dragging = false"
6
- @drop.prevent="dragging = false; doEmit($event.dataTransfer.files)"
6
+ @drop.prevent="dragging = false; doEmit(($event.dataTransfer as DataTransfer).files)"
7
7
  >
8
- <label :id="id" class="flex flex-col items-center justify-center w-full h-32 border-2 border-dashed rounded-lg cursor-pointer dark:hover:bg-gray-800
9
- hover:bg-gray-100 dark:hover:border-gray-500 dark:hover:bg-gray-600"
8
+ <label :id="id" class="flex flex-col items-center justify-center w-full border-2 border-dashed rounded-lg cursor-pointer
9
+ hover:bg-lightDropzoneBackgroundHover hover:border-lightDropzoneBorderHover dark:hover:border-darkDropzoneBorderHover dark:hover:bg-darkDropzoneBackgroundHover"
10
10
  :class="{
11
- 'border-blue-600 dark:border-blue-400': dragging,
12
- 'border-gray-300 dark:border-gray-600': !dragging,
13
- 'bg-blue-50 dark:bg-blue-800': dragging,
14
- 'bg-gray-50 dark:bg-gray-800': !dragging,
11
+ 'border-lightDropzoneBorderDragging dark:border-darkDropzoneBorderDragging': dragging,
12
+ 'border-lightDropzoneBorder dark:border-darkDropzoneBorder': !dragging,
13
+ 'bg-lightDropzoneBackgroundDragging dark:bg-darkDropzoneBackgroundDragging': dragging,
14
+ 'bg-lightDropzoneBackground dark:bg-darkDropzoneBackground': !dragging,
15
+ 'min-h-32 h-full': props.multiple,
16
+ 'h-32': !props.multiple,
15
17
  }"
16
18
  >
17
- <div class="flex flex-col items-center justify-center pt-5 pb-6">
18
-
19
-
20
- <svg v-if="!selectedFiles.length" class="w-8 h-8 mb-4 text-gray-500 dark:text-gray-400" aria-hidden="true" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 20 16">
21
- <path stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M13 13h3a3 3 0 0 0 0-6h-.025A5.56 5.56 0 0 0 16 6.5 5.5 5.5 0 0 0 5.207 5.021C5.137 5.017 5.071 5 5 5a4 4 0 0 0 0 8h2.167M10 15V6m0 0L8 8m2-2 2 2"/>
22
- </svg>
23
- <div v-else class="flex items-center justify-center w-full mt-1 mb-4">
24
- <template v-for="file in selectedFiles">
25
- <p class="text-sm text-gray-500 dark:text-gray-400 flex items-center gap-1">
26
- <IconFileSolid class="w-5 h-5" />
27
- {{ file.name }} ({{ humanifySize(file.size) }})
28
- </p>
29
- </template>
30
-
19
+ <input
20
+ :id="id"
21
+ type="file"
22
+ class="hidden"
23
+ :accept="props.extensions.join(',')"
24
+ @change="$event.target && doEmit(($event.target as HTMLInputElement).files!)"
25
+ :multiple="props.multiple || false"
26
+ />
27
+
28
+ <div class="flex flex-col items-center justify-center pt-5 pb-6">
29
+ <svg
30
+ v-if="!selectedFiles.length"
31
+ class="w-8 h-8 mb-4 text-lightDropzoneIcon dark:text-darkDropzoneIcon"
32
+ xmlns="http://www.w3.org/2000/svg"
33
+ fill="none"
34
+ viewBox="0 0 20 16"
35
+ >
36
+ <path
37
+ stroke="currentColor"
38
+ stroke-linecap="round"
39
+ stroke-linejoin="round"
40
+ stroke-width="2"
41
+ d="M13 13h3a3 3 0 0 0 0-6h-.025A5.56 5.56 0 0 0 16 6.5
42
+ 5.5 5.5 0 0 0 5.207 5.021C5.137 5.017 5.071 5 5 5a4 4 0
43
+ 0 0 0 8h2.167M10 15V6m0 0L8 8m2-2 2 2"
44
+ />
45
+ </svg>
46
+
47
+ <div
48
+ v-else
49
+ class="flex items-center justify-center py-1 flex-wrap gap-2 w-full gap-2 mt-1 mb-4 px-4"
50
+ >
51
+ <template v-for="(file, index) in selectedFiles" :key="index">
52
+ <div
53
+ class="text-sm text-lightDropzoneIcon dark:text-darkDropzoneIcon bg-lightDropzoneBackgroundHover dark:bg-darkDropzoneBackgroundHover rounded-md
54
+ flex items-center gap-1 px-2 py-1 group"
55
+ >
56
+ <IconFileSolid class="w-4 h-4 flex-shrink-0" />
57
+ <span class="truncate max-w-[200px]">{{ file.name }}</span>
58
+ <span class="text-xs">({{ humanifySize(file.size) }})</span>
59
+ <button
60
+ type="button"
61
+ @click.prevent.stop="removeFile(index)"
62
+ class="text-lightDropzoneIcon dark:text-darkDropzoneIcon hover:text-red-600 dark:hover:text-red-400
63
+ opacity-70 hover:opacity-100 transition-all"
64
+ :title="$t('Remove file')"
65
+ >
66
+ <svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
67
+ <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M6 18L18 6M6 6l12 12" />
68
+ </svg>
69
+ </button>
31
70
  </div>
32
-
33
- <p v-if="!selectedFiles.length" class="mb-2 text-sm text-gray-500 dark:text-gray-400"><span class="font-semibold">{{ $t('Click to upload') }}</span> {{ $t('or drag and drop') }}</p>
34
- <p class="text-xs text-gray-500 dark:text-gray-400">
35
- {{ props.extensions.join(', ').toUpperCase().replace(/\./g, '') }}
36
- <template v-if="props.maxSizeBytes">
37
- (Max size: {{ humanifySize(props.maxSizeBytes) }})
38
- </template>
39
- </p>
71
+ </template>
40
72
  </div>
41
- <input :id="id" type="file" class="hidden"
42
- :accept="props.extensions.join(', ')"
43
- @change="doEmit($event.target.files)"
44
- :multiple="props.multiple || false"
45
- />
73
+
74
+ <p
75
+ v-if="!selectedFiles.length"
76
+ class="mb-2 text-sm text-lightDropzoneText dark:text-darkDropzoneText"
77
+ >
78
+ <span class="font-semibold">{{ $t('Click to upload') }}</span>
79
+ {{ $t('or drag and drop') }}
80
+ </p>
81
+
82
+ <p class="text-xs text-lightDropzoneText dark:text-darkDropzoneText">
83
+ {{ props.extensions.join(', ').toUpperCase().replace(/\./g, '') }}
84
+ <template v-if="props.maxSizeBytes">
85
+ (Max size: {{ humanifySize(props.maxSizeBytes) }})
86
+ </template>
87
+ </p>
88
+ </div>
46
89
  </label>
47
- </form>
90
+ </form>
48
91
  </template>
49
92
 
50
93
  <script setup lang="ts">
@@ -71,12 +114,17 @@ const selectedFiles: Ref<{
71
114
  mime: string,
72
115
  }[]> = ref([]);
73
116
 
117
+ const storedFiles: Ref<File[]> = ref([]);
118
+
74
119
  watch(() => props.modelValue, (files) => {
75
- selectedFiles.value = Array.from(files).map(file => ({
76
- name: file.name,
77
- size: file.size,
78
- mime: file.type,
79
- }));
120
+ if (files && files.length > 0) {
121
+ selectedFiles.value = Array.from(files).map(file => ({
122
+ name: file.name,
123
+ size: file.size,
124
+ mime: file.type,
125
+ }));
126
+ storedFiles.value = Array.from(files);
127
+ }
80
128
  });
81
129
 
82
130
  function doEmit(filesIn: FileList) {
@@ -108,26 +156,37 @@ function doEmit(filesIn: FileList) {
108
156
  });
109
157
  return;
110
158
  }
111
-
159
+
112
160
  validFiles.push(file);
113
161
  });
114
162
 
115
163
  if (!multiple) {
116
- validFiles.splice(1);
164
+ storedFiles.value = validFiles.slice(0, 1);
165
+ } else {
166
+ storedFiles.value = [...storedFiles.value, ...validFiles];
117
167
  }
118
- selectedFiles.value = validFiles.map(file => ({
168
+
169
+ selectedFiles.value = storedFiles.value.map(file => ({
119
170
  name: file.name,
120
171
  size: file.size,
121
172
  mime: file.type,
122
173
  }));
123
174
 
124
- emit('update:modelValue', validFiles);
175
+ emit('update:modelValue', storedFiles.value);
176
+
125
177
  }
126
178
 
127
179
  const dragging = ref(false);
128
180
 
181
+ function removeFile(index: number) {
182
+ storedFiles.value = storedFiles.value.filter((_, i) => i !== index);
183
+ selectedFiles.value = selectedFiles.value.filter((_, i) => i !== index);
184
+ emit('update:modelValue', storedFiles.value);
185
+ }
186
+
129
187
  function clear() {
130
188
  selectedFiles.value = [];
189
+ storedFiles.value = [];
131
190
  emit('update:modelValue', []);
132
191
  const form = document.getElementById(id)?.closest('form');
133
192
  form?.reset();