plugin-ui-for-kzt 0.0.19 → 0.0.20

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 (41) hide show
  1. package/dist/components/BaseButton/BaseButton.vue.d.ts +0 -2
  2. package/dist/components/BaseCalendar/BaseCalendar.vue.d.ts +9 -1
  3. package/dist/components/BaseCheckbox/BaseCheckbox.vue.d.ts +0 -2
  4. package/dist/components/BaseDropdown/BaseDropdown.vue.d.ts +0 -2
  5. package/dist/components/BaseField/BaseField.vue.d.ts +98 -0
  6. package/dist/components/BaseInput/BaseInput.vue.d.ts +16 -14
  7. package/dist/components/BaseInputCalendar/BaseInputCalendar.vue.d.ts +13 -15
  8. package/dist/components/BaseInputCurrency/BaseInputCurrency.vue.d.ts +7 -9
  9. package/dist/components/BaseInputEmail/BaseInputEmail.vue.d.ts +7 -12
  10. package/dist/components/BaseInputPhone/BaseInputPhone.vue.d.ts +7 -12
  11. package/dist/components/BaseOpenedListItem/BaseOpenedListItem.vue.d.ts +0 -2
  12. package/dist/components/BaseRadio/BaseRadio.vue.d.ts +0 -2
  13. package/dist/components/BaseSegmentedButtons/BaseSegmentedButtons.vue.d.ts +0 -2
  14. package/dist/components/BaseSiteInput/BaseSiteInput.vue.d.ts +11 -7
  15. package/dist/components/BaseTextarea/BaseTextarea.vue.d.ts +7 -12
  16. package/dist/components/BaseToggle/BaseToggle.vue.d.ts +0 -2
  17. package/dist/components/BaseTooltip/BaseTooltip.vue.d.ts +1 -1
  18. package/dist/composables/kit/state.d.ts +1 -2
  19. package/dist/index.d.ts +2 -1
  20. package/dist/index.js +1 -1
  21. package/example/App.vue +170 -128
  22. package/package.json +1 -1
  23. package/src/components/BaseField/BaseField.vue +184 -0
  24. package/src/components/BaseField/README.md +85 -0
  25. package/src/components/BaseInput/BaseInput.vue +162 -228
  26. package/src/components/BaseInputCalendar/BaseInputCalendar.vue +10 -7
  27. package/src/components/BaseInputCurrency/BaseInputCurrency.vue +39 -78
  28. package/src/components/BaseInputEmail/BaseInputEmail.vue +2 -4
  29. package/src/components/BaseInputPhone/BaseInputPhone.vue +29 -89
  30. package/src/components/BaseSelect/BaseSelect.vue +9 -52
  31. package/src/components/BaseSiteInput/BaseSiteInput.vue +11 -63
  32. package/src/components/BaseTextarea/BaseTextarea.vue +3 -30
  33. package/src/composables/kit/state.ts +1 -2
  34. package/src/index.ts +5 -2
  35. package/src/styles/index.scss +87 -2
  36. package/src/styles/root.scss +1 -0
  37. package/src/styles/variables.scss +2 -1
  38. package/src/types/calendar.d.ts +2 -0
  39. package/src/types/field.d.ts +12 -0
  40. package/src/types/input.d.ts +26 -8
  41. package/src/types/utils.d.ts +0 -1
package/example/App.vue CHANGED
@@ -3,154 +3,196 @@
3
3
  <h1>Plugin UI KZT - Компоненты</h1>
4
4
  <p>Ниже представлены все компоненты библиотеки с примерами их использования.</p>
5
5
  <section>
6
- <base-site-input
7
- id="phone-input"
8
- v-model="phoneNumber"
9
- size="large"
10
- placeholder="Введите номер телефона"
11
- :error="phoneError"
12
- hint="Номер телефона должен быть не менее 5 символов"
13
- />
14
- </section>
15
- <section>
16
- <h2>BaseBadgeGroup.vue</h2>
17
-
18
- <base-badge-group
19
- size="medium"
20
- color="primary"
21
- badge-position="right"
6
+ <BaseField
7
+ size="large"
8
+ id="BaseInputEmail"
9
+ label="BaseInputEmail"
10
+ :focusable="false"
11
+ :tabIndex="8"
12
+ :error="errorValue"
22
13
  >
23
- <template #badge-text-container>
24
- New feature
25
- </template>
26
- <template #badge-text>
27
- We’ve just released a new feature
14
+ <template #default="data">
15
+ <BaseInputEmail
16
+ id="BaseInputEmail"
17
+ v-model="calendarModel"
18
+ label="Select date hereBaseInputEmail"
19
+ placeholder="Select date"
20
+ :error="data.error"
21
+ :size="data.size"
22
+ />
28
23
  </template>
