shared-ritm 1.3.14 → 1.3.15

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 (136) hide show
  1. package/README.md +103 -103
  2. package/dist/index.css +1 -1
  3. package/dist/shared-ritm.es.js +4291 -4257
  4. package/dist/shared-ritm.umd.js +154 -154
  5. package/dist/types/api/services/PhotoService.d.ts +40 -0
  6. package/dist/types/api/services/RepairsService.d.ts +1 -1
  7. package/dist/types/api/settings/ApiService.d.ts +1 -1
  8. package/package.json +65 -65
  9. package/src/App.vue +2461 -2461
  10. package/src/api/services/AuthService.ts +53 -53
  11. package/src/api/services/BrigadesService.ts +32 -32
  12. package/src/api/services/CommentsService.ts +24 -24
  13. package/src/api/services/ControlsService.ts +96 -96
  14. package/src/api/services/EquipmentService.ts +29 -29
  15. package/src/api/services/FileService.ts +17 -17
  16. package/src/api/services/GanttService.ts +17 -17
  17. package/src/api/services/InstrumentsService.ts +68 -68
  18. package/src/api/services/MetricsService.ts +110 -110
  19. package/src/api/services/ModulesService.ts +27 -27
  20. package/src/api/services/ProjectsService.ts +83 -83
  21. package/src/api/services/RepairsService.ts +124 -124
  22. package/src/api/services/ScheduleService.ts +69 -69
  23. package/src/api/services/SearchService.ts +22 -22
  24. package/src/api/services/TasksService.ts +150 -150
  25. package/src/api/services/UserService.ts +119 -119
  26. package/src/api/services/VideoService.ts +108 -108
  27. package/src/api/settings/ApiService.ts +124 -124
  28. package/src/api/types/Api_Auth.ts +105 -105
  29. package/src/api/types/Api_Brigades.ts +36 -36
  30. package/src/api/types/Api_Comment.ts +40 -40
  31. package/src/api/types/Api_Controls.ts +111 -111
  32. package/src/api/types/Api_Equipment.ts +3 -3
  33. package/src/api/types/Api_Files.ts +7 -7
  34. package/src/api/types/Api_Instruments.ts +156 -156
  35. package/src/api/types/Api_Modules.ts +21 -21
  36. package/src/api/types/Api_Projects.ts +62 -62
  37. package/src/api/types/Api_Repairs.ts +140 -140
  38. package/src/api/types/Api_Schedule.ts +64 -64
  39. package/src/api/types/Api_Search.ts +80 -80
  40. package/src/api/types/Api_Service.ts +9 -9
  41. package/src/api/types/Api_Tasks.ts +371 -371
  42. package/src/api/types/Api_User.ts +146 -146
  43. package/src/api/types/Api_Video.ts +198 -198
  44. package/src/common/app-button/AppButton.vue +173 -173
  45. package/src/common/app-checkbox/AppCheckbox.vue +26 -26
  46. package/src/common/app-date-picker/AppDatePicker.vue +81 -81
  47. package/src/common/app-datepicker/AppDatepicker.vue +218 -218
  48. package/src/common/app-dialogs/AppConfirmDialog.vue +99 -99
  49. package/src/common/app-dropdown/AppDropdown.vue +37 -37
  50. package/src/common/app-file/AppFile.vue +80 -80
  51. package/src/common/app-icon/AppIcon.vue +108 -108
  52. package/src/common/app-input/AppInput.vue +148 -148
  53. package/src/common/app-input-new/AppInputNew.vue +179 -179
  54. package/src/common/app-input-search/AppInputSearch.vue +174 -174
  55. package/src/common/app-layout/AppLayout.vue +84 -84
  56. package/src/common/app-layout/components/AppLayoutHeader.vue +273 -273
  57. package/src/common/app-layout/components/AppLayoutPage.vue +16 -16
  58. package/src/common/app-loader/index.vue +43 -43
  59. package/src/common/app-modal/index.vue +96 -96
  60. package/src/common/app-page-layout/AppPageLayout.vue +122 -122
  61. package/src/common/app-select/AppSelect.vue +157 -157
  62. package/src/common/app-sheet/AppSheet.vue +120 -120
  63. package/src/common/app-sheet-new/AppSheetNew.vue +244 -244
  64. package/src/common/app-sidebar/AppSidebar.vue +174 -168
  65. package/src/common/app-sidebar/components/SidebarMenu.vue +37 -37
  66. package/src/common/app-sidebar/components/SidebarMenuItem.vue +149 -149
  67. package/src/common/app-table/AppTable.vue +308 -308
  68. package/src/common/app-table/AppTableLayout.vue +137 -137
  69. package/src/common/app-table/components/ModalSelect.vue +294 -294
  70. package/src/common/app-table/components/TableModal.vue +356 -356
  71. package/src/common/app-table/components/TablePagination.vue +152 -152
  72. package/src/common/app-table/components/TableSearch.vue +76 -76
  73. package/src/common/app-table/controllers/useBaseTable.ts +45 -45
  74. package/src/common/app-table/controllers/useColumnSelector.ts +38 -38
  75. package/src/common/app-table/controllers/useTableModel.ts +102 -102
  76. package/src/common/app-toggle/AppToggle.vue +24 -24
  77. package/src/common/app-wrapper/AppWrapper.vue +28 -28
  78. package/src/global.d.ts +1 -1
  79. package/src/icons/components/arrow-down-icon.vue +25 -25
  80. package/src/icons/components/arrow-frame-icon.vue +19 -19
  81. package/src/icons/components/arrow-square.vue +22 -22
  82. package/src/icons/components/table-filter-icon.vue +30 -30
  83. package/src/icons/dialogs/RemoveIcon.vue +12 -12
  84. package/src/icons/dialogs/SafetyIcon.vue +12 -12
  85. package/src/icons/header/NotificationIcon.vue +18 -18
  86. package/src/icons/header/PersonIcon.vue +11 -11
  87. package/src/icons/header/SettingIcon.vue +14 -14
  88. package/src/icons/header/flashIcon.vue +24 -24
  89. package/src/icons/header/searchStatusIcon.vue +24 -24
  90. package/src/icons/header/smallCapsIcon.vue +34 -34
  91. package/src/icons/sidebar/assign-module-icon.vue +36 -36
  92. package/src/icons/sidebar/instrument-history-icon.vue +32 -32
  93. package/src/icons/sidebar/instrument-order-icon.vue +38 -38
  94. package/src/icons/sidebar/instrument-work-zone-icon.vue +18 -18
  95. package/src/icons/sidebar/instruments-icon.vue +45 -45
  96. package/src/icons/sidebar/logo-icon.vue +15 -15
  97. package/src/icons/sidebar/logout-icon.vue +13 -13
  98. package/src/icons/sidebar/modules-icon.vue +16 -16
  99. package/src/icons/sidebar/notifications-icon.vue +24 -24
  100. package/src/icons/sidebar/order-icon.vue +44 -44
  101. package/src/icons/sidebar/pass-icon.vue +38 -38
  102. package/src/icons/sidebar/positions-icon.vue +42 -42
  103. package/src/icons/sidebar/preorder-icon.vue +19 -19
  104. package/src/icons/sidebar/projects-icon.vue +31 -31
  105. package/src/icons/sidebar/repair-object-icon.vue +18 -18
  106. package/src/icons/sidebar/repairs-icon.vue +20 -20
  107. package/src/icons/sidebar/roles-icon.vue +26 -26
  108. package/src/icons/sidebar/status-history-icon.vue +24 -24
  109. package/src/icons/sidebar/tasks-icon.vue +28 -28
  110. package/src/icons/sidebar/tasks_tasks-icon.vue +39 -39
  111. package/src/icons/sidebar/tasks_today-icon.vue +27 -27
  112. package/src/icons/sidebar/teams-icon.vue +32 -32
  113. package/src/icons/sidebar/user-icon.vue +18 -18
  114. package/src/icons/sidebar/users-icon.vue +46 -46
  115. package/src/icons/sidebar/videosources-icon.vue +19 -19
  116. package/src/icons/sidebar/videowall-icon.vue +13 -13
  117. package/src/icons/sidebar/videozones-icon.vue +21 -21
  118. package/src/icons/sidebar/warehouses-icon.vue +43 -43
  119. package/src/icons/sidebar/workshop-icon.vue +100 -100
  120. package/src/icons/sidebar/workzones-icon.vue +22 -22
  121. package/src/icons/task/attention-icon.vue +13 -13
  122. package/src/icons/task/clock-icon.vue +10 -10
  123. package/src/icons/task/delete-icon.vue +10 -10
  124. package/src/icons/task/fire-icon.vue +16 -16
  125. package/src/index.ts +131 -131
  126. package/src/main.ts +28 -28
  127. package/src/quasar-user-options.ts +17 -17
  128. package/src/router/index.ts +10 -10
  129. package/src/shared/styles/general.css +124 -124
  130. package/src/shims-vue.d.ts +5 -5
  131. package/src/styles/variables.sass +12 -12
  132. package/src/utils/confirm.ts +12 -12
  133. package/src/utils/faceApiHelper.ts +132 -132
  134. package/src/utils/files.ts +19 -19
  135. package/src/utils/helpers.ts +59 -59
  136. package/src/utils/notification.ts +9 -9
