shared-ritm 1.3.124 → 1.3.126

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 (80) hide show
  1. package/dist/index.css +1 -1
  2. package/dist/shared-ritm.es.js +502 -481
  3. package/dist/shared-ritm.umd.js +2 -2
  4. package/dist/types/api/services/PhotoService.d.ts +40 -0
  5. package/dist/types/stories/Button.stories.d.ts +13 -0
  6. package/dist/types/stories/Checkbox.stories.d.ts +7 -0
  7. package/dist/types/stories/Confirm.stories.d.ts +8 -0
  8. package/dist/types/stories/DatePicker.stories.d.ts +8 -0
  9. package/dist/types/stories/Dropdown.stories.d.ts +8 -0
  10. package/dist/types/stories/File.stories.d.ts +8 -0
  11. package/dist/types/stories/Icon.stories.d.ts +7 -0
  12. package/dist/types/stories/Input.stories.d.ts +11 -0
  13. package/dist/types/stories/InputNew.stories.d.ts +12 -0
  14. package/dist/types/stories/InputSearch.stories.d.ts +10 -0
  15. package/dist/types/stories/Loader.stories.d.ts +8 -0
  16. package/dist/types/stories/Select.stories.d.ts +7 -0
  17. package/dist/types/stories/Toggle.stories.d.ts +8 -0
  18. package/package.json +70 -70
  19. package/src/App.vue +2461 -2461
  20. package/src/api/services/AuthService.ts +67 -67
  21. package/src/api/services/ControlsService.ts +100 -100
  22. package/src/api/services/EquipmentService.ts +68 -68
  23. package/src/api/services/GanttService.ts +58 -58
  24. package/src/api/services/InstrumentsService.ts +76 -76
  25. package/src/api/services/RepairsService.ts +111 -111
  26. package/src/api/services/TasksService.ts +165 -165
  27. package/src/api/services/UserIssueService.ts +32 -32
  28. package/src/api/services/UserService.ts +129 -129
  29. package/src/api/services/VideoService.ts +118 -118
  30. package/src/api/settings/ApiService.ts +185 -185
  31. package/src/api/types/Api_Auth.ts +121 -121
  32. package/src/api/types/Api_Controls.ts +112 -112
  33. package/src/api/types/Api_Equipment.ts +54 -54
  34. package/src/api/types/Api_Instruments.ts +182 -182
  35. package/src/api/types/Api_Metrics.ts +89 -89
  36. package/src/api/types/Api_Repairs.ts +200 -200
  37. package/src/api/types/Api_Search.ts +81 -81
  38. package/src/api/types/Api_Tasks.ts +378 -378
  39. package/src/api/types/Api_User.ts +161 -161
  40. package/src/api/types/Api_User_Issue.ts +36 -36
  41. package/src/api/types/Api_Video.ts +244 -244
  42. package/src/common/app-button/Button.stories.ts +369 -369
  43. package/src/common/app-checkbox/AppCheckbox.vue +33 -31
  44. package/src/common/app-checkbox/Checkbox.stories.ts +252 -252
  45. package/src/common/app-date-picker/DatePicker.stories.ts +66 -66
  46. package/src/common/app-datepicker/Datepicker.stories.ts +145 -145
  47. package/src/common/app-dialogs/AppConfirmDialog.vue +109 -109
  48. package/src/common/app-dialogs/Confirm.stories.ts +93 -93
  49. package/src/common/app-dropdown/Dropdown.stories.ts +94 -94
  50. package/src/common/app-file/File.stories.ts +104 -104
  51. package/src/common/app-icon/AppIcon.vue +110 -110
  52. package/src/common/app-icon/Icon.stories.ts +91 -91
  53. package/src/common/app-input/AppInput.vue +150 -150
  54. package/src/common/app-input/Input.stories.ts +160 -160
  55. package/src/common/app-input-new/AppInputNew.vue +181 -181
  56. package/src/common/app-input-new/InputNew.stories.ts +240 -240
  57. package/src/common/app-input-search/InputSearch.stories.ts +149 -149
  58. package/src/common/app-layout/components/AppLayoutHeader.vue +289 -289
  59. package/src/common/app-loader/Loader.stories.ts +114 -114
  60. package/src/common/app-modal/index.vue +101 -101
  61. package/src/common/app-select/AppSelect.vue +159 -159
  62. package/src/common/app-select/Select.stories.ts +155 -155
  63. package/src/common/app-sheet-new/AppSheetNew.vue +254 -254
  64. package/src/common/app-sidebar/AppSidebar.vue +177 -177
  65. package/src/common/app-table/AppTable.vue +313 -313
  66. package/src/common/app-table/components/ModalSelect.stories.ts +323 -323
  67. package/src/common/app-table/components/ModalSelect.vue +311 -302
  68. package/src/common/app-table/components/TableModal.vue +369 -369
  69. package/src/common/app-table/controllers/useColumnSelector.ts +45 -45
  70. package/src/common/app-table/controllers/useTableModel.ts +98 -98
  71. package/src/common/app-toggle/AppToggle.vue +12 -12
  72. package/src/common/app-toggle/Toggle.stories.ts +245 -245
  73. package/src/common/app-wrapper/AppWrapper.vue +31 -31
  74. package/src/configs/storybook.ts +14 -14
  75. package/src/icons/sidebar/user-requests-icon.vue +23 -23
  76. package/src/index.ts +134 -134
  77. package/src/shared/styles/general.css +140 -140
  78. package/src/styles/variables.sass +12 -12
  79. package/src/utils/files.ts +38 -38
  80. package/src/utils/helpers.ts +59 -59