29
- </base-badge-group>
30
- <base-badge-group
31
- size="large"
32
- color="gray"
33
- arrowRightIcon
24
+ </BaseField>
25
+ <BaseField
26
+ size="small"
27
+ id="calendar-input"
28
+ label="calendar Input"
29
+ :tabIndex="7"
30
+ :error="errorValue"
34
31
  >
35
- <template #badge-text-container>
36
- New feature
32
+ <template #default="data">
33
+ <base-input-calendar
34
+ id="calendar"
35
+ v-model="calendarModel"
36
+ label="Select date here"
37
+ :size="data.size"
38
+ placeholder="Select date"
39
+ :error="data.error"
40
+ :min-date="new Date()"
41
+ @validationError="handleError"
42
+ />
37
43
  </template>
38
- <template #badge-text>
39
- We’ve just released a new feature
40
- </template>
41
- </base-badge-group>
42
- </section>
43
- <section>
44
- <h2>BaseBadge.vue</h2>
45
- <base-badge
46
- size="extra-small"
47
- color="primary"
48
- class="badge-example"
49
- closable
44
+ </BaseField>
45
+ <BaseField
46
+ size="medium"
47
+ id="textarea-input"
48
+ label="text Input"
49
+ :tabIndex="6"
50
+ :error="errorValue"
50
51
  >
51
- <template #text>Пример бейджа</template>
52
- </base-badge>
53
- <base-badge
54
- size="small"
55
- color="gray"
56
- class="badge-example"
57
- has-dot
58
- >
59
- <template #text>Пример бейджа</template>
60
- </base-badge>
61
- <base-badge
62
- size="medium"
63
- color="error"
64
- class="badge-example"
65
- arrow-up-icon
52
+ <template #default="data">
53
+ <BaseTextarea
54
+ :size="data.size"
55
+ v-model="inputValue"
56
+ id="textarea-input"
57
+ :error="data.error"
58
+ >
59
+ </BaseTextarea>
60
+ </template>
61
+ </BaseField>
62
+ <BaseField
63
+ size="medium"
64
+ id="select-input"
65
+ label="select input"
66
+ :tabIndex="5"
67
+ :error="errorValue"
66
68
  >
67
- <template #text>Пример бейджа</template>
68
- </base-badge>
69
- <base-badge
70
- size="large"
71
- color="warning"
72
- class="badge-example"
73
- arrow-right-icon
69
+ <template #default="data">
70
+ <BaseSelect
71
+ :size="data.size"
72
+ v-model="selectValue"
73
+ :options="options"
74
+ id="select-input"
75
+ :error="data.error"
76
+ >
77
+ <template #left-icon>
78
+ <BaseIcon name="star" />
79
+ </template>
80
+ <template #right-icon>
81
+ <BaseIcon name="close" />
82
+ </template>
83
+ </BaseSelect>
84
+ </template>
85
+ </BaseField>
86
+ <BaseField
87
+ size="medium"
88
+ id="BaseInput "
89
+ label="BaseInput "
90
+ :tabIndex="0"
91
+ :error="errorValue"
74
92
  >
75
- <template #text>Пример бейджа</template>
76
- </base-badge>
77
- <base-badge
78
- size="large"
79
- color="success"
80
- class="badge-example"
93
+ <template #default="data">
94
+ <BaseInput
95
+ :size="data.size"
96
+ v-model="inputValue"
97
+ type="text"
98
+ placeholder="Type here"
99
+ id="BaseInput "
100
+ :error="data.error"
101
+ >
102
+ <template #left-icon>
103
+ <BaseIcon name="star" />
104
+ </template>
105
+ <template #right-icon>
106
+ <BaseIcon name="close" />
107
+ </template>
108
+ </BaseInput>
109
+ </template>
110
+ </BaseField>
111
+ <BaseField
112
+ size="medium"
113
+ id="input-phone"
114
+ label="input-phone"
115
+ :tabIndex="1"
116
+ :error="errorValue"
81
117
  >
82
- <template #text>
83
- <base-icon size="custom" name="help"></base-icon>
118
+ <template #default="data">
119
+ <BaseInputPhone
120
+ :size="data.size"
121
+ v-model="inputValue4"
122
+ type="text"
123
+ placeholder="Type here"
124
+ id="phone-input"
125
+ :error="`${data.error}`"
126
+ :focusable="false"
127
+ />
84
128
  </template>
85
- </base-badge>
86
- </section>
129
+ </BaseField>
87
130
 