@@ -1,294 +1,294 @@
1
- <template>
2
- <label v-if="label" class="field-label">
3
- {{ label }}
4
- <span v-if="rules?.length && isShowRequired" class="required">*</span>
5
- </label>
6
-
7
- <q-select
8
- ref="select"
9
- v-model="selected"
10
- :options="filteredOptions"
11
- :disable="isDisabled"
12
- :multiple="multiple"
13
- :popup-content-class="'custom-select-menu'"
14
- :hide-selected="!showChip && !simple"
15
- :placeholder="isEmpty || (multiple && !showChip) ? placeholder : null"
16
- :loading="loading"
17
- :option-value="optionValue || 'value'"
18
- :option-label="optionLabel || 'label'"
19
- emit-value
20
- filled
21
- map-options
22
- stack-label
23
- :use-input="!hideSearch"
24
- :use-chips="!simple"
25
- :hide-bottom-space="hideBottomSpace"
26
- input-debounce="100"
27
- autocomplete=""
28
- :rules="rules"
29
- @virtual-scroll="onScroll"
30
- @filter="filterFn"
31
- @add="clearInputAfterFilterSelection"
32
- >
33
- <template v-if="multiple || showChip || 'selected' in slots" #selected-item="scope">
34
- <slot v-if="'selected' in slots && scope.opt" name="selected" :opt="scope.opt" />
35
- <q-chip
36
- v-else-if="scope.opt"
37
- removable
38
- :tabindex="scope.tabindex"
39
- :style="{ backgroundColor: chipColor }"
40
- icon-remove="close"
41
- text-color="secondary"
42
- @remove="scope.removeAtIndex(scope.index)"
43
- >
44
- {{ scope.opt[optionLabel || 'label'] }}
45
- </q-chip>
46
- </template>
47
-
48
- <template #append>
49
- <q-icon
50
- v-if="!isDisabled && !isEmpty"
51
- name="close"
52
- class="cursor-pointer clear-input"
53
- @click.stop="handleClear"
54
- />
55
- </template>
56
-
57
- <template #no-option>
58
- <div class="q-pa-sm">
59
- <q-item>
60
- <q-item-section class="wrapper-empty-text">
61
- {{ emptyText || 'Ничего не найдено' }}
62
- <button
63
- v-if="allowCreate && internalSearch && internalSearch.trim()"
64
- class="add-new-items"
65
- @click="handleCreateFromSearch"
66
- >
67
- Добавить
68
- </button>
69
- </q-item-section>
70
- </q-item>
71
- </div>
72
- </template>
73
-
74
- <template #option="scope">
75
- <q-item v-if="scope.opt.__loading" class="q-py-md q-ml-md">
76
- <q-spinner-dots size="24px" color="primary" />
77
- </q-item>
78
- <q-item v-else-if="'option' in slots" v-bind="scope.itemProps">
79
- <slot name="option" :opt="scope.opt" />
80
- </q-item>
81
- <q-item v-else v-bind="scope.itemProps">
82
- <q-item-section>{{ scope.opt[optionLabel || 'label'] }}</q-item-section>
83
- </q-item>
84
- </template>
85
- </q-select>
86
- </template>
87
-
88
- <script setup lang="ts">
89
- import { computed, defineEmits, defineProps, ref, Ref, useSlots } from 'vue'
90
- import { QSelect } from 'quasar'
91
- type Option = Record<string, any>
92
-
93
- interface AppQSelectProps {
94
- modelValue: any
95
- options: Option[]
96
- placeholder?: string | undefined
97
- emptyText?: string
98
- optionLabel?: string
99
- optionValue?: string
100
- label?: string
101
- multiple?: boolean
102
- loading?: boolean
103
- isShowRequired?: boolean
104
- isDisabled?: boolean
105
- allowCreate?: boolean
106
- isSearch?: boolean
107
- hideSearch?: boolean
108
- showChip?: boolean
109
- simple?: boolean
110
- hideBottomSpace?: boolean
111
- chipColor?: string
112
- height?: string
113
- borderColor?: string
114
- borderRadius?: string
115
- menuWidth?: string
116
- rules?: ((val: any) => boolean | string)[]
117
- }
118
-
119
- const props = defineProps<AppQSelectProps>()
120
- const emit = defineEmits(['update:modelValue', 'update:scroll', 'update:search', 'clear', 'create'])
121
-
122
- const slots = useSlots()
123
-
124
- const select = ref<QSelect | null>(null)
125
- const lcText: Ref<string> = ref('')
126
- const internalSearch = ref('')
127
-
128
- const selected = computed({
129
- get() {
130
- return props.modelValue
131
- },
132
- set(value) {
133
- emit('update:modelValue', value)
134
- },
135
- })
136
-
137
- const isEmpty = computed(() => {
138
- if (!props.modelValue && props.modelValue !== 0) return true
139
-
140
- return Array.isArray(props.modelValue) && !props.modelValue.length
141
- })
142
-
143
- function handleClear() {
144
- const emptyValue = props.multiple ? [] : null
145
- selected.value = emptyValue
146
- lcText.value = ''
147
- emit('update:modelValue', emptyValue)
148
- emit('clear')
149
- }
150
-
151
- const filteredOptions = computed(() => {
152
- const labelKey = props.optionLabel || 'label'
153
-
154
- const baseOptions =
155
- 'option' in slots
156
- ? props.options
157
- : props.options.filter(x => {
158
- const label = x[labelKey]
159
- return typeof label === 'string' && label.toLowerCase().includes(lcText.value)
160
- })
161
-
162
- if (props.loading) {
163
- return [
164
- ...baseOptions,
165
- {
166
- __loading: true,
167
- label: '__loading__',
168
- value: '__loading__',
169
- },
170
- ]
171
- }
172
-
173
- return baseOptions
174
- })
175
-
176
- function handleCreateFromSearch() {
177
- const labelKey = props.optionLabel || 'label'
178
- const valueKey = props.optionValue || 'value'
179
-
180
- const trimmed = internalSearch.value?.trim()
181
- if (!trimmed) return
182
-
183
- const newOption = {
184
- [labelKey]: trimmed,
185
- [valueKey]: trimmed,
186
- }
187
- emit('create', newOption)
188
- handleClear()
189
- }
190
-
191
- function filterFn(val: string, update: (cb: () => void) => void) {
192
- debouncedFilter(val, update)
193
- }
194
-
195
- const debouncedFilter = debounce((val: string, update: (cb: () => void) => void) => {
196
- internalSearch.value = val
197
- emit('update:search', val)
198
- update(() => {
199
- lcText.value = val.toLowerCase()
200
- })
201
- }, 500)
202
-
203
- function onScroll({ to, ref: qSelectRef }) {
204
- const totalOptions = qSelectRef.options.length
205
- if (to >= totalOptions - 1 && !lcText.value) {
206
- emit('update:scroll')
207
- }
208
- }
209
-
210
- function clearInputAfterFilterSelection() {
211
- if (!props.hideSearch && select.value) {
212
- select.value.updateInputValue('')
213
- }
214
- }
215
-
216
- function debounce<T>(fn: T, ms) {
217
- let timeoutId
218
- return function (...args) {
219
- clearTimeout(timeoutId)
220
- return new Promise(resolve => {
221
- timeoutId = setTimeout(() => {
222
- resolve(fn(...args))
223
- }, ms)
224
- })
225
- }
226
- }
227
- </script>
228
-
229
- <style lang="scss" scoped>
230
- .wrapper-empty-text {
231
- display: flex;
232
- align-items: center;
233
- justify-content: space-between;
234
- flex-direction: row;
235
- .add-new-items {
236
- background: #3f8cff;
237
- color: #fff;
238
- outline: none;
239
- border: none;
240
- border-radius: 4px;
241
- padding: 5px 10px;
242
- cursor: pointer;
243
- }
244
- }
245
- .field-label {
246
- font-size: 14px;
247
- font-weight: 700;
248
- color: #7d8592;
249
- }
250
- .required {
251
- color: #f65160;
252
- font-weight: bold;
253
- }
254
-
255
- ::v-deep(.q-placeholder) {
256
- color: #7d8592;
257
- }
258
- ::v-deep(.q-field__control) {
259
- border-radius: 8px;
260
- border: 1px solid #d8e0f0;
261
- background: #fff;
262
- box-shadow: 0 1px 2px 0 rgba(184, 200, 224, 0.22);
263
- }
264
- :deep(.q-field__control:before) {
265
- display: none;
266
- }
267
- :global(.q-field--filled.q-field--highlighted .q-field__control:before) {
268
- background: #fff !important;
269
- border: none;
270
- }
271
- ::v-deep(.q-field--with-bottom) {
272
- padding-bottom: 0;
273
- }
274
- ::v-deep(.q-field__bottom) {
275
- padding: 0;
276
- }
277
- .clear-input {
278
- color: #d8e0f0;
279
- }
280
- ::v-deep(.q-chip) {
281
- border-radius: 4px;
282
- background: #e9eff9;
283
- color: #1d425d;
284
- font-family: NunitoSansFont, sans-serif;
285
- font-size: 14px;
286
- height: 30px;
287
- line-height: 30px;
288
- padding: 0 15px;
289
- .q-chip__icon {
290
- color: #3f8cff;
291
- margin-left: 5px;
292
- }
293
- }
294
- </style>
1
+ <template>
2
+ <label v-if="label" class="field-label">
3
+ {{ label }}
4
+ <span v-if="rules?.length && isShowRequired" class="required">*</span>
5
+ </label>
6
+
7
+ <q-select
8
+ ref="select"
9
+ v-model="selected"
10
+ :options="filteredOptions"
11
+ :disable="isDisabled"
12
+ :multiple="multiple"
13
+ :popup-content-class="'custom-select-menu'"
14
+ :hide-selected="!showChip && !simple"
15
+ :placeholder="isEmpty || (multiple && !showChip) ? placeholder : null"
16
+ :loading="loading"
17
+ :option-value="optionValue || 'value'"
18
+ :option-label="optionLabel || 'label'"
19
+ emit-value
20
+ filled
21
+ map-options
22
+ stack-label
23
+ :use-input="!hideSearch"
24
+ :use-chips="!simple"
25
+ :hide-bottom-space="hideBottomSpace"
26
+ input-debounce="100"
27
+ autocomplete=""
28
+ :rules="rules"
29
+ @virtual-scroll="onScroll"
30
+ @filter="filterFn"
31
+ @add="clearInputAfterFilterSelection"
32
+ >
33
+ <template v-if="multiple || showChip || 'selected' in slots" #selected-item="scope">
34
+ <slot v-if="'selected' in slots && scope.opt" name="selected" :opt="scope.opt" />
35
+ <q-chip
36
+ v-else-if="scope.opt"
37
+ removable
38
+ :tabindex="scope.tabindex"
39
+ :style="{ backgroundColor: chipColor }"
40
+ icon-remove="close"
41
+ text-color="secondary"
42
+ @remove="scope.removeAtIndex(scope.index)"
43
+ >
44
+ {{ scope.opt[optionLabel || 'label'] }}
45
+ </q-chip>
46
+ </template>
47
+
48
+ <template #append>
49
+ <q-icon
50
+ v-if="!isDisabled && !isEmpty"
51
+ name="close"
52
+ class="cursor-pointer clear-input"
53
+ @click.stop="handleClear"
54
+ />
55
+ </template>
56
+
57
+ <template #no-option>
58
+ <div class="q-pa-sm">
59
+ <q-item>
60
+ <q-item-section class="wrapper-empty-text">
61
+ {{ emptyText || 'Ничего не найдено' }}
62
+ <button
63
+ v-if="allowCreate && internalSearch && internalSearch.trim()"
64
+ class="add-new-items"
65
+ @click="handleCreateFromSearch"
66
+ >
67
+ Добавить
68
+ </button>
69
+ </q-item-section>
70
+ </q-item>
71
+ </div>
72
+ </template>
73
+
74
+ <template #option="scope">
75
+ <q-item v-if="scope.opt.__loading" class="q-py-md q-ml-md">
76
+ <q-spinner-dots size="24px" color="primary" />
77
+ </q-item>
78
+ <q-item v-else-if="'option' in slots" v-bind="scope.itemProps">
79
+ <slot name="option" :opt="scope.opt" />
80
+ </q-item>
81
+ <q-item v-else v-bind="scope.itemProps">
82
+ <q-item-section>{{ scope.opt[optionLabel || 'label'] }}</q-item-section>
83
+ </q-item>
84
+ </template>
85
+ </q-select>
86
+ </template>
87
+
88
+ <script setup lang="ts">
89
+ import { computed, defineEmits, defineProps, ref, Ref, useSlots } from 'vue'
90
+ import { QSelect } from 'quasar'
91
+ type Option = Record<string, any>
92
+
93
+ interface AppQSelectProps {
94
+ modelValue: any
95
+ options: Option[]
96
+ placeholder?: string | undefined
97
+ emptyText?: string
98
+ optionLabel?: string
99
+ optionValue?: string
100
+ label?: string
101
+ multiple?: boolean
102
+ loading?: boolean
103
+ isShowRequired?: boolean
104
+ isDisabled?: boolean
105
+ allowCreate?: boolean
106
+ isSearch?: boolean
107
+ hideSearch?: boolean
108
+ showChip?: boolean
109
+ simple?: boolean
110
+ hideBottomSpace?: boolean
111
+ chipColor?: string
112
+ height?: string
113
+ borderColor?: string
114
+ borderRadius?: string
115
+ menuWidth?: string
116
+ rules?: ((val: any) => boolean | string)[]
117
+ }
118
+
119
+ const props = defineProps<AppQSelectProps>()
120
+ const emit = defineEmits(['update:modelValue', 'update:scroll', 'update:search', 'clear', 'create'])
121
+
122
+ const slots = useSlots()
123
+
124
+ const select = ref<QSelect | null>(null)
125
+ const lcText: Ref<string> = ref('')
126
+ const internalSearch = ref('')
127
+
128
+ const selected = computed({
129
+ get() {
130
+ return props.modelValue
131
+ },
132
+ set(value) {
133
+ emit('update:modelValue', value)
134
+ },
135
+ })
136
+
137
+ const isEmpty = computed(() => {
138
+ if (!props.modelValue && props.modelValue !== 0) return true
139
+
140
+ return Array.isArray(props.modelValue) && !props.modelValue.length
141
+ })
142
+
143
+ function handleClear() {
144
+ const emptyValue = props.multiple ? [] : null
145
+ selected.value = emptyValue
146
+ lcText.value = ''
147
+ emit('update:modelValue', emptyValue)
148
+ emit('clear')
149
+ }
150
+
151
+ const filteredOptions = computed(() => {
152
+ const labelKey = props.optionLabel || 'label'
153
+
154
+ const baseOptions =
155
+ 'option' in slots
156
+ ? props.options
157
+ : props.options.filter(x => {
158
+ const label = x[labelKey]
159
+ return typeof label === 'string' && label.toLowerCase().includes(lcText.value)
160
+ })
161
+
162
+ if (props.loading) {
163
+ return [
164
+ ...baseOptions,
165
+ {
166
+ __loading: true,
167
+ label: '__loading__',
168
+ value: '__loading__',
169
+ },
170
+ ]
171
+ }
172
+
173
+ return baseOptions
174
+ })
175
+
176
+ function handleCreateFromSearch() {
177
+ const labelKey = props.optionLabel || 'label'
178
+ const valueKey = props.optionValue || 'value'
179
+
180
+ const trimmed = internalSearch.value?.trim()
181
+ if (!trimmed) return
182
+
183
+ const newOption = {
184
+ [labelKey]: trimmed,
185
+ [valueKey]: trimmed,
186
+ }
187
+ emit('create', newOption)
188
+ handleClear()
189
+ }
190
+
191
+ function filterFn(val: string, update: (cb: () => void) => void) {
192
+ debouncedFilter(val, update)
193
+ }
194
+
195
+ const debouncedFilter = debounce((val: string, update: (cb: () => void) => void) => {
196
+ internalSearch.value = val
197
+ emit('update:search', val)
198
+ update(() => {
199
+ lcText.value = val.toLowerCase()
200
+ })
201
+ }, 500)
202
+
203
+ function onScroll({ to, ref: qSelectRef }) {
204
+ const totalOptions = qSelectRef.options.length
205
+ if (to >= totalOptions - 1 && !lcText.value) {
206
+ emit('update:scroll')
207
+ }
208
+ }
209
+
210
+ function clearInputAfterFilterSelection() {
211
+ if (!props.hideSearch && select.value) {
212
+ select.value.updateInputValue('')
213
+ }
214
+ }
215
+
216
+ function debounce<T>(fn: T, ms) {
217
+ let timeoutId
218
+ return function (...args) {
219
+ clearTimeout(timeoutId)
220
+ return new Promise(resolve => {
221
+ timeoutId = setTimeout(() => {
222
+ resolve(fn(...args))
223
+ }, ms)
224
+ })
225
+ }
226
+ }
227
+ </script>
228
+
229
+ <style lang="scss" scoped>
230
+ .wrapper-empty-text {
231
+ display: flex;
232
+ align-items: center;
233
+ justify-content: space-between;
234
+ flex-direction: row;
235
+ .add-new-items {
236
+ background: #3f8cff;
237
+ color: #fff;
238
+ outline: none;
239
+ border: none;
240
+ border-radius: 4px;
241
+ padding: 5px 10px;
242
+ cursor: pointer;
243
+ }
244
+ }
245
+ .field-label {
246
+ font-size: 14px;
247
+ font-weight: 700;
248
+ color: #7d8592;
249
+ }
250
+ .required {
251
+ color: #f65160;
252
+ font-weight: bold;
253
+ }
254
+
255
+ ::v-deep(.q-placeholder) {
256
+ color: #7d8592;
257
+ }
258
+ ::v-deep(.q-field__control) {
259
+ border-radius: 8px;
260
+ border: 1px solid #d8e0f0;
261
+ background: #fff;
262
+ box-shadow: 0 1px 2px 0 rgba(184, 200, 224, 0.22);
263
+ }
264
+ :deep(.q-field__control:before) {
265
+ display: none;
266
+ }
267
+ :global(.q-field--filled.q-field--highlighted .q-field__control:before) {
268
+ background: #fff !important;
269
+ border: none;
270
+ }
271
+ ::v-deep(.q-field--with-bottom) {
272
+ padding-bottom: 0;
273
+ }
274
+ ::v-deep(.q-field__bottom) {
275
+ padding: 0;
276
+ }
277
+ .clear-input {
278
+ color: #d8e0f0;
279
+ }
280
+ ::v-deep(.q-chip) {
281
+ border-radius: 4px;
282
+ background: #e9eff9;
283
+ color: #1d425d;
284
+ font-family: NunitoSansFont, sans-serif;
285
+ font-size: 14px;
286
+ height: 30px;
287
+ line-height: 30px;
288
+ padding: 0 15px;
289
+ .q-chip__icon {
290
+ color: #3f8cff;
291
+ margin-left: 5px;
292
+ }
293
+ }
294
+ </style>