adminforth 1.3.54-next.3 → 1.3.54-next.30

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 (135) hide show
  1. package/dist/auth.js +42 -56
  2. package/dist/auth.js.map +1 -0
  3. package/dist/basePlugin.js +1 -0
  4. package/dist/basePlugin.js.map +1 -0
  5. package/dist/dataConnectors/baseConnector.js +108 -122
  6. package/dist/dataConnectors/baseConnector.js.map +1 -0
  7. package/dist/dataConnectors/clickhouse.js +132 -150
  8. package/dist/dataConnectors/clickhouse.js.map +1 -0
  9. package/dist/dataConnectors/mongo.js +75 -101
  10. package/dist/dataConnectors/mongo.js.map +1 -0
  11. package/dist/dataConnectors/postgres.js +124 -143
  12. package/dist/dataConnectors/postgres.js.map +1 -0
  13. package/dist/dataConnectors/sqlite.js +113 -130
  14. package/dist/dataConnectors/sqlite.js.map +1 -0
  15. package/dist/index.js +197 -217
  16. package/dist/index.js.map +1 -0
  17. package/dist/modules/codeInjector.js +480 -486
  18. package/dist/modules/codeInjector.js.map +1 -0
  19. package/dist/modules/configValidator.js +31 -22
  20. package/dist/modules/configValidator.js.map +1 -0
  21. package/dist/modules/operationalResource.js +50 -70
  22. package/dist/modules/operationalResource.js.map +1 -0
  23. package/dist/modules/restApi.js +104 -116
  24. package/dist/modules/restApi.js.map +1 -0
  25. package/dist/modules/styleGenerator.js +1 -0
  26. package/dist/modules/styleGenerator.js.map +1 -0
  27. package/dist/modules/styles.js +1 -0
  28. package/dist/modules/styles.js.map +1 -0
  29. package/dist/modules/utils.js +1 -0
  30. package/dist/modules/utils.js.map +1 -0
  31. package/dist/plugins/audit-log/types.js +2 -0
  32. package/dist/plugins/audit-log/types.js.map +1 -0
  33. package/dist/plugins/chat-gpt/types.js +2 -0
  34. package/dist/plugins/chat-gpt/types.js.map +1 -0
  35. package/dist/plugins/email-password-reset/types.js +2 -0
  36. package/dist/plugins/email-password-reset/types.js.map +1 -0
  37. package/dist/plugins/foreign-inline-list/types.js +2 -0
  38. package/dist/plugins/foreign-inline-list/types.js.map +1 -0
  39. package/dist/plugins/import-export/types.js +2 -0
  40. package/dist/plugins/import-export/types.js.map +1 -0
  41. package/dist/plugins/rich-editor/custom/async-queue.js +29 -0
  42. package/dist/plugins/rich-editor/custom/async-queue.js.map +1 -0
  43. package/dist/plugins/rich-editor/dist/async-queue.js +41 -0
  44. package/dist/plugins/rich-editor/dist/custom/async-queue.js +29 -0
  45. package/dist/plugins/rich-editor/dist/custom/async-queue.js.map +1 -0
  46. package/dist/plugins/rich-editor/types.js +16 -0
  47. package/dist/plugins/rich-editor/types.js.map +1 -0
  48. package/dist/plugins/two-factors-auth/types.js +2 -0
  49. package/dist/plugins/two-factors-auth/types.js.map +1 -0
  50. package/dist/plugins/upload/types.js +2 -0
  51. package/dist/plugins/upload/types.js.map +1 -0
  52. package/dist/servers/express.js +30 -42
  53. package/dist/servers/express.js.map +1 -0
  54. package/dist/types/AdminForthConfig.js +1 -0
  55. package/dist/types/AdminForthConfig.js.map +1 -0
  56. package/dist/types/FrontendAPI.js +1 -0
  57. package/dist/types/FrontendAPI.js.map +1 -0
  58. package/package.json +7 -4
  59. package/auth.ts +0 -140
  60. package/basePlugin.ts +0 -70
  61. package/dataConnectors/baseConnector.ts +0 -216
  62. package/dataConnectors/clickhouse.ts +0 -341
  63. package/dataConnectors/mongo.ts +0 -202
  64. package/dataConnectors/postgres.ts +0 -306
  65. package/dataConnectors/sqlite.ts +0 -254
  66. package/index.ts +0 -428
  67. package/modules/codeInjector.ts +0 -736
  68. package/modules/configValidator.ts +0 -571
  69. package/modules/operationalResource.ts +0 -98
  70. package/modules/restApi.ts +0 -718
  71. package/modules/styleGenerator.ts +0 -55
  72. package/modules/styles.ts +0 -126
  73. package/modules/utils.ts +0 -472
  74. package/servers/express.ts +0 -259
  75. package/spa/.eslintrc.cjs +0 -14
  76. package/spa/README.md +0 -39
  77. package/spa/env.d.ts +0 -1
  78. package/spa/index.html +0 -23
  79. package/spa/package-lock.json +0 -4573
  80. package/spa/package.json +0 -49
  81. package/spa/postcss.config.js +0 -6
  82. package/spa/public/assets/favicon.png +0 -0
  83. package/spa/src/App.vue +0 -418
  84. package/spa/src/assets/base.css +0 -2
  85. package/spa/src/assets/logo.svg +0 -19
  86. package/spa/src/components/AcceptModal.vue +0 -45
  87. package/spa/src/components/Breadcrumbs.vue +0 -41
  88. package/spa/src/components/BreadcrumbsWithButtons.vue +0 -26
  89. package/spa/src/components/CustomDatePicker.vue +0 -176
  90. package/spa/src/components/CustomDateRangePicker.vue +0 -218
  91. package/spa/src/components/CustomRangePicker.vue +0 -156
  92. package/spa/src/components/Dropdown.vue +0 -168
  93. package/spa/src/components/Filters.vue +0 -222
  94. package/spa/src/components/HelloWorld.vue +0 -17
  95. package/spa/src/components/MenuLink.vue +0 -27
  96. package/spa/src/components/ResourceForm.vue +0 -290
  97. package/spa/src/components/ResourceListTable.vue +0 -460
  98. package/spa/src/components/SingleSkeletLoader.vue +0 -13
  99. package/spa/src/components/SkeleteLoader.vue +0 -23
  100. package/spa/src/components/ThreeDotsMenu.vue +0 -43
  101. package/spa/src/components/Toast.vue +0 -78
  102. package/spa/src/components/ValueRenderer.vue +0 -114
  103. package/spa/src/components/icons/IconCalendar.vue +0 -5
  104. package/spa/src/components/icons/IconCommunity.vue +0 -7
  105. package/spa/src/components/icons/IconDocumentation.vue +0 -7
  106. package/spa/src/components/icons/IconEcosystem.vue +0 -7
  107. package/spa/src/components/icons/IconSupport.vue +0 -7
  108. package/spa/src/components/icons/IconTime.vue +0 -5
  109. package/spa/src/components/icons/IconTooling.vue +0 -19
  110. package/spa/src/composables/useFrontendApi.ts +0 -26
  111. package/spa/src/composables/useStores.ts +0 -131
  112. package/spa/src/index.scss +0 -31
  113. package/spa/src/main.ts +0 -18
  114. package/spa/src/router/index.ts +0 -59
  115. package/spa/src/spa_types/core.ts +0 -53
  116. package/spa/src/stores/core.ts +0 -148
  117. package/spa/src/stores/filters.ts +0 -27
  118. package/spa/src/stores/modal.ts +0 -48
  119. package/spa/src/stores/toast.ts +0 -31
  120. package/spa/src/stores/user.ts +0 -72
  121. package/spa/src/utils.ts +0 -149
  122. package/spa/src/views/CreateView.vue +0 -167
  123. package/spa/src/views/EditView.vue +0 -170
  124. package/spa/src/views/ListView.vue +0 -279
  125. package/spa/src/views/LoginView.vue +0 -192
  126. package/spa/src/views/ResourceParent.vue +0 -17
  127. package/spa/src/views/ShowView.vue +0 -186
  128. package/spa/tailwind.config.js +0 -17
  129. package/spa/tsconfig.app.json +0 -14
  130. package/spa/tsconfig.json +0 -11
  131. package/spa/tsconfig.node.json +0 -19
  132. package/spa/vite.config.ts +0 -56
  133. package/tsconfig.json +0 -112
  134. package/types/AdminForthConfig.ts +0 -1762
  135. package/types/FrontendAPI.ts +0 -143