88
- <section>
89
- <h2>BaseUpload.vue</h2>
90
- <base-upload
91
- :accepted-formats="accept"
92
- >
93
- <template #title>my title</template>
94
- </base-upload>
95
- </section>
96
-
97
- <section>
98
- <div>
99
- <base-swiper
100
- :images="[
101
- 'https://i0.wp.com/vasilemovileanu.com/wp-content/uploads/2018/09/Autumn.-2001.-Oil-on-canvas-65-x-60-cm.jpg?fit=1231%2C1400&ssl=1',
102
- 'https://i.pinimg.com/originals/00/1b/7b/001b7bc4a4cd9132918accae2c7d5c23.png',
103
- 'https://i.pinimg.com/736x/90/71/bc/9071bcfb179dafd923d0f90653338ea1.jpg'
104
- ]"
105
- :paginationSettings="{ type: 'dotsWithBackground', color: 'primary' }"
106
- v-model:current-slide="currentSlide"
107
- />
108
- <!-- :autoplay="true"
109
- :autoplay-interval="2000" -->
110
- </div>
111
- <section>
112
- medium size
113
- <base-textarea class="text-area" size="medium" placeholder="test"></base-textarea>
114
- </section>
115
- </section>
116
- <section style="height: 300px;">
117
- <p></p>
118
- <base-textarea class="text-area" size="custom" placeholder="test"></base-textarea>
119
- </section>
131
+ <BaseField
132
+ size="medium"
133
+ id="BaseInputCurrency"
134
+ label="BaseInputCurrency"
135
+ :tabIndex="2"
136
+ :error="errorValue"
137
+ >
138
+ <template #default="data">
139
+ <BaseInputCurrency
140
+ :size="data.size"
141
+ v-model="inputValue3"
142
+ type="text"
143
+ placeholder="Type here"
144
+ id="BaseInputCurrency"
145
+ :error="`${data.error}`"
146
+ />
147
+ </template>
148
+ </BaseField>
120
149
 
121
- <section>
122
- <div>
123
- <base-button @click="openModal">Открыть модал</base-button>
124
- </div>
150
+ <BaseField
151
+ size="medium"
152
+ id="BaseSiteInput"
153
+ label="BaseSiteInput"
154
+ :tabIndex="3"
155
+ :focusable="false"
156
+ :error="errorValue"
157
+ >
158
+ <template #default="data">
159
+ <BaseSiteInput
160
+ :size="data.size"
161
+ v-model="inputValue3"
162
+ placeholder="write site url"
163
+ id="BaseSiteInput"
164
+ :error="`${data.error}`"
165
+ />
166
+ </template>
167
+ </BaseField>
168
+
125
169
  </section>
126
170
  </div>
127
171
  </template>
128
172
  <script setup lang="ts">
129
173
  import { ref, watch } from 'vue';
130
- import { useModal } from '../src/composables/useModal';
131
- import MyCustomModal from './MyCustomModal.vue';
132
174
 
