@zap-wunschlachen/wl-shared-components 1.0.76 → 1.0.78
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.
- package/.github/workflows/playwright.yml +229 -229
- package/.github/workflows/static.yml +61 -61
- package/.github/workflows/update-snapshots.yml +37 -37
- package/.prettierrc.json +8 -8
- package/.storybook/main.ts +18 -18
- package/.storybook/preview.ts +37 -37
- package/.storybook/storyWrapper.vue +18 -18
- package/.storybook/withVuetifyTheme.decorator.ts +21 -21
- package/App.vue +139 -139
- package/README.md +56 -56
- package/docs/assets.md +62 -62
- package/heroicons.ts +75 -75
- package/index.html +19 -19
- package/package.json +71 -71
- package/playwright.config.ts +48 -48
- package/public/background.svg +60 -60
- package/public/style.css +187 -187
- package/public/technologies.svg +22 -22
- package/scripts/check-translations.ts +352 -352
- package/src/assets/css/base.css +242 -242
- package/src/assets/css/variables.css +176 -176
- package/src/components/Accordion/Accordion.css +65 -65
- package/src/components/Accordion/AccordionGroup.vue +88 -88
- package/src/components/Accordion/AccordionItem.vue +272 -272
- package/src/components/Accordion/presets/default.css +4 -4
- package/src/components/Accordion/presets/elevated.css +25 -25
- package/src/components/Accordion/presets/filled.css +26 -26
- package/src/components/Accordion/presets/index.css +5 -5
- package/src/components/Accordion/presets/plain.css +34 -34
- package/src/components/Appointment/Card/Actions.css +54 -54
- package/src/components/Appointment/Card/Actions.vue +99 -99
- package/src/components/Appointment/Card/AnamneseNotification.css +20 -20
- package/src/components/Appointment/Card/AnamneseNotification.vue +23 -23
- package/src/components/Appointment/Card/Card.css +99 -99
- package/src/components/Appointment/Card/Card.vue +97 -97
- package/src/components/Appointment/Card/Details.css +62 -62
- package/src/components/Appointment/Card/Details.vue +44 -44
- package/src/components/Audio/Audio.vue +187 -187
- package/src/components/Audio/Waveform.vue +118 -118
- package/src/components/Banner/Banner.css +29 -29
- package/src/components/Banner/Banner.vue +89 -89
- package/src/components/Button/Button.vue +257 -257
- package/src/components/CheckBox/CheckBox.css +234 -234
- package/src/components/CheckBox/Checkbox.vue +184 -184
- package/src/components/DateInput/DateInput.css +2 -2
- package/src/components/DateInput/DateInput.vue +376 -370
- package/src/components/Dialog/Dialog.css +6 -6
- package/src/components/Dialog/Dialog.vue +46 -46
- package/src/components/EditField/EditField.css +19 -19
- package/src/components/EditField/EditField.vue +211 -211
- package/src/components/ErrorPage/ErrorPage.css +172 -172
- package/src/components/IconBullet/IconBullet.vue +104 -104
- package/src/components/IconBullet/IconBulletList.vue +55 -55
- package/src/components/Icons/AdvanceAppointments.vue +161 -161
- package/src/components/Icons/Audio/CloudFailed.vue +27 -27
- package/src/components/Icons/Audio/CloudSaved.vue +28 -28
- package/src/components/Icons/Audio/Delete.vue +22 -22
- package/src/components/Icons/Audio/Pause.vue +25 -25
- package/src/components/Icons/Audio/Play.vue +22 -22
- package/src/components/Icons/Calendar.vue +28 -28
- package/src/components/Icons/CalendarNotification.vue +137 -137
- package/src/components/Icons/Chair.vue +43 -43
- package/src/components/Icons/ChairNotification.vue +46 -46
- package/src/components/Icons/Circle.vue +66 -66
- package/src/components/Icons/FavIcon.vue +69 -69
- package/src/components/Icons/FilledCircle.vue +11 -11
- package/src/components/Icons/Group3.vue +57 -57
- package/src/components/Icons/Play.vue +16 -16
- package/src/components/Icons/RingNotification.vue +65 -65
- package/src/components/Icons/SolidArrowRight.vue +14 -14
- package/src/components/Icons/checkbox.vue +19 -19
- package/src/components/Icons/outlineChecked.vue +38 -38
- package/src/components/Input/Input.css +234 -234
- package/src/components/Input/Input.vue +281 -281
- package/src/components/Laboratory/AppointmentCard/AppointmentCard.css +7 -7
- package/src/components/Laboratory/AppointmentCard/AppointmentCard.vue +116 -116
- package/src/components/Laboratory/ChatBoxImage/ChatBoxImage.vue +81 -81
- package/src/components/Laboratory/ChatMessage/ChatMessage.vue +113 -113
- package/src/components/Laboratory/ChatMessage/ChatMessageBadge.css +4 -4
- package/src/components/Laboratory/ChatMessage/ChatMessageBadge.vue +99 -99
- package/src/components/Laboratory/ChatNotification/ChatNotification.vue +130 -130
- package/src/components/Laboratory/DocumentCard/DocumentCard.css +3 -3
- package/src/components/Laboratory/DocumentCard/DocumentCard.vue +50 -50
- package/src/components/Laboratory/DocumentCard/DocumentCardItem.vue +53 -53
- package/src/components/Laboratory/InfoCard/InfoCard.vue +162 -162
- package/src/components/Laboratory/MainColumnsBar/MainColumnsBar.vue +102 -102
- package/src/components/Laboratory/ProgressCircle/ProgressCircle.vue +152 -152
- package/src/components/Laboratory/ProgressLinear/ProgressLinear.css +33 -33
- package/src/components/Laboratory/ProgressLinear/ProgressLinear.vue +75 -75
- package/src/components/Laboratory/SelectionColumnBar/SelectionColumnBar.vue +92 -92
- package/src/components/Laboratory/StatusNotification/StatusNotification.vue +49 -49
- package/src/components/Laboratory/TagLabel/TagLabel.vue +126 -126
- package/src/components/Laboratory/TagLabelGroup/TagLabelGroup.vue +97 -97
- package/src/components/Laboratory/TicketCard/TicketCard.css +3 -3
- package/src/components/Laboratory/TicketCard/TicketCard.vue +143 -143
- package/src/components/Laboratory/TimeLine/TimeLineEvent.css +18 -18
- package/src/components/Laboratory/TimeLine/TimeLineEvent.vue +119 -119
- package/src/components/Laboratory/TimeLine/Timeline.css +4 -4
- package/src/components/Laboratory/TimeLine/Timeline.vue +30 -30
- package/src/components/Loader/Loader.css +78 -78
- package/src/components/MaintenanceBanner/MaintenanceBanner.css +353 -353
- package/src/components/MaintenanceBanner/MaintenanceBanner.vue +140 -140
- package/src/components/MaintenanceBanner/MaintenanceIllustration.vue +54 -54
- package/src/components/Modal/Modal.css +5 -5
- package/src/components/Modal/Modal.vue +29 -29
- package/src/components/NotificationBubble/NotificationBubble.css +4 -4
- package/src/components/NotificationBubble/NotificationBubble.vue +90 -90
- package/src/components/OtpInput/OtpInput.css +43 -43
- package/src/components/OtpInput/OtpInput.vue +181 -181
- package/src/components/PhoneInput/PhoneInput.css +151 -126
- package/src/components/PhoneInput/PhoneInput.vue +230 -139
- package/src/components/RadioGroup/RadioGroup.css +65 -0
- package/src/components/RadioGroup/RadioGroup.vue +134 -0
- package/src/components/Select/Select.css +172 -172
- package/src/components/Select/Select.vue +377 -377
- package/src/components/SelectAutocomplete/SelectAutocomplete.css +172 -172
- package/src/components/SelectAutocomplete/SelectAutocomplete.vue +414 -414
- package/src/components/TextArea/TextArea.css +269 -269
- package/src/components/TextArea/TextArea.vue +207 -207
- package/src/components/TickBox/TickBox.css +116 -116
- package/src/components/TickBox/TickBox.vue +172 -172
- package/src/components/Tile/Tile.css +106 -106
- package/src/components/Tile/Tile.vue +173 -173
- package/src/components/accessibility.css +218 -218
- package/src/components/index.ts +110 -109
- package/src/constants/iconEnums.ts +3 -3
- package/src/i18n/i18n.ts +15 -15
- package/src/i18n/locales/de.json +30 -30
- package/src/i18n/locales/en.json +30 -30
- package/src/index.ts +43 -43
- package/src/main.ts +11 -11
- package/src/pages/AccordionGroupPage.vue +873 -873
- package/src/pages/AllPage.vue +2483 -2365
- package/src/pages/SelectPage.vue +1302 -1302
- package/src/pages/TilePage.vue +902 -902
- package/src/plugins/vuetify.ts +54 -54
- package/src/shims-vue.d.ts +30 -30
- package/src/utils/index.ts +733 -733
- package/src/vite-env.d.ts +1 -1
- package/tests/unit/accessibility/component-a11y.spec.ts +657 -657
- package/tests/unit/components/Accordion/AccordionGroup.spec.ts +228 -228
- package/tests/unit/components/Accordion/AccordionItem.spec.ts +257 -257
- package/tests/unit/components/Appointment/AnamneseNotification.spec.ts +176 -176
- package/tests/unit/components/Appointment/Card/Actions.spec.ts +436 -436
- package/tests/unit/components/Appointment/Card/Card.spec.ts +531 -531
- package/tests/unit/components/Appointment/Card/Details.spec.ts +395 -395
- package/tests/unit/components/Audio/Audio.spec.ts +403 -403
- package/tests/unit/components/Audio/Waveform.spec.ts +483 -483
- package/tests/unit/components/Background/Background.spec.ts +177 -177
- package/tests/unit/components/Core/AnamneseAnswerDialog.spec.ts +344 -0
- package/tests/unit/components/Core/Banner.spec.ts +187 -0
- package/tests/unit/components/Core/Button.spec.ts +346 -346
- package/tests/unit/components/Core/Checkbox.spec.ts +544 -544
- package/tests/unit/components/Core/DateInput.spec.ts +702 -702
- package/tests/unit/components/Core/Dialog.spec.ts +448 -448
- package/tests/unit/components/Core/EditField.spec.ts +541 -541
- package/tests/unit/components/Core/Input.spec.ts +512 -512
- package/tests/unit/components/Core/List.spec.ts +163 -0
- package/tests/unit/components/Core/ListItem.spec.ts +205 -0
- package/tests/unit/components/Core/Modal.spec.ts +518 -518
- package/tests/unit/components/Core/NotificationBubble.spec.ts +606 -606
- package/tests/unit/components/Core/OtpInput.spec.ts +708 -708
- package/tests/unit/components/Core/PhoneInput.spec.ts +757 -619
- package/tests/unit/components/Core/RadioGroup.spec.ts +318 -0
- package/tests/unit/components/Core/Select.spec.ts +712 -712
- package/tests/unit/components/Core/SelectAutocomplete.spec.ts +361 -0
- package/tests/unit/components/Core/TextArea.spec.ts +565 -565
- package/tests/unit/components/Core/TickBox.spec.ts +836 -836
- package/tests/unit/components/Core/Tile.spec.ts +286 -0
- package/tests/unit/components/DateInput/DateInput.spec.ts +128 -0
- package/tests/unit/components/ErrorPage/ErrorPage.spec.ts +313 -313
- package/tests/unit/components/ErrorPage/ErrorPageLogo.spec.ts +153 -153
- package/tests/unit/components/IconBullet/IconBullet.spec.ts +356 -356
- package/tests/unit/components/IconBullet/IconBulletList.spec.ts +371 -371
- package/tests/unit/components/Icons/AdvanceAppointments.spec.ts +186 -186
- package/tests/unit/components/Icons/Audio/CloudFailed.spec.ts +108 -108
- package/tests/unit/components/Icons/Audio/CloudSaved.spec.ts +149 -149
- package/tests/unit/components/Icons/Audio/Delete.spec.ts +158 -158
- package/tests/unit/components/Icons/Audio/Pause.spec.ts +208 -208
- package/tests/unit/components/Icons/Audio/Play.spec.ts +217 -217
- package/tests/unit/components/Icons/CalendarNotification.spec.ts +193 -193
- package/tests/unit/components/Icons/Chair.spec.ts +241 -241
- package/tests/unit/components/Icons/ChairNotification.spec.ts +318 -318
- package/tests/unit/components/Icons/Circle.spec.ts +255 -255
- package/tests/unit/components/Icons/FavIcon.spec.ts +259 -259
- package/tests/unit/components/Icons/FilledCircle.spec.ts +274 -274
- package/tests/unit/components/Icons/Group3.spec.ts +362 -362
- package/tests/unit/components/Icons/Logo.spec.ts +229 -229
- package/tests/unit/components/Icons/MiniLogo.spec.ts +38 -38
- package/tests/unit/components/Icons/RingNotification.spec.ts +400 -400
- package/tests/unit/components/Icons/SolidArrowRight.spec.ts +49 -49
- package/tests/unit/components/Icons/calendar.spec.ts +293 -293
- package/tests/unit/components/Icons/checkbox.spec.ts +315 -315
- package/tests/unit/components/Icons/outlineChecked.spec.ts +441 -441
- package/tests/unit/components/Icons/play.spec.ts +315 -315
- package/tests/unit/components/Laboratory/AppointmentCard.spec.ts +167 -167
- package/tests/unit/components/Laboratory/ChatBoxImage.spec.ts +179 -179
- package/tests/unit/components/Laboratory/ChatMessage.spec.ts +263 -263
- package/tests/unit/components/Laboratory/ChatMessageBadge.spec.ts +282 -282
- package/tests/unit/components/Laboratory/ChatNotification.spec.ts +256 -256
- package/tests/unit/components/Laboratory/DocumentCard.spec.ts +228 -228
- package/tests/unit/components/Laboratory/DocumentCardItem.spec.ts +236 -236
- package/tests/unit/components/Laboratory/InfoCard.spec.ts +308 -308
- package/tests/unit/components/Laboratory/MainColumnsBar.spec.ts +251 -251
- package/tests/unit/components/Laboratory/ProgressCircle.spec.ts +290 -290
- package/tests/unit/components/Laboratory/ProgressLinear.spec.ts +275 -275
- package/tests/unit/components/Laboratory/SelectionColumnBar.spec.ts +288 -288
- package/tests/unit/components/Laboratory/StatusNotification.spec.ts +296 -296
- package/tests/unit/components/Laboratory/TagLabel.spec.ts +353 -353
- package/tests/unit/components/Laboratory/TagLabelGroup.spec.ts +377 -377
- package/tests/unit/components/Laboratory/TicketCard.spec.ts +351 -351
- package/tests/unit/components/Laboratory/TimeLineEvent.spec.ts +381 -381
- package/tests/unit/components/Laboratory/Timeline.spec.ts +419 -419
- package/tests/unit/components/Loader/Loader.spec.ts +197 -197
- package/tests/unit/components/MaintenanceBanner/MaintenanceBanner.spec.ts +302 -302
- package/tests/unit/constants/iconEnums.spec.ts +39 -39
- package/tests/unit/i18n/i18n.spec.ts +88 -88
- package/tests/unit/plugins/vuetify.spec.ts +182 -182
- package/tests/unit/setup.ts +237 -237
- package/tests/unit/src/components/index.spec.ts.skip +192 -192
- package/tests/unit/src/index.spec.ts.skip +182 -182
- package/tests/unit/src/main.spec.ts +111 -111
- package/tests/unit/utils/accessibility.spec.ts +318 -318
- package/tests/unit/utils/anamnese.spec.ts +531 -0
- package/tsconfig.json +26 -26
- package/vite.config.ts +29 -29
- package/vitest.config.ts +91 -91
|
@@ -1,414 +1,414 @@
|
|
|
1
|
-
<template>
|
|
2
|
-
<v-autocomplete
|
|
3
|
-
ref="inputRef"
|
|
4
|
-
class="wl-select"
|
|
5
|
-
:class="{
|
|
6
|
-
'border-on-hover': borderOnHover
|
|
7
|
-
}"
|
|
8
|
-
rounded="lg"
|
|
9
|
-
:density="density"
|
|
10
|
-
:variant="props.variant"
|
|
11
|
-
:label="label"
|
|
12
|
-
:chips="chips"
|
|
13
|
-
:append-icon="appendIcon"
|
|
14
|
-
:append-inner-icon="appendInnerIcon"
|
|
15
|
-
:prepend-icon="prependIcon"
|
|
16
|
-
:prepend-inner-icon="prependInnerIcon"
|
|
17
|
-
:menu-icon="menuIcon"
|
|
18
|
-
:clear-icon="clearIcon"
|
|
19
|
-
:closable-chips="closableChips"
|
|
20
|
-
:multiple="multiple"
|
|
21
|
-
:clearable="clearable"
|
|
22
|
-
:items="items"
|
|
23
|
-
:disabled="disabled"
|
|
24
|
-
:error="error"
|
|
25
|
-
:item-title="itemTitle"
|
|
26
|
-
:item-value="itemValue"
|
|
27
|
-
:return-object="returnObject"
|
|
28
|
-
:placeholder="placeholder"
|
|
29
|
-
:persistent-placeholder="persistentPlaceholder"
|
|
30
|
-
:auto-select-first="autoSelectFirst"
|
|
31
|
-
:custom-filter="customFilter"
|
|
32
|
-
:hide-no-data="hideNoData"
|
|
33
|
-
:hide-details="hideDetails"
|
|
34
|
-
:aria-invalid="error || undefined"
|
|
35
|
-
:aria-label="ariaLabel || undefined"
|
|
36
|
-
:aria-describedby="ariaDescribedby || undefined"
|
|
37
|
-
:menu-props="computedMenuProps"
|
|
38
|
-
v-model="internalValue"
|
|
39
|
-
@click:append="onClickAppend"
|
|
40
|
-
@click:append-inner="onClickAppendInner"
|
|
41
|
-
@click:clear="onClickClear"
|
|
42
|
-
@click:prepend="onClickPrepend"
|
|
43
|
-
@click:prepend-inner="onClickPrependInner"
|
|
44
|
-
@update:search="onUpdateSearch"
|
|
45
|
-
data-testid="root"
|
|
46
|
-
:rules="[v => !!v || 'Please select a value from the list']"
|
|
47
|
-
>
|
|
48
|
-
<template v-slot:menu>
|
|
49
|
-
<slot name="menu"></slot>
|
|
50
|
-
</template>
|
|
51
|
-
|
|
52
|
-
<template v-if="$slots['item']" #item="slotProps">
|
|
53
|
-
<slot name="item" v-bind="slotProps"></slot>
|
|
54
|
-
</template>
|
|
55
|
-
|
|
56
|
-
<template v-if="$slots['selection']" #selection="slotProps">
|
|
57
|
-
<slot name="selection" v-bind="slotProps"></slot>
|
|
58
|
-
</template>
|
|
59
|
-
|
|
60
|
-
<template v-if="$slots['no-data']" #no-data>
|
|
61
|
-
<slot name="no-data"></slot>
|
|
62
|
-
</template>
|
|
63
|
-
|
|
64
|
-
<template v-if="$slots['prepend']" #prepend>
|
|
65
|
-
<slot name="prepend"></slot>
|
|
66
|
-
</template>
|
|
67
|
-
|
|
68
|
-
<template v-if="$slots['prepend-inner']" #prepend-inner>
|
|
69
|
-
<slot name="prepend-inner"></slot>
|
|
70
|
-
</template>
|
|
71
|
-
|
|
72
|
-
<template v-if="$slots['append']" #append>
|
|
73
|
-
<slot name="append"></slot>
|
|
74
|
-
</template>
|
|
75
|
-
|
|
76
|
-
<template v-if="$slots['append-inner']" #append-inner>
|
|
77
|
-
<slot name="append-inner"></slot>
|
|
78
|
-
</template>
|
|
79
|
-
</v-autocomplete>
|
|
80
|
-
</template>
|
|
81
|
-
|
|
82
|
-
<script setup>
|
|
83
|
-
import './SelectAutocomplete.css';
|
|
84
|
-
import { ref, watch, nextTick, computed, inject } from 'vue';
|
|
85
|
-
import { defineProps, defineEmits } from 'vue';
|
|
86
|
-
import { siteColors } from "../../utils/index";
|
|
87
|
-
|
|
88
|
-
// Inject theme colors from ThemeProvider, fallback to global siteColors
|
|
89
|
-
const injectedThemeColors = inject('themeColors', null);
|
|
90
|
-
const colors = computed(() => {
|
|
91
|
-
if (injectedThemeColors) {
|
|
92
|
-
return injectedThemeColors.value;
|
|
93
|
-
}
|
|
94
|
-
return siteColors;
|
|
95
|
-
});
|
|
96
|
-
|
|
97
|
-
// Compute menu props with domain-specific class and inline CSS variables for dropdown styling
|
|
98
|
-
const computedMenuProps = computed(() => ({
|
|
99
|
-
class: 'wl-select-menu',
|
|
100
|
-
style: {
|
|
101
|
-
'--select-item-text': colors.value['select_item_text'],
|
|
102
|
-
'--select-item-default-bg': colors.value['select_item_default_bg'],
|
|
103
|
-
'--select-item-hovered-bg': colors.value['select_item_hovered_bg'],
|
|
104
|
-
'--select-item-selected-left-border': colors.value['select_item_selected_left_border'],
|
|
105
|
-
'--select-item-selected-bg': colors.value['select_item_selected_bg']
|
|
106
|
-
}
|
|
107
|
-
}));
|
|
108
|
-
|
|
109
|
-
// Define component props with default values and descriptions
|
|
110
|
-
const props = defineProps({
|
|
111
|
-
/**
|
|
112
|
-
* Disables the select input when true, making it non-interactive.
|
|
113
|
-
* Default is false.
|
|
114
|
-
*/
|
|
115
|
-
disabled: {
|
|
116
|
-
type: Boolean,
|
|
117
|
-
default: false,
|
|
118
|
-
},
|
|
119
|
-
|
|
120
|
-
/**
|
|
121
|
-
* Changes the select input state, when true, put it into error state.
|
|
122
|
-
* Default is false.
|
|
123
|
-
*/
|
|
124
|
-
error: {
|
|
125
|
-
type: Boolean,
|
|
126
|
-
default: false,
|
|
127
|
-
},
|
|
128
|
-
|
|
129
|
-
/**
|
|
130
|
-
* Allows clearing the selected value when true.
|
|
131
|
-
* Default is true.
|
|
132
|
-
*/
|
|
133
|
-
clearable: {
|
|
134
|
-
type: Boolean,
|
|
135
|
-
default: false,
|
|
136
|
-
},
|
|
137
|
-
|
|
138
|
-
/**
|
|
139
|
-
* The label displayed above the select input.
|
|
140
|
-
* Default is 'Label'.
|
|
141
|
-
*/
|
|
142
|
-
label: {
|
|
143
|
-
type: String,
|
|
144
|
-
default: 'Label',
|
|
145
|
-
},
|
|
146
|
-
|
|
147
|
-
/**
|
|
148
|
-
* Enables the display of selected items as chips when true.
|
|
149
|
-
* Default is false.
|
|
150
|
-
*/
|
|
151
|
-
chips: {
|
|
152
|
-
type: Boolean,
|
|
153
|
-
default: false,
|
|
154
|
-
},
|
|
155
|
-
|
|
156
|
-
/**
|
|
157
|
-
* Icon to append to the select field.
|
|
158
|
-
*/
|
|
159
|
-
appendIcon: {
|
|
160
|
-
type: String,
|
|
161
|
-
},
|
|
162
|
-
|
|
163
|
-
/**
|
|
164
|
-
* Inner icon to append inside the select field.
|
|
165
|
-
*/
|
|
166
|
-
appendInnerIcon: {
|
|
167
|
-
type: String,
|
|
168
|
-
},
|
|
169
|
-
|
|
170
|
-
/**
|
|
171
|
-
* Icon to prepend to the select field.
|
|
172
|
-
*/
|
|
173
|
-
prependIcon: {
|
|
174
|
-
type: String,
|
|
175
|
-
},
|
|
176
|
-
|
|
177
|
-
/**
|
|
178
|
-
* Inner icon to prepend inside the select field.
|
|
179
|
-
*/
|
|
180
|
-
prependInnerIcon: {
|
|
181
|
-
type: String,
|
|
182
|
-
},
|
|
183
|
-
|
|
184
|
-
/**
|
|
185
|
-
* The icon for the menu toggle.
|
|
186
|
-
* Default is an empty string (no icon).
|
|
187
|
-
*/
|
|
188
|
-
menuIcon: {
|
|
189
|
-
type: String,
|
|
190
|
-
default: '',
|
|
191
|
-
},
|
|
192
|
-
|
|
193
|
-
/**
|
|
194
|
-
* The icon to use for the clear button.
|
|
195
|
-
* Default is 'x-mark'.
|
|
196
|
-
*/
|
|
197
|
-
clearIcon: {
|
|
198
|
-
type: String,
|
|
199
|
-
default: 'x-mark',
|
|
200
|
-
},
|
|
201
|
-
|
|
202
|
-
/**
|
|
203
|
-
* Enables closing chips when true.
|
|
204
|
-
* Default is false.
|
|
205
|
-
*/
|
|
206
|
-
closableChips: {
|
|
207
|
-
type: Boolean,
|
|
208
|
-
default: false,
|
|
209
|
-
},
|
|
210
|
-
|
|
211
|
-
/**
|
|
212
|
-
* Allows multiple selections when true.
|
|
213
|
-
* Default is false.
|
|
214
|
-
*/
|
|
215
|
-
multiple: {
|
|
216
|
-
type: Boolean,
|
|
217
|
-
default: false,
|
|
218
|
-
},
|
|
219
|
-
|
|
220
|
-
/**
|
|
221
|
-
* The items available for selection.
|
|
222
|
-
* Default is an empty array.
|
|
223
|
-
*/
|
|
224
|
-
items: {
|
|
225
|
-
type: Array,
|
|
226
|
-
default: () => [],
|
|
227
|
-
},
|
|
228
|
-
|
|
229
|
-
/**
|
|
230
|
-
* The selected value (can be a string, array, or object).
|
|
231
|
-
*/
|
|
232
|
-
modelValue: {
|
|
233
|
-
type: [String, Array, Object],
|
|
234
|
-
default: undefined,
|
|
235
|
-
},
|
|
236
|
-
|
|
237
|
-
/**
|
|
238
|
-
* The density of the select input.
|
|
239
|
-
* Controls spacing and size. Default is 'compact'.
|
|
240
|
-
*/
|
|
241
|
-
density: {
|
|
242
|
-
type: String,
|
|
243
|
-
default: 'compact',
|
|
244
|
-
},
|
|
245
|
-
|
|
246
|
-
/**
|
|
247
|
-
* When true, the selected item is returned as an object.
|
|
248
|
-
* Default is true.
|
|
249
|
-
*/
|
|
250
|
-
returnObject: {
|
|
251
|
-
type: Boolean,
|
|
252
|
-
default: true,
|
|
253
|
-
},
|
|
254
|
-
|
|
255
|
-
/**
|
|
256
|
-
* The property name of the item to display as the title.
|
|
257
|
-
*/
|
|
258
|
-
itemTitle: {
|
|
259
|
-
type: String,
|
|
260
|
-
default: 'title',
|
|
261
|
-
},
|
|
262
|
-
|
|
263
|
-
/**
|
|
264
|
-
* The property name of the item to use as the value.
|
|
265
|
-
*/
|
|
266
|
-
itemValue: {
|
|
267
|
-
type: String,
|
|
268
|
-
default: 'value',
|
|
269
|
-
},
|
|
270
|
-
|
|
271
|
-
variant: {
|
|
272
|
-
type: String,
|
|
273
|
-
default: 'outlined',
|
|
274
|
-
},
|
|
275
|
-
|
|
276
|
-
borderOnHover: {
|
|
277
|
-
type: Boolean,
|
|
278
|
-
default: false
|
|
279
|
-
},
|
|
280
|
-
|
|
281
|
-
placeholder: {
|
|
282
|
-
type: String,
|
|
283
|
-
default: ''
|
|
284
|
-
},
|
|
285
|
-
persistentPlaceholder:{
|
|
286
|
-
type: Boolean,
|
|
287
|
-
default: false
|
|
288
|
-
},
|
|
289
|
-
|
|
290
|
-
/**
|
|
291
|
-
* Automatically highlights the first matching item so it gets selected
|
|
292
|
-
* when the user blurs or presses Enter without manually picking an item.
|
|
293
|
-
* Default is false.
|
|
294
|
-
*/
|
|
295
|
-
autoSelectFirst: {
|
|
296
|
-
type: Boolean,
|
|
297
|
-
default: false,
|
|
298
|
-
},
|
|
299
|
-
|
|
300
|
-
/**
|
|
301
|
-
* Custom filter function for matching items against the search query.
|
|
302
|
-
* Signature: (value: string, query: string, item: { raw: object }) => boolean
|
|
303
|
-
* When not provided, Vuetify's default filter (item-title match) is used.
|
|
304
|
-
*/
|
|
305
|
-
customFilter: {
|
|
306
|
-
type: Function,
|
|
307
|
-
default: undefined,
|
|
308
|
-
},
|
|
309
|
-
|
|
310
|
-
/**
|
|
311
|
-
* Hides the menu when there are no items to display.
|
|
312
|
-
* Default is false.
|
|
313
|
-
*/
|
|
314
|
-
hideNoData: {
|
|
315
|
-
type: Boolean,
|
|
316
|
-
default: false,
|
|
317
|
-
},
|
|
318
|
-
|
|
319
|
-
/**
|
|
320
|
-
* Controls whether validation messages (errors, hints) are displayed below the field.
|
|
321
|
-
* Default is true (hidden) for backwards compatibility.
|
|
322
|
-
* Set to false or 'auto' to show validation/error messages.
|
|
323
|
-
*/
|
|
324
|
-
hideDetails: {
|
|
325
|
-
type: [Boolean, String],
|
|
326
|
-
default: true,
|
|
327
|
-
},
|
|
328
|
-
|
|
329
|
-
/**
|
|
330
|
-
* Accessible label for screen readers. Use when the visible label is empty
|
|
331
|
-
* or not descriptive enough (WCAG 4.1.2).
|
|
332
|
-
*/
|
|
333
|
-
ariaLabel: {
|
|
334
|
-
type: String,
|
|
335
|
-
default: undefined,
|
|
336
|
-
},
|
|
337
|
-
|
|
338
|
-
/**
|
|
339
|
-
* ID of the element(s) that describe this input (e.g. error messages, hints).
|
|
340
|
-
* Links the input to supplementary information for screen readers (WCAG 1.3.1).
|
|
341
|
-
*/
|
|
342
|
-
ariaDescribedby: {
|
|
343
|
-
type: String,
|
|
344
|
-
default: undefined,
|
|
345
|
-
},
|
|
346
|
-
});
|
|
347
|
-
|
|
348
|
-
// Define events that can be emitted by the component
|
|
349
|
-
const emits = defineEmits([
|
|
350
|
-
'update:modelValue', // Emit when the model value is updated
|
|
351
|
-
'click:append', // Emit when the append icon is clicked
|
|
352
|
-
'click:appendInner', // Emit when the append inner icon is clicked
|
|
353
|
-
'click:clear', // Emit when the clear button is clicked
|
|
354
|
-
'click:prepend', // Emit when the prepend icon is clicked
|
|
355
|
-
'click:prependInner', // Emit when the prepend inner icon is clicked
|
|
356
|
-
'update:search', // Emit search updates for filtering items
|
|
357
|
-
]);
|
|
358
|
-
|
|
359
|
-
// Define a reactive reference for internal value based on the modelValue prop
|
|
360
|
-
const internalValue = ref(props.modelValue);
|
|
361
|
-
|
|
362
|
-
// Watch for changes in modelValue prop and update internalValue accordingly
|
|
363
|
-
watch(
|
|
364
|
-
() => props.modelValue,
|
|
365
|
-
(newValue) => {
|
|
366
|
-
internalValue.value = newValue;
|
|
367
|
-
},
|
|
368
|
-
);
|
|
369
|
-
|
|
370
|
-
// Watch for changes in internalValue and emit updated modelValue to parent
|
|
371
|
-
watch(internalValue, (newValue) => {
|
|
372
|
-
emits('update:modelValue', newValue);
|
|
373
|
-
});
|
|
374
|
-
|
|
375
|
-
// Emit click events for various icon and button interactions
|
|
376
|
-
const onClickAppend = (event) => emits('click:append', event);
|
|
377
|
-
const onClickAppendInner = (event) => emits('click:appendInner', event);
|
|
378
|
-
const onClickClear = (event) => emits('click:clear', event);
|
|
379
|
-
const onClickPrepend = (event) => emits('click:prepend', event);
|
|
380
|
-
const onClickPrependInner = (event) => emits('click:prependInner', event);
|
|
381
|
-
|
|
382
|
-
// Emit search updates for filtering or searching items
|
|
383
|
-
const onUpdateSearch = (searchValue) => emits('update:search', searchValue);
|
|
384
|
-
|
|
385
|
-
const inputRef = ref(null)
|
|
386
|
-
|
|
387
|
-
const getNativeInput = () => {
|
|
388
|
-
return inputRef.value?.$el?.querySelector('input');
|
|
389
|
-
}
|
|
390
|
-
|
|
391
|
-
const blur = () => {
|
|
392
|
-
nextTick(() => {
|
|
393
|
-
getNativeInput()?.blur();
|
|
394
|
-
});
|
|
395
|
-
};
|
|
396
|
-
|
|
397
|
-
const select = () => {
|
|
398
|
-
nextTick(() => {
|
|
399
|
-
getNativeInput()?.select();
|
|
400
|
-
});
|
|
401
|
-
};
|
|
402
|
-
|
|
403
|
-
const focus = () => {
|
|
404
|
-
nextTick(() => {
|
|
405
|
-
getNativeInput()?.focus();
|
|
406
|
-
});
|
|
407
|
-
};
|
|
408
|
-
|
|
409
|
-
defineExpose({
|
|
410
|
-
blur,
|
|
411
|
-
select,
|
|
412
|
-
focus
|
|
413
|
-
});
|
|
414
|
-
</script>
|
|
1
|
+
<template>
|
|
2
|
+
<v-autocomplete
|
|
3
|
+
ref="inputRef"
|
|
4
|
+
class="wl-select"
|
|
5
|
+
:class="{
|
|
6
|
+
'border-on-hover': borderOnHover
|
|
7
|
+
}"
|
|
8
|
+
rounded="lg"
|
|
9
|
+
:density="density"
|
|
10
|
+
:variant="props.variant"
|
|
11
|
+
:label="label"
|
|
12
|
+
:chips="chips"
|
|
13
|
+
:append-icon="appendIcon"
|
|
14
|
+
:append-inner-icon="appendInnerIcon"
|
|
15
|
+
:prepend-icon="prependIcon"
|
|
16
|
+
:prepend-inner-icon="prependInnerIcon"
|
|
17
|
+
:menu-icon="menuIcon"
|
|
18
|
+
:clear-icon="clearIcon"
|
|
19
|
+
:closable-chips="closableChips"
|
|
20
|
+
:multiple="multiple"
|
|
21
|
+
:clearable="clearable"
|
|
22
|
+
:items="items"
|
|
23
|
+
:disabled="disabled"
|
|
24
|
+
:error="error"
|
|
25
|
+
:item-title="itemTitle"
|
|
26
|
+
:item-value="itemValue"
|
|
27
|
+
:return-object="returnObject"
|
|
28
|
+
:placeholder="placeholder"
|
|
29
|
+
:persistent-placeholder="persistentPlaceholder"
|
|
30
|
+
:auto-select-first="autoSelectFirst"
|
|
31
|
+
:custom-filter="customFilter"
|
|
32
|
+
:hide-no-data="hideNoData"
|
|
33
|
+
:hide-details="hideDetails"
|
|
34
|
+
:aria-invalid="error || undefined"
|
|
35
|
+
:aria-label="ariaLabel || undefined"
|
|
36
|
+
:aria-describedby="ariaDescribedby || undefined"
|
|
37
|
+
:menu-props="computedMenuProps"
|
|
38
|
+
v-model="internalValue"
|
|
39
|
+
@click:append="onClickAppend"
|
|
40
|
+
@click:append-inner="onClickAppendInner"
|
|
41
|
+
@click:clear="onClickClear"
|
|
42
|
+
@click:prepend="onClickPrepend"
|
|
43
|
+
@click:prepend-inner="onClickPrependInner"
|
|
44
|
+
@update:search="onUpdateSearch"
|
|
45
|
+
data-testid="root"
|
|
46
|
+
:rules="[v => !!v || 'Please select a value from the list']"
|
|
47
|
+
>
|
|
48
|
+
<template v-slot:menu>
|
|
49
|
+
<slot name="menu"></slot>
|
|
50
|
+
</template>
|
|
51
|
+
|
|
52
|
+
<template v-if="$slots['item']" #item="slotProps">
|
|
53
|
+
<slot name="item" v-bind="slotProps"></slot>
|
|
54
|
+
</template>
|
|
55
|
+
|
|
56
|
+
<template v-if="$slots['selection']" #selection="slotProps">
|
|
57
|
+
<slot name="selection" v-bind="slotProps"></slot>
|
|
58
|
+
</template>
|
|
59
|
+
|
|
60
|
+
<template v-if="$slots['no-data']" #no-data>
|
|
61
|
+
<slot name="no-data"></slot>
|
|
62
|
+
</template>
|
|
63
|
+
|
|
64
|
+
<template v-if="$slots['prepend']" #prepend>
|
|
65
|
+
<slot name="prepend"></slot>
|
|
66
|
+
</template>
|
|
67
|
+
|
|
68
|
+
<template v-if="$slots['prepend-inner']" #prepend-inner>
|
|
69
|
+
<slot name="prepend-inner"></slot>
|
|
70
|
+
</template>
|
|
71
|
+
|
|
72
|
+
<template v-if="$slots['append']" #append>
|
|
73
|
+
<slot name="append"></slot>
|
|
74
|
+
</template>
|
|
75
|
+
|
|
76
|
+
<template v-if="$slots['append-inner']" #append-inner>
|
|
77
|
+
<slot name="append-inner"></slot>
|
|
78
|
+
</template>
|
|
79
|
+
</v-autocomplete>
|
|
80
|
+
</template>
|
|
81
|
+
|
|
82
|
+
<script setup>
|
|
83
|
+
import './SelectAutocomplete.css';
|
|
84
|
+
import { ref, watch, nextTick, computed, inject } from 'vue';
|
|
85
|
+
import { defineProps, defineEmits } from 'vue';
|
|
86
|
+
import { siteColors } from "../../utils/index";
|
|
87
|
+
|
|
88
|
+
// Inject theme colors from ThemeProvider, fallback to global siteColors
|
|
89
|
+
const injectedThemeColors = inject('themeColors', null);
|
|
90
|
+
const colors = computed(() => {
|
|
91
|
+
if (injectedThemeColors) {
|
|
92
|
+
return injectedThemeColors.value;
|
|
93
|
+
}
|
|
94
|
+
return siteColors;
|
|
95
|
+
});
|
|
96
|
+
|
|
97
|
+
// Compute menu props with domain-specific class and inline CSS variables for dropdown styling
|
|
98
|
+
const computedMenuProps = computed(() => ({
|
|
99
|
+
class: 'wl-select-menu',
|
|
100
|
+
style: {
|
|
101
|
+
'--select-item-text': colors.value['select_item_text'],
|
|
102
|
+
'--select-item-default-bg': colors.value['select_item_default_bg'],
|
|
103
|
+
'--select-item-hovered-bg': colors.value['select_item_hovered_bg'],
|
|
104
|
+
'--select-item-selected-left-border': colors.value['select_item_selected_left_border'],
|
|
105
|
+
'--select-item-selected-bg': colors.value['select_item_selected_bg']
|
|
106
|
+
}
|
|
107
|
+
}));
|
|
108
|
+
|
|
109
|
+
// Define component props with default values and descriptions
|
|
110
|
+
const props = defineProps({
|
|
111
|
+
/**
|
|
112
|
+
* Disables the select input when true, making it non-interactive.
|
|
113
|
+
* Default is false.
|
|
114
|
+
*/
|
|
115
|
+
disabled: {
|
|
116
|
+
type: Boolean,
|
|
117
|
+
default: false,
|
|
118
|
+
},
|
|
119
|
+
|
|
120
|
+
/**
|
|
121
|
+
* Changes the select input state, when true, put it into error state.
|
|
122
|
+
* Default is false.
|
|
123
|
+
*/
|
|
124
|
+
error: {
|
|
125
|
+
type: Boolean,
|
|
126
|
+
default: false,
|
|
127
|
+
},
|
|
128
|
+
|
|
129
|
+
/**
|
|
130
|
+
* Allows clearing the selected value when true.
|
|
131
|
+
* Default is true.
|
|
132
|
+
*/
|
|
133
|
+
clearable: {
|
|
134
|
+
type: Boolean,
|
|
135
|
+
default: false,
|
|
136
|
+
},
|
|
137
|
+
|
|
138
|
+
/**
|
|
139
|
+
* The label displayed above the select input.
|
|
140
|
+
* Default is 'Label'.
|
|
141
|
+
*/
|
|
142
|
+
label: {
|
|
143
|
+
type: String,
|
|
144
|
+
default: 'Label',
|
|
145
|
+
},
|
|
146
|
+
|
|
147
|
+
/**
|
|
148
|
+
* Enables the display of selected items as chips when true.
|
|
149
|
+
* Default is false.
|
|
150
|
+
*/
|
|
151
|
+
chips: {
|
|
152
|
+
type: Boolean,
|
|
153
|
+
default: false,
|
|
154
|
+
},
|
|
155
|
+
|
|
156
|
+
/**
|
|
157
|
+
* Icon to append to the select field.
|
|
158
|
+
*/
|
|
159
|
+
appendIcon: {
|
|
160
|
+
type: String,
|
|
161
|
+
},
|
|
162
|
+
|
|
163
|
+
/**
|
|
164
|
+
* Inner icon to append inside the select field.
|
|
165
|
+
*/
|
|
166
|
+
appendInnerIcon: {
|
|
167
|
+
type: String,
|
|
168
|
+
},
|
|
169
|
+
|
|
170
|
+
/**
|
|
171
|
+
* Icon to prepend to the select field.
|
|
172
|
+
*/
|
|
173
|
+
prependIcon: {
|
|
174
|
+
type: String,
|
|
175
|
+
},
|
|
176
|
+
|
|
177
|
+
/**
|
|
178
|
+
* Inner icon to prepend inside the select field.
|
|
179
|
+
*/
|
|
180
|
+
prependInnerIcon: {
|
|
181
|
+
type: String,
|
|
182
|
+
},
|
|
183
|
+
|
|
184
|
+
/**
|
|
185
|
+
* The icon for the menu toggle.
|
|
186
|
+
* Default is an empty string (no icon).
|
|
187
|
+
*/
|
|
188
|
+
menuIcon: {
|
|
189
|
+
type: String,
|
|
190
|
+
default: '',
|
|
191
|
+
},
|
|
192
|
+
|
|
193
|
+
/**
|
|
194
|
+
* The icon to use for the clear button.
|
|
195
|
+
* Default is 'x-mark'.
|
|
196
|
+
*/
|
|
197
|
+
clearIcon: {
|
|
198
|
+
type: String,
|
|
199
|
+
default: 'x-mark',
|
|
200
|
+
},
|
|
201
|
+
|
|
202
|
+
/**
|
|
203
|
+
* Enables closing chips when true.
|
|
204
|
+
* Default is false.
|
|
205
|
+
*/
|
|
206
|
+
closableChips: {
|
|
207
|
+
type: Boolean,
|
|
208
|
+
default: false,
|
|
209
|
+
},
|
|
210
|
+
|
|
211
|
+
/**
|
|
212
|
+
* Allows multiple selections when true.
|
|
213
|
+
* Default is false.
|
|
214
|
+
*/
|
|
215
|
+
multiple: {
|
|
216
|
+
type: Boolean,
|
|
217
|
+
default: false,
|
|
218
|
+
},
|
|
219
|
+
|
|
220
|
+
/**
|
|
221
|
+
* The items available for selection.
|
|
222
|
+
* Default is an empty array.
|
|
223
|
+
*/
|
|
224
|
+
items: {
|
|
225
|
+
type: Array,
|
|
226
|
+
default: () => [],
|
|
227
|
+
},
|
|
228
|
+
|
|
229
|
+
/**
|
|
230
|
+
* The selected value (can be a string, array, or object).
|
|
231
|
+
*/
|
|
232
|
+
modelValue: {
|
|
233
|
+
type: [String, Array, Object],
|
|
234
|
+
default: undefined,
|
|
235
|
+
},
|
|
236
|
+
|
|
237
|
+
/**
|
|
238
|
+
* The density of the select input.
|
|
239
|
+
* Controls spacing and size. Default is 'compact'.
|
|
240
|
+
*/
|
|
241
|
+
density: {
|
|
242
|
+
type: String,
|
|
243
|
+
default: 'compact',
|
|
244
|
+
},
|
|
245
|
+
|
|
246
|
+
/**
|
|
247
|
+
* When true, the selected item is returned as an object.
|
|
248
|
+
* Default is true.
|
|
249
|
+
*/
|
|
250
|
+
returnObject: {
|
|
251
|
+
type: Boolean,
|
|
252
|
+
default: true,
|
|
253
|
+
},
|
|
254
|
+
|
|
255
|
+
/**
|
|
256
|
+
* The property name of the item to display as the title.
|
|
257
|
+
*/
|
|
258
|
+
itemTitle: {
|
|
259
|
+
type: String,
|
|
260
|
+
default: 'title',
|
|
261
|
+
},
|
|
262
|
+
|
|
263
|
+
/**
|
|
264
|
+
* The property name of the item to use as the value.
|
|
265
|
+
*/
|
|
266
|
+
itemValue: {
|
|
267
|
+
type: String,
|
|
268
|
+
default: 'value',
|
|
269
|
+
},
|
|
270
|
+
|
|
271
|
+
variant: {
|
|
272
|
+
type: String,
|
|
273
|
+
default: 'outlined',
|
|
274
|
+
},
|
|
275
|
+
|
|
276
|
+
borderOnHover: {
|
|
277
|
+
type: Boolean,
|
|
278
|
+
default: false
|
|
279
|
+
},
|
|
280
|
+
|
|
281
|
+
placeholder: {
|
|
282
|
+
type: String,
|
|
283
|
+
default: ''
|
|
284
|
+
},
|
|
285
|
+
persistentPlaceholder:{
|
|
286
|
+
type: Boolean,
|
|
287
|
+
default: false
|
|
288
|
+
},
|
|
289
|
+
|
|
290
|
+
/**
|
|
291
|
+
* Automatically highlights the first matching item so it gets selected
|
|
292
|
+
* when the user blurs or presses Enter without manually picking an item.
|
|
293
|
+
* Default is false.
|
|
294
|
+
*/
|
|
295
|
+
autoSelectFirst: {
|
|
296
|
+
type: Boolean,
|
|
297
|
+
default: false,
|
|
298
|
+
},
|
|
299
|
+
|
|
300
|
+
/**
|
|
301
|
+
* Custom filter function for matching items against the search query.
|
|
302
|
+
* Signature: (value: string, query: string, item: { raw: object }) => boolean
|
|
303
|
+
* When not provided, Vuetify's default filter (item-title match) is used.
|
|
304
|
+
*/
|
|
305
|
+
customFilter: {
|
|
306
|
+
type: Function,
|
|
307
|
+
default: undefined,
|
|
308
|
+
},
|
|
309
|
+
|
|
310
|
+
/**
|
|
311
|
+
* Hides the menu when there are no items to display.
|
|
312
|
+
* Default is false.
|
|
313
|
+
*/
|
|
314
|
+
hideNoData: {
|
|
315
|
+
type: Boolean,
|
|
316
|
+
default: false,
|
|
317
|
+
},
|
|
318
|
+
|
|
319
|
+
/**
|
|
320
|
+
* Controls whether validation messages (errors, hints) are displayed below the field.
|
|
321
|
+
* Default is true (hidden) for backwards compatibility.
|
|
322
|
+
* Set to false or 'auto' to show validation/error messages.
|
|
323
|
+
*/
|
|
324
|
+
hideDetails: {
|
|
325
|
+
type: [Boolean, String],
|
|
326
|
+
default: true,
|
|
327
|
+
},
|
|
328
|
+
|
|
329
|
+
/**
|
|
330
|
+
* Accessible label for screen readers. Use when the visible label is empty
|
|
331
|
+
* or not descriptive enough (WCAG 4.1.2).
|
|
332
|
+
*/
|
|
333
|
+
ariaLabel: {
|
|
334
|
+
type: String,
|
|
335
|
+
default: undefined,
|
|
336
|
+
},
|
|
337
|
+
|
|
338
|
+
/**
|
|
339
|
+
* ID of the element(s) that describe this input (e.g. error messages, hints).
|
|
340
|
+
* Links the input to supplementary information for screen readers (WCAG 1.3.1).
|
|
341
|
+
*/
|
|
342
|
+
ariaDescribedby: {
|
|
343
|
+
type: String,
|
|
344
|
+
default: undefined,
|
|
345
|
+
},
|
|
346
|
+
});
|
|
347
|
+
|
|
348
|
+
// Define events that can be emitted by the component
|
|
349
|
+
const emits = defineEmits([
|
|
350
|
+
'update:modelValue', // Emit when the model value is updated
|
|
351
|
+
'click:append', // Emit when the append icon is clicked
|
|
352
|
+
'click:appendInner', // Emit when the append inner icon is clicked
|
|
353
|
+
'click:clear', // Emit when the clear button is clicked
|
|
354
|
+
'click:prepend', // Emit when the prepend icon is clicked
|
|
355
|
+
'click:prependInner', // Emit when the prepend inner icon is clicked
|
|
356
|
+
'update:search', // Emit search updates for filtering items
|
|
357
|
+
]);
|
|
358
|
+
|
|
359
|
+
// Define a reactive reference for internal value based on the modelValue prop
|
|
360
|
+
const internalValue = ref(props.modelValue);
|
|
361
|
+
|
|
362
|
+
// Watch for changes in modelValue prop and update internalValue accordingly
|
|
363
|
+
watch(
|
|
364
|
+
() => props.modelValue,
|
|
365
|
+
(newValue) => {
|
|
366
|
+
internalValue.value = newValue;
|
|
367
|
+
},
|
|
368
|
+
);
|
|
369
|
+
|
|
370
|
+
// Watch for changes in internalValue and emit updated modelValue to parent
|
|
371
|
+
watch(internalValue, (newValue) => {
|
|
372
|
+
emits('update:modelValue', newValue);
|
|
373
|
+
});
|
|
374
|
+
|
|
375
|
+
// Emit click events for various icon and button interactions
|
|
376
|
+
const onClickAppend = (event) => emits('click:append', event);
|
|
377
|
+
const onClickAppendInner = (event) => emits('click:appendInner', event);
|
|
378
|
+
const onClickClear = (event) => emits('click:clear', event);
|
|
379
|
+
const onClickPrepend = (event) => emits('click:prepend', event);
|
|
380
|
+
const onClickPrependInner = (event) => emits('click:prependInner', event);
|
|
381
|
+
|
|
382
|
+
// Emit search updates for filtering or searching items
|
|
383
|
+
const onUpdateSearch = (searchValue) => emits('update:search', searchValue);
|
|
384
|
+
|
|
385
|
+
const inputRef = ref(null)
|
|
386
|
+
|
|
387
|
+
const getNativeInput = () => {
|
|
388
|
+
return inputRef.value?.$el?.querySelector('input');
|
|
389
|
+
}
|
|
390
|
+
|
|
391
|
+
const blur = () => {
|
|
392
|
+
nextTick(() => {
|
|
393
|
+
getNativeInput()?.blur();
|
|
394
|
+
});
|
|
395
|
+
};
|
|
396
|
+
|
|
397
|
+
const select = () => {
|
|
398
|
+
nextTick(() => {
|
|
399
|
+
getNativeInput()?.select();
|
|
400
|
+
});
|
|
401
|
+
};
|
|
402
|
+
|
|
403
|
+
const focus = () => {
|
|
404
|
+
nextTick(() => {
|
|
405
|
+
getNativeInput()?.focus();
|
|
406
|
+
});
|
|
407
|
+
};
|
|
408
|
+
|
|
409
|
+
defineExpose({
|
|
410
|
+
blur,
|
|
411
|
+
select,
|
|
412
|
+
focus
|
|
413
|
+
});
|
|
414
|
+
</script>
|