@@ -1,369 +1,369 @@
1
- <template>
2
- <q-dialog :model-value="modelValue" @update:model-value="toggleModal">
3
- <q-card style="min-width: 700px">
4
- <q-card-section>
5
- <div class="modal-title">{{ title }}</div>
6
- </q-card-section>
7
-
8
- <q-card-section>
9
- <q-form ref="formRef" @submit.prevent="submit">
10
- <div v-for="field in fields" :key="field.key" class="field-wrapper">
11
- <label v-if="field.type === 'text'" :data-test="`${field.key}-label`" class="field-label">
12
- {{ field.label }}
13
- <span v-if="field.rules?.length && mode !== 'view'" class="required">*</span>
14
- </label>
15
-
16
- <q-input
17
- v-if="field.type === 'text'"
18
- v-model="formData[field.key]"
19
- :data-test="`${field.key}-input`"
20
- :rules="field.rules"
21
- :readonly="mode === 'view' || (field.key === 'uuid' && mode === 'edit')"
22
- filled
23
- :placeholder="field.placeholder"
24
- >
25
- <template #append>
26
- <q-icon
27
- v-if="mode !== 'view' && formData[field.key] && !(field.key === 'uuid' && mode === 'edit')"
28
- name="close"
29
- class="cursor-pointer clear-input"
30
- @click="() => handleClear(field.key)"
31
- />
32
- <q-btn
33
- v-if="field.key === 'uuid' && mode === 'create'"
34
- flat
35
- no-caps
36
- label="UUID"
37
- size="sm"
38
- class="q-ml-sm uuid-btn"
39
- @click="generateUuid"
40
- />
41
- <q-icon
42
- v-else-if="field.key === 'uuid' && mode !== 'create'"
43
- name="content_copy"
44
- class="cursor-pointer q-ml-sm copy-icon"
45
- color="primary"
46
- @click="copyToClipboard(formData[field.key])"
47
- />
48
- </template>
49
- </q-input>
50
-
51
- <app-modal-select
52
- v-else-if="field.type === 'select'"
53
- v-model="formData[field.key]"
54
- :data-test="`${field.key}-select`"
55
- :options="filteredOptions[field.key] || field.options"
56
- :rules="field.rules"
57
- :placeholder="mode === 'view' ? '' : field.placeholder"
58
- :multiple="!field.simple"
59
- :show-chip="!field.simple"
60
- :is-disabled="mode === 'view'"
61
- :loading="field.loading"
62
- :label="field.label"
63
- :is-show-required="mode !== 'view'"
64
- empty-text="Нет данных"
65
- option-label="label"
66
- option-value="value"
67
- chip-color="#e9eff9"
68
- :simple="field.simple"
69
- @update:search="val => field.onSearch?.(val)"
70
- @update:scroll="() => field.onScroll?.()"
71
- @clear="
72
- () => {
73
- handleClear(field.key)
74
- nextTick(() => formRef.value?.validate())
75
- }
76
- "
77
- />
78
- </div>
79
- </q-form>
80
- </q-card-section>
81
-
82
- <q-card-actions align="center">
83
- <q-btn v-if="mode === 'view'" class="remove" flat label="Удалить" :loading="loading" @click="emit('delete')" />
84
- <q-btn
85
- v-if="mode !== 'view'"
86
- class="confirm"
87
- data-test="save-create-button"
88
- flat
89
- :label="mode === 'edit' ? 'Сохранить' : 'Создать'"
90
- :disable="isSubmitDisabled"
91
- :loading="loading"
92
- @click="submit"
93
- />
94
- <q-btn
95
- v-else
96
- class="confirm"
97
- data-test="edit-button"
98
- flat
99
- label="Редактировать"
100
- :disable="loading"
101
- @click="$emit('edit')"
102
- />
103
- <q-btn class="cancel" data-test="close-button" flat label="Закрыть" :disable="loading" @click="close" />
104
- </q-card-actions>
105
- </q-card>
106
- </q-dialog>
107
- </template>
108
-
109
- <script setup lang="ts">
110
- import { ref, watch, defineProps, defineEmits, nextTick, computed } from 'vue'
111
- import { useQuasar } from 'quasar'
112
- import AppModalSelect from '../components/ModalSelect.vue'
113
- import { notificationSettings } from '@/utils/notification'
114
- import { isEqual, normalizeValue, uuidv4 } from '@/utils/helpers'
115
-
116
- type ModalMode = 'view' | 'edit' | 'create'
117
- type FieldType = 'text' | 'select'
118
-
119
- interface FieldOption {
120
- label: string
121
- value: string
122
- }
123
- interface FieldSchema {
124
- key: string
125
- label: string
126
- type: FieldType
127
- rules?: ((val: any) => boolean | string)[]
128
- options?: FieldOption[]
129
- placeholder?: string
130
- onSearch?: (val: string) => void
131
- onScroll?: () => void
132
- loading?: boolean
133
- simple?: boolean
134
- }
135
-
136
- const props = defineProps<{
137
- modelValue: boolean
138
- title: string
139
- mode: ModalMode
140
- fields: FieldSchema[]
141
- initialData?: Record<string, any>
142
- loading?: boolean
143
- }>()
144
- const emit = defineEmits<{
145
- (e: 'update:modelValue', val: boolean, hasChanges?: boolean): void
146
- (e: 'submit', data: Record<string, any>): void
147
- (e: 'edit'): void
148
- (e: 'delete'): void
149
- }>()
150
-
151
- const $q = useQuasar()
152
- const formRef = ref()
153
- const formData = ref<Record<string, any>>({})
154
- const filteredOptions = ref<Record<string, FieldOption[]>>({})
155
- const isSubmitDisabled = computed(() => {
156
- if (props.mode === 'view') return false
157
-
158
- const hasEmptyRequired = props.fields.some(field => {
159
- const isRequired = field.rules?.some(rule => rule('') !== true)
160
- const val = formData.value[field.key]
161
- const normalized = normalizeValue(val)
162
-
163
- const empty =
164
- normalized === null ||
165
- normalized === undefined ||
166
- (typeof normalized === 'string' && normalized.trim() === '') ||
167
- (Array.isArray(normalized) && normalized.length === 0)
168
-
169
- return isRequired && empty
170
- })
171
-
172
- if (hasEmptyRequired) return true
173
-
174
- const hasChanges = props.fields.some(field => {
175
- const key = field.key
176
- const current = normalizeValue(formData.value[key])
177
- const initial = normalizeValue(props.initialData?.[key])
178
- return !isEqual(current, initial)
179
- })
180
-
181
- return !hasChanges
182
- })
183
- function toggleModal(val: boolean) {
184
- if (val) {
185
- emit('update:modelValue', val)
186
- } else {
187
- close()
188
- }
189
- }
190
- function checkChanges() {
191
- if (props.mode === 'view') {
192
- return false
193
- }
194
-
195
- if (!props.initialData || !Object.keys(props.initialData).length) {
196
- return props.fields?.some(
197
- field =>
198
- formData.value?.[field.key] &&
199
- !(Array.isArray(formData.value?.[field.key]) && !formData.value?.[field.key].length),
200
- )
201
- }
202
-
203
- return props.fields?.some(field => !isEqual(formData.value?.[field.key], props.initialData?.[field.key]))
204
- }
205
- function close() {
206
- emit('update:modelValue', false, checkChanges())
207
- }
208
- function submit() {
209
- formRef.value?.validate().then(ok => {
210
- if (!ok) return
211
-
212
- const changed: Record<string, any> = {}
213
-
214
- for (const field of props.fields) {
215
- const key = field.key
216
- const current = formData.value[key]
217
- const initial = props.initialData?.[key]
218
-
219
- const normalizedCurrent = field.type === 'select' ? normalizeValue(current) : current
220
- const normalizedInitial = field.type === 'select' ? normalizeValue(initial) : initial
221
-
222
- if (!isEqual(normalizedCurrent, normalizedInitial)) {
223
- changed[key] = normalizedCurrent
224
- }
225
- }
226
-
227
- emit('submit', changed)
228
- })
229
- }
230
- function handleClear(key: string) {
231
- const field = props.fields.find(f => f.key === key)
232
- formData.value[key] = field?.type === 'select' ? (field.simple ? null : []) : ''
233
-
234
- nextTick(() => {
235
- formRef.value?.validate()
236
- })
237
- }
238
- function generateUuid() {
239
- formData.value.uuid = uuidv4()
240
- }
241
- function copyToClipboard(text: string) {
242
- navigator.clipboard.writeText(text).then(() => {
243
- $q.notify(notificationSettings('success', 'UUID скопирован'))
244
- })
245
- }
246
-
247
- watch(
248
- () => props.modelValue,
249
- val => {
250
- if (val) {
251
- formData.value = { ...(props.initialData ?? {}) }
252
- }
253
- },
254
- { immediate: true },
255
- )
256
- </script>
257
-
258
- <style lang="scss">
259
- .custom-select-menu {
260
- max-height: 250px !important;
261
- overflow-y: auto !important;
262
- }
263
- </style>
264
- <style scoped lang="scss">
265
- .uuid-btn {
266
- height: 32px;
267
- padding: 0 10px;
268
- border: 1px solid #3f8cff;
269
- color: #3f8cff;
270
- font-weight: 700;
271
- font-size: 14px;
272
- background: white;
273
- border-radius: 6px;
274
- }
275
-
276
- .q-card {
277
- border-radius: 12px;
278
- background: #fff;
279
- box-shadow: 0 4px 4px 0 rgba(0, 0, 0, 0.25);
280
- font-family: NunitoSansFont, sans-serif;
281
-
282
- .modal-title {
283
- color: #1d425d;
284
- text-align: center;
285
- font-size: 32px;
286
- font-weight: 700;
287
- padding: 18px 29px;
288
- }
289
-
290
- .q-card__section {
291
- padding: 0;
292
- }
293
-
294
- .q-form {
295
- padding: 0 29px;
296
- }
297
-
298
- .field-wrapper {
299
- display: flex;
300
- flex-direction: column;
301
- margin-bottom: 15px;
302
- }
303
-
304
- .field-label {
305
- font-size: 14px;
306
- font-weight: 700;
307
- color: #7d8592;
308
- }
309
- ::v-deep(.q-placeholder) {
310
- color: #7d8592;
311
- }
312
- ::v-deep(.q-field__control) {
313
- border-radius: 8px;
314
- border: 1px solid #d8e0f0;
315
- background: #fff;
316
- box-shadow: 0px 1px 2px 0px rgba(184, 200, 224, 0.22);
317
- }
318
-
319
- ::v-deep(.q-field--filled .q-field__control:before) {
320
- background: #fff !important;
321
- border: none;
322
- }
323
-
324
- ::v-deep(.q-field--with-bottom) {
325
- padding-bottom: 0;
326
- }
327
-
328
- ::v-deep(.q-field__bottom) {
329
- padding: 0;
330
- }
331
-
332
- .clear-input {
333
- color: #d8e0f0;
334
- }
335
-
336
- .required {
337
- color: #f65160;
338
- font-weight: bold;
339
- }
340
-
341
- &__actions {
342
- padding: 18px 0;
343
- border-radius: 0px 0px 16px 16px;
344
- background: #f4f9fd;
345
-
346
- button {
347
- padding: 13px 37px;
348
- border-radius: 4px;
349
- font-size: 16px;
350
- font-weight: 700;
351
- text-transform: none;
352
-
353
- &.remove {
354
- border: 1px solid #f65160;
355
- color: #f65160;
356
- }
357
- &.confirm {
358
- background: #3f8cff;
359
- color: #fff;
360
- }
361
-
362
- &.cancel {
363
- color: #3f8cff;
364
- border: 1px solid #3f8cff;
365
- }
366
- }
367
- }
368
- }
369
- </style>
1
+ <template>
2
+ <q-dialog :model-value="modelValue" @update:model-value="toggleModal">
3
+ <q-card style="min-width: 700px">
4
+ <q-card-section>
5
+ <div class="modal-title">{{ title }}</div>
6
+ </q-card-section>
7
+
8
+ <q-card-section>
9
+ <q-form ref="formRef" @submit.prevent="submit">
10
+ <div v-for="field in fields" :key="field.key" class="field-wrapper">
11
+ <label v-if="field.type === 'text'" :data-test="`${field.key}-label`" class="field-label">
12
+ {{ field.label }}
13
+ <span v-if="field.rules?.length && mode !== 'view'" class="required">*</span>
14
+ </label>
15
+
16
+ <q-input
17
+ v-if="field.type === 'text'"
18
+ v-model="formData[field.key]"
19
+ :data-test="`${field.key}-input`"
20
+ :rules="field.rules"
21
+ :readonly="mode === 'view' || (field.key === 'uuid' && mode === 'edit')"
22
+ filled
23
+ :placeholder="field.placeholder"
24
+ >
25
+ <template #append>
26
+ <q-icon
27
+ v-if="mode !== 'view' && formData[field.key] && !(field.key === 'uuid' && mode === 'edit')"
28
+ name="close"
29
+ class="cursor-pointer clear-input"
30
+ @click="() => handleClear(field.key)"
31
+ />
32
+ <q-btn
33
+ v-if="field.key === 'uuid' && mode === 'create'"
34
+ flat
35
+ no-caps
36
+ label="UUID"
37
+ size="sm"
38
+ class="q-ml-sm uuid-btn"
39
+ @click="generateUuid"
40
+ />
41
+ <q-icon
42
+ v-else-if="field.key === 'uuid' && mode !== 'create'"
43
+ name="content_copy"
44
+ class="cursor-pointer q-ml-sm copy-icon"
45
+ color="primary"
46
+ @click="copyToClipboard(formData[field.key])"
47
+ />
48
+ </template>
49
+ </q-input>
50
+
51
+ <app-modal-select
52
+ v-else-if="field.type === 'select'"
53
+ v-model="formData[field.key]"
54
+ :data-test="`${field.key}-select`"
55
+ :options="filteredOptions[field.key] || field.options"
56
+ :rules="field.rules"
57
+ :placeholder="mode === 'view' ? '' : field.placeholder"
58
+ :multiple="!field.simple"
59
+ :show-chip="!field.simple"
60
+ :is-disabled="mode === 'view'"
61
+ :loading="field.loading"
62
+ :label="field.label"
63
+ :is-show-required="mode !== 'view'"
64
+ empty-text="Нет данных"
65
+ option-label="label"
66
+ option-value="value"
67
+ chip-color="#e9eff9"
68
+ :simple="field.simple"
69
+ @update:search="val => field.onSearch?.(val)"
70
+ @update:scroll="() => field.onScroll?.()"
71
+ @clear="
72
+ () => {
73
+ handleClear(field.key)
74
+ nextTick(() => formRef.value?.validate())
75
+ }
76
+ "
77
+ />
78
+ </div>
79
+ </q-form>
80
+ </q-card-section>
81
+
82
+ <q-card-actions align="center">
83
+ <q-btn v-if="mode === 'view'" class="remove" flat label="Удалить" :loading="loading" @click="emit('delete')" />
84
+ <q-btn
85
+ v-if="mode !== 'view'"
86
+ class="confirm"
87
+ data-test="save-create-button"
88
+ flat
89
+ :label="mode === 'edit' ? 'Сохранить' : 'Создать'"
90
+ :disable="isSubmitDisabled"
91
+ :loading="loading"
92
+ @click="submit"
93
+ />
94
+ <q-btn
95
+ v-else
96
+ class="confirm"
97
+ data-test="edit-button"
98
+ flat
99
+ label="Редактировать"
100
+ :disable="loading"
101
+ @click="$emit('edit')"
102
+ />
103
+ <q-btn class="cancel" data-test="close-button" flat label="Закрыть" :disable="loading" @click="close" />
104
+ </q-card-actions>
105
+ </q-card>
106
+ </q-dialog>
107
+ </template>
108
+
109
+ <script setup lang="ts">
110
+ import { ref, watch, defineProps, defineEmits, nextTick, computed } from 'vue'
111
+ import { useQuasar } from 'quasar'
112
+ import AppModalSelect from '../components/ModalSelect.vue'
113
+ import { notificationSettings } from '@/utils/notification'
114
+ import { isEqual, normalizeValue, uuidv4 } from '@/utils/helpers'
115
+
116
+ type ModalMode = 'view' | 'edit' | 'create'
117
+ type FieldType = 'text' | 'select'
118
+
119
+ interface FieldOption {
120
+ label: string
121
+ value: string
122
+ }
123
+ interface FieldSchema {
124
+ key: string
125
+ label: string
126
+ type: FieldType
127
+ rules?: ((val: any) => boolean | string)[]
128
+ options?: FieldOption[]
129
+ placeholder?: string
130
+ onSearch?: (val: string) => void
131
+ onScroll?: () => void
132
+ loading?: boolean
133
+ simple?: boolean
134
+ }
135
+
136
+ const props = defineProps<{
137
+ modelValue: boolean
138
+ title: string
139
+ mode: ModalMode
140
+ fields: FieldSchema[]
141
+ initialData?: Record<string, any>
142
+ loading?: boolean
143
+ }>()
144
+ const emit = defineEmits<{
145
+ (e: 'update:modelValue', val: boolean, hasChanges?: boolean): void
146
+ (e: 'submit', data: Record<string, any>): void
147
+ (e: 'edit'): void
148
+ (e: 'delete'): void
149
+ }>()
150
+
151
+ const $q = useQuasar()
152
+ const formRef = ref()
153
+ const formData = ref<Record<string, any>>({})
154
+ const filteredOptions = ref<Record<string, FieldOption[]>>({})
155
+ const isSubmitDisabled = computed(() => {
156
+ if (props.mode === 'view') return false
157
+
158
+ const hasEmptyRequired = props.fields.some(field => {
159
+ const isRequired = field.rules?.some(rule => rule('') !== true)
160
+ const val = formData.value[field.key]
161
+ const normalized = normalizeValue(val)
162
+
163
+ const empty =
164
+ normalized === null ||
165
+ normalized === undefined ||
166
+ (typeof normalized === 'string' && normalized.trim() === '') ||
167
+ (Array.isArray(normalized) && normalized.length === 0)
168
+
169
+ return isRequired && empty
170
+ })
171
+
172
+ if (hasEmptyRequired) return true
173
+
174
+ const hasChanges = props.fields.some(field => {
175
+ const key = field.key
176
+ const current = normalizeValue(formData.value[key])
177
+ const initial = normalizeValue(props.initialData?.[key])
178
+ return !isEqual(current, initial)
179
+ })
180
+
181
+ return !hasChanges
182
+ })
183
+ function toggleModal(val: boolean) {
184
+ if (val) {
185
+ emit('update:modelValue', val)
186
+ } else {
187
+ close()
188
+ }
189
+ }
190
+ function checkChanges() {
191
+ if (props.mode === 'view') {
192
+ return false
193
+ }
194
+
195
+ if (!props.initialData || !Object.keys(props.initialData).length) {
196
+ return props.fields?.some(
197
+ field =>
198
+ formData.value?.[field.key] &&
199
+ !(Array.isArray(formData.value?.[field.key]) && !formData.value?.[field.key].length),
200
+ )
201
+ }
202
+
203
+ return props.fields?.some(field => !isEqual(formData.value?.[field.key], props.initialData?.[field.key]))
204
+ }
205
+ function close() {
206
+ emit('update:modelValue', false, checkChanges())
207
+ }
208
+ function submit() {
209
+ formRef.value?.validate().then(ok => {
210
+ if (!ok) return
211
+
212
+ const changed: Record<string, any> = {}
213
+
214
+ for (const field of props.fields) {
215
+ const key = field.key
216
+ const current = formData.value[key]
217
+ const initial = props.initialData?.[key]
218
+
219
+ const normalizedCurrent = field.type === 'select' ? normalizeValue(current) : current
220
+ const normalizedInitial = field.type === 'select' ? normalizeValue(initial) : initial
221
+
222
+ if (!isEqual(normalizedCurrent, normalizedInitial)) {
223
+ changed[key] = normalizedCurrent
224
+ }
225
+ }
226
+
227
+ emit('submit', changed)
228
+ })
229
+ }
230
+ function handleClear(key: string) {
231
+ const field = props.fields.find(f => f.key === key)
232
+ formData.value[key] = field?.type === 'select' ? (field.simple ? null : []) : ''
233
+
234
+ nextTick(() => {
235
+ formRef.value?.validate()
236
+ })
237
+ }
238
+ function generateUuid() {
239
+ formData.value.uuid = uuidv4()
240
+ }
241
+ function copyToClipboard(text: string) {
242
+ navigator.clipboard.writeText(text).then(() => {
243
+ $q.notify(notificationSettings('success', 'UUID скопирован'))
244
+ })
245
+ }
246
+
247
+ watch(
248
+ () => props.modelValue,
249
+ val => {
250
+ if (val) {
251
+ formData.value = { ...(props.initialData ?? {}) }
252
+ }
253
+ },
254
+ { immediate: true },
255
+ )
256
+ </script>
257
+
258
+ <style lang="scss">
259
+ .custom-select-menu {
260
+ max-height: 250px !important;
261
+ overflow-y: auto !important;
262
+ }
263
+ </style>
264
+ <style scoped lang="scss">
265
+ .uuid-btn {
266
+ height: 32px;
267
+ padding: 0 10px;
268
+ border: 1px solid #3f8cff;
269
+ color: #3f8cff;
270
+ font-weight: 700;
271
+ font-size: 14px;
272
+ background: white;
273
+ border-radius: 6px;
274
+ }
275
+
276
+ .q-card {
277
+ border-radius: 12px;
278
+ background: #fff;
279
+ box-shadow: 0 4px 4px 0 rgba(0, 0, 0, 0.25);
280
+ font-family: NunitoSansFont, sans-serif;
281
+
282
+ .modal-title {
283
+ color: #1d425d;
284
+ text-align: center;
285
+ font-size: 32px;
286
+ font-weight: 700;
287
+ padding: 18px 29px;
288
+ }
289
+
290
+ .q-card__section {
291
+ padding: 0;
292
+ }
293
+
294
+ .q-form {
295
+ padding: 0 29px;
296
+ }
297
+
298
+ .field-wrapper {
299
+ display: flex;
300
+ flex-direction: column;
301
+ margin-bottom: 15px;
302
+ }
303
+
304
+ .field-label {
305
+ font-size: 14px;
306
+ font-weight: 700;
307
+ color: #7d8592;
308
+ }
309
+ ::v-deep(.q-placeholder) {
310
+ color: #7d8592;
311
+ }
312
+ ::v-deep(.q-field__control) {
313
+ border-radius: 8px;
314
+ border: 1px solid #d8e0f0;
315
+ background: #fff;
316
+ box-shadow: 0px 1px 2px 0px rgba(184, 200, 224, 0.22);
317
+ }
318
+
319
+ ::v-deep(.q-field--filled .q-field__control:before) {
320
+ background: #fff !important;
321
+ border: none;
322
+ }
323
+
324
+ ::v-deep(.q-field--with-bottom) {
325
+ padding-bottom: 0;
326
+ }
327
+
328
+ ::v-deep(.q-field__bottom) {
329
+ padding: 0;
330
+ }
331
+
332
+ .clear-input {
333
+ color: #d8e0f0;
334
+ }
335
+
336
+ .required {
337
+ color: #f65160;
338
+ font-weight: bold;
339
+ }
340
+
341
+ &__actions {
342
+ padding: 18px 0;
343
+ border-radius: 0px 0px 16px 16px;
344
+ background: #f4f9fd;
345
+
346
+ button {
347
+ padding: 13px 37px;
348
+ border-radius: 4px;
349
+ font-size: 16px;
350
+ font-weight: 700;
351
+ text-transform: none;
352
+
353
+ &.remove {
354
+ border: 1px solid #f65160;
355
+ color: #f65160;
356
+ }
357
+ &.confirm {
358
+ background: #3f8cff;
359
+ color: #fff;
360
+ }
361
+
362
+ &.cancel {
363
+ color: #3f8cff;
364
+ border: 1px solid #3f8cff;
365
+ }
366
+ }
367
+ }
368
+ }
369
+ </style>