@@ -1,176 +0,0 @@
1
- <template>
2
- <div>
3
- <div class="grid w-40 gap-4 mb-2">
4
- <div>
5
- <label for="start-time" class="block mb-2 text-sm font-medium text-gray-900 dark:text-white">{{ label }}</label>
6
-
7
- <div class="relative">
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"/>
10
- </div>
11
-
12
- <input ref="datepickerStartEl" type="text"
13
- class="bg-gray-50 border leading-none border-gray-300 text-gray-900 text-sm rounded-lg focus:ring-blue-500 focus:border-blue-500 block w-full p-2.5 dark:bg-gray-700 dark:border-gray-600 dark:placeholder-gray-400 dark:text-white dark:focus:ring-blue-500 dark:focus:border-blue-500"
14
- placeholder="Select date">
15
- </div>
16
- </div>
17
- </div>
18
-
19
- <div>
20
- <div class="grid w-40 gap-4 mb-2" :class="{hidden: !showTimeInputs}">
21
- <div>
22
- <div class="relative">
23
- <div class="absolute inset-y-0 end-0 top-0 flex items-center pe-3.5 pointer-events-none">
24
- <IconTime class="w-4 h-4 text-gray-500 dark:text-gray-400 bg-white dark:bg-gray-700"/>
25
- </div>
26
-
27
- <input v-model="startTime" type="time" id="start-time" onfocus="this.showPicker()" onclick="this.showPicker()" step="1"
28
- class="bg-gray-50 border leading-none border-gray-300 text-gray-900 text-sm rounded-lg focus:ring-blue-500 focus:border-blue-500 block w-full p-2.5 dark:bg-gray-700 dark:border-gray-600 dark:placeholder-gray-400 dark:text-white dark:focus:ring-blue-500 dark:focus:border-blue-500"
29
- value="00:00" required/>
30
- </div>
31
- </div>
32
- </div>
33
-
34
- <button type="button"
35
- class="text-lightPrimary dark:text-darkPrimary text-base font-medium hover:underline p-0 inline-flex items-center mb-2"
36
- @click="toggleTimeInputs">{{ showTimeInputs ? 'Hide time' : 'Show time' }}
37
- <svg class="w-8 h-8 ms-0.5" :class="{'rotate-180': showTimeInputs}" aria-hidden="true"
38
- xmlns="http://www.w3.org/2000/svg" width="24" height="24"
39
- fill="none" viewBox="0 0 24 24">
40
- <path stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" stroke-width="2"
41
- d="m8 10 4 4 4-4"/>
42
- </svg>
43
- </button>
44
- </div>
45
- </div>
46
- </template>
47
- <script setup>
48
- import {ref, computed, onMounted, watch, onBeforeUnmount} from 'vue';
49
- import dayjs from 'dayjs';
50
- import utc from 'dayjs/plugin/utc';
51
-
52
- import {useCoreStore} from '@/stores/core';
53
-
54
- import Datepicker from "flowbite-datepicker/Datepicker";
55
- import IconCalendar from "@/components/icons/IconCalendar.vue";
56
- import IconTime from "@/components/icons/IconTime.vue";
57
-
58
- const coreStore = useCoreStore();
59
- dayjs.extend(utc)
60
-
61
- const props = defineProps({
62
- valueStart: {
63
- default: undefined
64
- },
65
- column: {
66
- type: Object,
67
- },
68
- label: {
69
- type: String,
70
- },
71
- autoHide: {
72
- type: Boolean,
73
- }
74
- });
75
-
76
- const emit = defineEmits(['update:valueStart']);
77
-
78
- const datepickerStartEl = ref();
79
-
80
- const showTimeInputs = ref(false);
81
-
82
- const startDate = ref('');
83
-
84
- const startTime = ref('');
85
-
86
- const datepickerObject = ref('')
87
-
88
- const start = computed(() => {
89
- if (!startDate.value) {
90
- return;
91
- }
92
-
93
- let date = dayjs(startDate.value);
94
-
95
- if (startTime.value) {
96
- date = addTimeToDate(formatTime(startTime.value), date)
97
- }
98
-
99
- return date.utc().toISOString();
100
- })
101
-
102
- async function updateFromProps() {
103
- if (!props.valueStart) {
104
- datepickerStartEl.value.value = '';
105
- startTime.value = '';
106
- } else {
107
- // wait ref to initialize
108
- await (new Promise(resolve => setTimeout(resolve, 0)));
109
- datepickerObject.value.setDate(dayjs(props.valueStart).format('DD MMM YYYY'));
110
- startTime.value = dayjs(props.valueStart).format('HH:mm:ss')
111
- }
112
- }
113
-
114
- onMounted(() => {
115
- updateFromProps();
116
-
117
- watch(() => [props.valueStart], (value) => {
118
- updateFromProps();
119
- });
120
- })
121
-
122
- watch(start, () => {
123
- //console.log('⚡ emit', start.value)
124
- emit('update:valueStart', start.value)
125
- })
126
-
127
- function initDatepickers() {
128
- const options = {format: 'dd M yyyy'};
129
-
130
- if (props.autoHide) {
131
- options.autohide = true;
132
- }
133
-
134
- datepickerObject.value = new Datepicker(datepickerStartEl.value, options);
135
-
136
- addChangeDateListener();
137
- }
138
-
139
- function addChangeDateListener() {
140
- datepickerStartEl.value.addEventListener('changeDate', setStartDate)
141
- }
142
-
143
- function removeChangeDateListener() {
144
- datepickerStartEl.value.removeEventListener('changeDate', setStartDate);
145
- }
146
-
147
- function destroyDatepickerElement() {
148
- datepickerObject.value.destroy();
149
- }
150
-
151
- function setStartDate(event) {
152
- startDate.value = event.detail.date
153
- }
154
-
155
- function formatTime(time) {
156
- return time.split(':').map(Number).length === 2 ? time + ':00' : time;
157
- }
158
-
159
- function addTimeToDate(time, date) {
160
- const [hours, minutes, seconds] = time.split(':').map(Number)
161
- return date.hour(hours).minute(minutes).second(seconds)
162
- }
163
-
164
- const toggleTimeInputs = () => {
165
- showTimeInputs.value = !showTimeInputs.value
166
- }
167
-
168
- onMounted(() => {
169
- initDatepickers();
170
- });
171
-
172
- onBeforeUnmount(() => {
173
- removeChangeDateListener();
174
- destroyDatepickerElement();
175
- });
176
- </script>
@@ -1,218 +0,0 @@
1
- <template>
2
- <div>
3
- <div class="mx-auto grid grid-cols-2 gap-4 mb-2">
4
- <div class="relative">
5
- <div class="absolute inset-y-0 end-0 top-0 flex items-center pe-3.5 pointer-events-none">
6
- <IconCalendar class="w-4 h-4 text-gray-500 dark:text-gray-400"/>
7
- </div>
8
-
9
- <input ref="datepickerStartEl" type="text"
10
- class="bg-gray-50 border leading-none border-gray-300 text-gray-900 text-sm rounded-lg focus:ring-blue-500 focus:border-blue-500 block w-full p-2.5 dark:bg-gray-700 dark:border-gray-600 dark:placeholder-gray-400 dark:text-white dark:focus:ring-blue-500 dark:focus:border-blue-500"
11
- placeholder="From">
12
- </div>
13
-
14
- <div class="relative">
15
- <div class="absolute inset-y-0 end-0 top-0 flex items-center pe-3.5 pointer-events-none">
16
- <IconCalendar class="w-4 h-4 text-gray-500 dark:text-gray-400"/>
17
- </div>
18
-
19
- <input ref="datepickerEndEl" type="text"
20
- class="bg-gray-50 border leading-none border-gray-300 text-gray-900 text-sm rounded-lg focus:ring-blue-500 focus:border-blue-500 block w-full p-2.5 dark:bg-gray-700 dark:border-gray-600 dark:placeholder-gray-400 dark:text-white dark:focus:ring-blue-500 dark:focus:border-blue-500"
21
- placeholder="To">
22
- </div>
23
- </div>
24
-
25
- <div>
26
- <div class="mx-auto grid grid-cols-2 gap-4 mb-2" :class="{hidden: !showTimeInputs}">
27
- <div>
28
- <div class="relative">
29
- <div class="absolute inset-y-0 end-0 top-0 flex items-center pe-3.5 pointer-events-none">
30
- <IconTime class="w-4 h-4 text-gray-500 dark:text-gray-400 bg-white dark:bg-gray-700"/>
31
- </div>
32
-
33
- <input v-model="startTime" type="time" id="start-time"
34
- class="bg-gray-50 border leading-none border-gray-300 text-gray-900 text-sm rounded-lg focus:ring-blue-500 focus:border-blue-500 block w-full p-2.5 dark:bg-gray-700 dark:border-gray-600 dark:placeholder-gray-400 dark:text-white dark:focus:ring-blue-500 dark:focus:border-blue-500"
35
- value="00:00" required/>
36
- </div>
37
- </div>
38
-
39
- <div>
40
- <div class="relative">
41
- <div class="absolute inset-y-0 end-0 top-0 flex items-center pe-3.5 pointer-events-none">
42
- <IconTime class="w-4 h-4 text-gray-500 dark:text-gray-400 bg-white dark:bg-gray-700"/>
43
- </div>
44
-
45
- <input v-model="endTime" type="time" id="end-time"
46
- class="bg-gray-50 border leading-none border-gray-300 text-gray-900 text-sm rounded-lg focus:ring-blue-500 focus:border-blue-500 block w-full p-2.5 dark:bg-gray-700 dark:border-gray-600 dark:placeholder-gray-400 dark:text-white dark:focus:ring-blue-500 dark:focus:border-blue-500"
47
- value="00:00" required/>
48
- </div>
49
- </div>
50
- </div>
51
-
52
- <button type="button"
53
- class="text-lightPrimary dark:text-darkPrimary text-base font-medium hover:underline p-0 inline-flex items-center mb-2"
54
- @click="toggleTimeInputs">{{ showTimeInputs ? 'Hide time' : 'Show time' }}
55
- <svg class="w-8 h-8 ms-0.5 relative top-px" :class="{'rotate-180': showTimeInputs}" aria-hidden="true"
56
- xmlns="http://www.w3.org/2000/svg" width="24" height="24"
57
- fill="none" viewBox="0 0 24 24">
58
- <path stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" stroke-width="2"
59
- d="m8 10 4 4 4-4"/>
60
- </svg>
61
- </button>
62
- </div>
63
- </div>
64
- </template>
65
- <script setup>
66
- import {ref, computed, onMounted, watch, onBeforeUnmount} from 'vue';
67
- import dayjs from 'dayjs';
68
- import utc from 'dayjs/plugin/utc';
69
-
70
- import {useCoreStore} from '@/stores/core';
71
-
72
- import Datepicker from "flowbite-datepicker/Datepicker";
73
- import IconCalendar from "@/components/icons/IconCalendar.vue";
74
- import IconTime from "@/components/icons/IconTime.vue";
75
-
76
- const coreStore = useCoreStore();
77
- dayjs.extend(utc)
78
-
79
- const props = defineProps({
80
- valueStart: {
81
- default: undefined
82
- },
83
- valueEnd: {
84
- default: undefined
85
- },
86
- column: {
87
- type: Object,
88
- },
89
- });
90
-
91
- const emit = defineEmits(['update:valueStart', 'update:valueEnd']);
92
-
93
- const datepickerStartEl = ref();
94
- const datepickerEndEl = ref();
95
-
96
- const showTimeInputs = ref(false);
97
-
98
- const startDate = ref('');
99
- const endDate = ref('');
100
-
101
- const startTime = ref('');
102
- const endTime = ref('');
103
-
104
- const datepickerStartObject = ref('')
105
- const datepickerEndObject = ref('')
106
-
107
- const start = computed(() => {
108
- if (!startDate.value) {
109
- return;
110
- }
111
-
112
- let date = dayjs(startDate.value);
113
-
114
- if (startTime.value) {
115
- date = addTimeToDate(formatTime(startTime.value), date)
116
- }
117
-
118
- return date.utc().toISOString();
119
- })
120
-
121
- const end = computed(() => {
122
- if (!endDate.value) {
123
- return;
124
- }
125
-
126
- let date = dayjs(endDate.value);
127
-
128
- if (endTime.value) {
129
- date = addTimeToDate(formatTime(endTime.value), date)
130
- } else {
131
- date = addTimeToDate('23:59:59', date)
132
- }
133
-
134
- return date.utc().toISOString();
135
- })
136
-
137
- function updateFromProps() {
138
- if (!props.valueStart) {
139
- datepickerStartEl.value.value = '';
140
- startTime.value = '';
141
- }
142
- if (!props.valueEnd) {
143
- datepickerEndEl.value.value = '';
144
- endTime.value = '';
145
- }
146
- }
147
-
148
- onMounted(() => {
149
- updateFromProps();
150
-
151
- watch(() => [props.valueStart, props.valueEnd], (value) => {
152
- updateFromProps();
153
- });
154
- })
155
-
156
- watch(start, () => {
157
- //console.log('⚡ emit', start.value)
158
- emit('update:valueStart', start.value)
159
- })
160
-
161
- watch(end, () => {
162
- //console.log('⚡ emit', end.value)
163
- emit('update:valueEnd', end.value)
164
- })
165
-
166
- function initDatepickers() {
167
-
168
- datepickerStartObject.value = new Datepicker(datepickerStartEl.value, {format: 'dd M yyyy'});
169
-
170
- datepickerEndObject.value = new Datepicker(datepickerEndEl.value, {format: 'dd M yyyy'});
171
- addChangeDateListener();
172
- }
173
-
174
- function addChangeDateListener() {
175
- datepickerStartEl.value.addEventListener('changeDate', setStartDate)
176
- datepickerEndEl.value.addEventListener('changeDate', setEndDate)
177
- }
178
-
179
- function removeChangeDateListener() {
180
- datepickerStartEl.value.removeEventListener('changeDate', setStartDate);
181
- datepickerEndEl.value.removeEventListener('changeDate', setEndDate);
182
- }
183
-
184
- function destroyDatepickerElement() {
185
- datepickerStartObject.value.destroy();
186
- datepickerEndObject.value.destroy();
187
- }
188
-
189
- function setStartDate(event) {
190
- startDate.value = event.detail.date
191
- }
192
-
193
- function setEndDate(event) {
194
- endDate.value = event.detail.date
195
- }
196
-
197
- function formatTime(time) {
198
- return time.split(':').map(Number).length === 2 ? time + ':00' : time;
199
- }
200
-
201
- function addTimeToDate(time, date) {
202
- const [hours, minutes, seconds] = time.split(':').map(Number)
203
- return date.hour(hours).minute(minutes).second(seconds)
204
- }
205
-
206
- const toggleTimeInputs = () => {
207
- showTimeInputs.value = !showTimeInputs.value
208
- }
209
-
210
- onMounted(() => {
211
- initDatepickers();
212
- });
213
-
214
- onBeforeUnmount(() => {
215
- removeChangeDateListener();
216
- destroyDatepickerElement();
217
- })
218
- </script>
@@ -1,156 +0,0 @@
1
- <template>
2
- <div class="flex flex-wrap gap-2">
3
- <input
4
- :min="minFormatted"
5
- :max="maxFormatted"
6
- type="number" aria-describedby="helper-text-explanation"
7
- class="bg-gray-50 border border-gray-300 text-gray-900 text-sm rounded-lg focus:ring-blue-500 focus:border-blue-500 block w-20 p-2.5 dark:bg-gray-700 dark:border-gray-600 dark:placeholder-gray-400 dark:text-white dark:focus:ring-blue-500 dark:focus:border-blue-500"
8
- placeholder="From"
9
- v-model="start"
10
- >
11
-
12
- <input
13
- :min="minFormatted"
14
- :max="maxFormatted"
15
- type="number" aria-describedby="helper-text-explanation"
16
- class="bg-gray-50 border border-gray-300 text-gray-900 text-sm rounded-lg focus:ring-blue-500 focus:border-blue-500 block w-20 p-2.5 dark:bg-gray-700 dark:border-gray-600 dark:placeholder-gray-400 dark:text-white dark:focus:ring-blue-500 dark:focus:border-blue-500"
17
- placeholder="To"
18
- v-model="end"
19
- >
20
-
21
- <button
22
- v-if="isChanged"
23
- type="button"
24
- class="flex items-center p-0.5 ml-auto px-3 text-sm font-medium text-gray-900 focus:outline-none bg-white rounded border border-gray-300 hover:bg-gray-100 hover:text-lightPrimary focus:z-10 focus:ring-4 focus:ring-gray-100 dark:focus:ring-gray-700 dark:bg-gray-800 dark:text-gray-400 dark:border-gray-600 dark:hover:text-white dark:hover:bg-gray-700 disabled:opacity-50 disabled:cursor-not-allowed"
25
- @click="clear">Clear
26
- </button>
27
-
28
- <div v-if="min && max" class="w-full px-2.5">
29
- <vue-slider
30
- class="custom-slider"
31
- :dot-size="20"
32
- height="7.99px"
33
- :min="minFormatted"
34
- :max="maxFormatted"
35
- v-model="sliderValue"
36
- @update:model-value="updateFromSlider($event)"
37
- />
38
- </div>
39
- </div>
40
- </template>
41
- <script setup lang="ts">
42
- import VueSlider from 'vue-slider-component';
43
- import 'vue-slider-component/theme/antd.css'
44
- import {computed, onMounted, ref, watch} from "vue";
45
- import debounce from 'debounce'
46
-
47
- const props = defineProps({
48
- valueStart: {
49
- default: '',
50
- },
51
- valueEnd: {
52
- default: '',
53
- },
54
- min: {},
55
- max: {},
56
- });
57
-
58
- const emit = defineEmits(['update:valueStart', 'update:valueEnd']);
59
-
60
- const minFormatted = computed(() => Math.floor(props.min));
61
- const maxFormatted = computed(() => Math.ceil(props.max));
62
-
63
- const isChanged = computed(() => {
64
- return start.value && start.value !== minFormatted.value || end.value && end.value !== maxFormatted.value;
65
- });
66
-
67
- const start = ref(props.valueStart);
68
- const end = ref(props.valueEnd);
69
-
70
- const sliderValue = ref([minFormatted.value, maxFormatted.value]);
71
-
72
- const updateFromSlider =
73
- debounce((value: [number, number]) => {
74
- start.value = value[0] === minFormatted.value ? '': value[0];
75
- end.value = value[1] === maxFormatted.value ? '': value[1];
76
- }, 500);
77
-
78
- onMounted(() => {
79
- updateStartFromProps();
80
- updateEndFromProps();
81
-
82
- watch(() => props.valueStart, (value) => {
83
- updateStartFromProps();
84
- });
85
-
86
- watch(() => props.valueEnd, (value) => {
87
- updateEndFromProps();
88
- });
89
- })
90
-
91
- function updateStartFromProps() {
92
- start.value = props.valueStart;
93
- setSliderValues(start.value, end.value)
94
- }
95
-
96
- function updateEndFromProps() {
97
- end.value = props.valueEnd;
98
- setSliderValues(start.value, end.value)
99
- }
100
-
101
- watch(start, () => {
102
- console.log('⚡ emit', start.value)
103
- emit('update:valueStart', start.value)
104
- })
105
-
106
- watch(end, () => {
107
- console.log('⚡ emit', end.value)
108
- emit('update:valueEnd', end.value);
109
- })
110
-
111
- watch([minFormatted,maxFormatted], () => {
112
- setSliderValues(minFormatted.value, maxFormatted.value)
113
- })
114
-
115
- const clear = () => {
116
- start.value = ''
117
- end.value = ''
118
- setSliderValues('', '')
119
- }
120
-
121
- function setSliderValues(start, end) {
122
- sliderValue.value = [start || minFormatted.value, end || maxFormatted.value];
123
- }
124
- </script>
125
-
126
- <style lang="scss" scoped>
127
- .custom-slider {
128
- &:deep(.vue-slider-rail) {
129
- background-color: rgb(229 231 235);
130
- }
131
-
132
- &:deep(.vue-slider-dot-handle) {
133
- // apply bg-blue-500 to the handle when active
134
- @apply bg-lightPrimary;
135
- border: none;
136
- box-shadow: none;
137
- }
138
-
139
- &:deep(.vue-slider-dot-handle:hover) {
140
- @apply bg-lightPrimary;
141
- filter: brightness(1.1);
142
- border: none;
143
- box-shadow: none;
144
- }
145
-
146
- &:deep(.vue-slider-process) {
147
- @apply bg-lightPrimaryOpacity;
148
-
149
- }
150
-
151
- &:deep(.vue-slider-process:hover) {
152
- filter: brightness(1.1);
153
- @apply bg-lightPrimaryOpacity;
154
- }
155
- }
156
- </style>