133
- const accept = ['.jpg', '.pdf']
134
- const phoneNumber = ref('');
135
- const phoneError = ref('');
136
- watch(phoneNumber, (newValue) => {
137
- console.log('Новый номер телефона:', newValue.length);
138
- if (newValue.length > 17) {
139
- console.log('Номер телефона должен быть не менее 5 символов');
140
- phoneError.value = 'Некорректный номер телефона';
141
- } else {
142
- console.log('Номер телефона корректен');
143
- phoneError.value = '';
144
- }
175
+ const inputValue = ref('as');
176
+ const selectValue = ref('');
177
+ const options = ref([
178
+ { id: '1', name: 'Option 1', value: 'option1' },
179
+ { id: '2', name: 'Option 2', value: 'option2' },
180
+ { id: '3', name: 'Option 3', value: 'option3' },
181
+ ]);
182
+ const inputValue3 = ref('');
183
+ const calendarModel = ref('');
184
+ const inputValue4 = ref('');
185
+ const errorValue = ref('');
186
+ errorValue.value = errorValue.value.length < 3 ? 'Input must be at least 3 characters long' : '';
187
+ watch(inputValue, (newValue) => {
188
+ console.log('Input 1 value changed:', newValue);
189
+ errorValue.value = newValue.length < 3 ? 'Input must be at least 3 characters long' : '';
145
190
  });
146
191
 
147
- const modal = useModal();
148
-
149
- const openModal = () => {
150
- modal.open('myModal', { closable: true }, MyCustomModal);
192
+ const handleError = (value: string) => {
193
+ console.error('Validation error:', value);
194
+ errorValue.value = value;
151
195
  };
152
-
153
- const currentSlide = ref(0);
154
196
  </script>
155
197
 
156
198
  <style lang="scss" scoped>
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "plugin-ui-for-kzt",
3
- "version": "0.0.19",
3
+ "version": "0.0.20",
4
4
  "description": "plugin-ui for kazaktelekom",
5
5
  "main": "dist/index.js",
6
6
  "types": "dist/index.d.ts",
@@ -0,0 +1,184 @@
1
+ <template>
2
+ <div class="base-field" :class="classList">
3
+ <label class="base-field__label" :for="id">
4
+ <span v-if="label" class="base-field__label-text">{{ label }}</span>
5
+ <div class="base-field__wrapper" :class="{ 'base-field__wrapper--focusable': focusable}" :tabindex="tabIndex">
6
+ <slot :data="componentAttrs" />
7
+ </div>
8
+ <transition name="error">
9
+ <div
10
+ v-if="(error && typeof error === 'string') || hint"
11
+ class="base-field__hint"
12
+ aria-live="polite"
13
+ >
14
+ {{ error || hint }}
15
+ </div>
16
+ </transition>
17
+ </label>
18
+ </div>
19
+ </template>
20
+
21
+ <script setup lang="ts">
22
+ import { computed, useAttrs } from 'vue';
23
+ import { useKitSize } from '../../composables/kit/size';
24
+ import { useKitState } from '../../composables/kit/state';
25
+
26
+ import type { TFieldProps } from '../../types/field';
27
+
28
+ interface IDefaultSlotProps {
29
+ id: string;
30
+ error?: string;
31
+ size?: string;
32
+ focusable?: boolean;
33
+ }
34
+
35
+ const props = withDefaults(defineProps<TFieldProps>(), {
36
+ size: 'medium',
37
+ id: '',
38
+ label: '',
39
+ hint: '',
40
+ error: '',
41
+ focusable: true,
42
+ tabIndex: 0,
43
+ });
44
+
45
+ defineEmits<{
46
+ (event: 'update:modelValue', value: string): void;
47
+ (event: 'error', value: string): void;
48
+ }>();
49
+
50
+ const attrs = useAttrs();
51
+ const { stateAttrs } = useKitState(props);
52
+ const { sizeClassList } = useKitSize(props);
53
+
54
+ const componentAttrs = computed<IDefaultSlotProps>(() => ({
55
+ ...(attrs || {}),
56
+ ...(stateAttrs.value || {}),
57
+ id: props.id,
58
+ error: props.error,
59
+ size: props.size,
60
+ }));
61
+
62
+ const classList = computed(() => [
63
+ ...sizeClassList.value,
64
+ { '--is-error': props.error && typeof props.error === 'string' },
65
+ ]);
66
+ </script>
67
+
68
+ <style scoped lang="scss">
69
+ @import '../../styles/variables';
70
+ @import '../../styles/root';
71
+
72
+ .base-field {
73
+ width: 100%;
74
+
75
+ &__wrapper {
76
+ display: flex;
77
+ flex-direction: column;
78
+ width: 100%;
79
+ position: relative;
80
+ transition: all var(--transition);
81
+
82
+ &--focusable {
83
+ @include focused {
84
+ border: 1px solid var(--primary-blue-500);
85
+ outline: 4px solid var(--effects-primary-focus);
86
+ }
87
+ }
88
+ }
89
+
90
+ &__label {
91
+ display: flex;
92
+ flex-direction: column;
93
+ gap: 6px;
94
+ width: 100%;
95
+ }
96
+
97
+ &__label-text {
98
+ color: var(--primary-text-primary);
99
+ }
100
+
101
+ &__hint {
102
+ color: var(--primary-text-secondary);
103
+ overflow: hidden;
104
+ max-height: 100px;
105
+ transition: all 0.3s ease;
106
+ }
107
+
108
+ &.--is-error {
109
+ .base-field__wrapper {
110
+ &--focusable {
111
+ @include focused {
112
+ border: none;
113
+ outline: 4px solid var(--effects-error-focus);
114
+ }
115
+ }
116
+ }
117
+
118
+ .base-field__hint {
119
+ color: var(--error-red);
120
+ }
121
+ }
122
+
123
+ &.--small-size {
124
+ .base-field__wrapper {
125
+ border-radius: 10px;
126
+ }
127
+
128
+ .base-field__label-text {
129
+ font: var(--typography-text-s-medium);
130
+ }
131
+
132
+ .base-field__hint {
133
+ font: var(--typography-text-s-regular);
134
+ }
135
+ }
136
+
137
+ &.--medium-size {
138
+ .base-field__wrapper {
139
+ border-radius: 12px;
140
+ }
141
+
142
+ .base-field__label-text {
143
+ font: var(--typography-text-s-medium);
144
+ }
145
+
146
+ .base-field__hint {
147
+ font: var(--typography-text-s-regular);
148
+ }
149
+ }
150
+
151
+ &.--large-size {
152
+ .base-field__wrapper {
153
+ border-radius: 12px;
154
+ }
155
+
156
+ .base-field__label-text {
157
+ font: var(--typography-text-m-medium);
158
+ }
159
+
160
+ .base-field__hint {
161
+ font: var(--typography-text-m-regular);
162
+ }
163
+ }
164
+ }
165
+
166
+ .error-enter-active,
167
+ .error-leave-active {
168
+ transition: opacity 0.3s ease, max-height 0.3s ease, transform 0.3s ease;
169
+ }
170
+
171
+ .error-enter-from,
172
+ .error-leave-to {
173
+ opacity: 0;
174
+ transform: translateY(10px);
175
+ max-height: 0;
176
+ }
177
+
178
+ .error-enter-to,
179
+ .error-leave-from {
180
+ opacity: 1;
181
+ transform: translateY(0);
182
+ max-height: 100px;
183
+ }
184
+ </style>
@@ -0,0 +1,85 @@
1
+ BaseField Component
2
+ BaseField — это универсальный компонент-обертка для полей ввода, который предоставляет стилизованную обертку, метку (label), подсказку (hint) и обработку ошибок. Компонент предназначен для использования в связке с другими компонентами ввода, такими как BaseInput, через механизм слотов Vue.
3
+
4
+
5
+ Использование
6
+ Компонент BaseField используется как обертка для других компонентов ввода, таких как BaseInput. Он предоставляет метку, подсказку и стили для обработки ошибок.
7
+ Пример использования
8
+ <template>
9
+ <BaseField
10
+ id="test-input"
11
+ label="Test Input"
12
+ hint="Enter your text"
13
+ :error="errorValue"
14
+ size="medium"
15
+ >
16
+ <BaseInput
17
+ v-model="inputValue"
18
+ type="text"
19
+ placeholder="Type here"
20
+ id="test-input"
21
+ />
22
+ </BaseField>
23
+ </template>
24
+
25
+ <script setup lang="ts">
26
+ import { ref } from 'vue';
27
+ import BaseField from './BaseField.vue';
28
+ import BaseInput from './BaseInput.vue';
29
+
30
+ const inputValue = ref('');
31
+ const errorValue = ref('Input must be at least 3 characters long');
32
+ </script>
33
+
34
+
35
+ События
36
+
37
+ update:modelValue
38
+ value: string
39
+ Эмитируется при обновлении значения поля ввода.
40
+
41
+ error
42
+ value: string
43
+ Эмитируется при возникновении ошибки в поле ввода.
44
+
45
+
46
+ Слоты
47
+
48
+ default
49
+ Основной слот для размещения компонента ввода (например, BaseInput).
50
+
51
+ Слот передает объект с пропсами, включая componentAttrs, который содержит объединенные атрибуты из useAttrs и stateAttrs.
52
+
53
+
54
+ Стили
55
+ Компонент использует SCSS для стилизации. Основные классы:
56
+
57
+ .base-field: Основной контейнер компонента.
58
+ .base-field__label: Контейнер для метки и поля ввода.
59
+ .base-field__label-text: Текст метки.
60
+ .base-field__wrapper: Обертка для содержимого слота.
61
+ .base-field__hint: Текст подсказки или ошибки.
62
+ .--is-error: Модификатор для отображения состояния ошибки.
63
+ .--small-size, .--medium-size, .--large-size: Модификаторы для разных размеров.
64
+
65
+
66
+ Типизация
67
+ Компонент использует TypeScript и ожидает тип TFieldProps из файла types/field. Пример структуры:
68
+ export interface TFieldProps {
69
+ id?: string;
70
+ label?: string;
71
+ hint?: string;
72
+ error?: string | boolean;
73
+ size?: string;
74
+ focusable?: boolean;
75
+ tabIndex?: number;
76
+ }
77
+
78
+ Примечания
79
+
80
+ Если focusable равно false, атрибут tabindex не применяется, и обертка не будет фокусируемой.
81
+
82
+ Ограничения
83
+
84
+ Компонент предполагает наличие вложенного компонента ввода в слоте (например, BaseInput). Без содержимого слота функциональность ограничена.
85
+ Стили и размеры зависят от внешних переменных и composable-функций, которые должны быть настроены в проекте.