@true-engineering/true-react-common-ui-kit 1.8.0 → 1.8.1

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 (238) hide show
  1. package/LICENSE +201 -201
  2. package/dist/components/Flag/augment.d.ts +1 -1
  3. package/dist/components/FlexibleTable/types.d.ts +1 -1
  4. package/dist/components/Icon/complexIcons/augment.d.ts +1 -1
  5. package/dist/true-react-common-ui-kit.js +58 -58
  6. package/dist/true-react-common-ui-kit.js.map +1 -1
  7. package/dist/true-react-common-ui-kit.umd.cjs +58 -58
  8. package/dist/true-react-common-ui-kit.umd.cjs.map +1 -1
  9. package/dist/vite-env.d.ts +1 -1
  10. package/package.json +91 -91
  11. package/src/components/AccountInfo/AccountInfo.stories.tsx +35 -35
  12. package/src/components/AccountInfo/AccountInfo.styles.ts +55 -55
  13. package/src/components/AccountInfo/AccountInfo.tsx +106 -106
  14. package/src/components/AccountInfo/index.ts +2 -2
  15. package/src/components/AddButton/AddButton.stories.tsx +21 -21
  16. package/src/components/AddButton/AddButton.styles.ts +34 -34
  17. package/src/components/AddButton/AddButton.tsx +49 -49
  18. package/src/components/AddButton/index.ts +2 -2
  19. package/src/components/Button/Button.stories.tsx +61 -61
  20. package/src/components/Button/Button.styles.ts +196 -196
  21. package/src/components/Button/Button.tsx +195 -195
  22. package/src/components/Button/index.ts +2 -2
  23. package/src/components/Checkbox/Checkbox.stories.tsx +35 -35
  24. package/src/components/Checkbox/Checkbox.styles.ts +62 -62
  25. package/src/components/Checkbox/Checkbox.tsx +106 -106
  26. package/src/components/Checkbox/index.ts +2 -2
  27. package/src/components/CloseButton/CloseButton.styles.ts +34 -34
  28. package/src/components/CloseButton/CloseButton.tsx +37 -37
  29. package/src/components/CloseButton/index.ts +2 -2
  30. package/src/components/Colors/Colors.stories.tsx +7 -7
  31. package/src/components/Colors/Colors.styles.ts +38 -38
  32. package/src/components/Colors/Colors.tsx +34 -34
  33. package/src/components/Colors/index.ts +2 -2
  34. package/src/components/CssBaseline/CssBaseline.styles.ts +15 -15
  35. package/src/components/CssBaseline/CssBaseline.tsx +17 -17
  36. package/src/components/CssBaseline/index.ts +2 -2
  37. package/src/components/DateInput/DateInput.stories.tsx +63 -63
  38. package/src/components/DateInput/DateInput.styles.ts +14 -14
  39. package/src/components/DateInput/DateInput.tsx +60 -60
  40. package/src/components/DateInput/index.ts +2 -2
  41. package/src/components/DatePicker/DatePicker.stories.tsx +96 -96
  42. package/src/components/DatePicker/DatePicker.styles.ts +54 -54
  43. package/src/components/DatePicker/DatePicker.tsx +358 -358
  44. package/src/components/DatePicker/DatePickerHeader/DatePickerHeader.styles.ts +84 -84
  45. package/src/components/DatePicker/DatePickerHeader/DatePickerHeader.tsx +94 -94
  46. package/src/components/DatePicker/DatePickerHeader/index.ts +1 -1
  47. package/src/components/DatePicker/DatePickerInput/DatePickerInput.styles.ts +25 -25
  48. package/src/components/DatePicker/DatePickerInput/DatePickerInput.tsx +31 -31
  49. package/src/components/DatePicker/DatePickerInput/index.ts +1 -1
  50. package/src/components/DatePicker/index.ts +4 -4
  51. package/src/components/Description/Description.stories.tsx +29 -29
  52. package/src/components/Description/Description.styles.ts +31 -31
  53. package/src/components/Description/Description.tsx +69 -69
  54. package/src/components/Description/index.ts +2 -2
  55. package/src/components/FiltersPane/FilterInterval/FilterInterval.styles.ts +64 -64
  56. package/src/components/FiltersPane/FilterInterval/FilterInterval.tsx +162 -162
  57. package/src/components/FiltersPane/FilterInterval/index.ts +1 -1
  58. package/src/components/FiltersPane/FilterMultiSelect/FilterMultiSelect.tsx +14 -14
  59. package/src/components/FiltersPane/FilterMultiSelect/index.ts +1 -1
  60. package/src/components/FiltersPane/FilterSelect/FilterSelect.styles.ts +144 -144
  61. package/src/components/FiltersPane/FilterSelect/FilterSelect.tsx +397 -397
  62. package/src/components/FiltersPane/FilterSelect/index.ts +1 -1
  63. package/src/components/FiltersPane/FilterSelect/locales.ts +37 -37
  64. package/src/components/FiltersPane/FilterValueView/FilterValueView.styles.tsx +15 -15
  65. package/src/components/FiltersPane/FilterValueView/FilterValueView.tsx +186 -186
  66. package/src/components/FiltersPane/FilterValueView/index.tsx +1 -1
  67. package/src/components/FiltersPane/FilterWithDates/FilterWithDates.styles.ts +60 -60
  68. package/src/components/FiltersPane/FilterWithDates/FilterWithDates.tsx +222 -222
  69. package/src/components/FiltersPane/FilterWithDates/index.ts +1 -1
  70. package/src/components/FiltersPane/FilterWithPeriod/FilterWithPeriod.styles.ts +17 -17
  71. package/src/components/FiltersPane/FilterWithPeriod/FilterWithPeriod.tsx +231 -231
  72. package/src/components/FiltersPane/FilterWithPeriod/index.ts +1 -1
  73. package/src/components/FiltersPane/FilterWrapper/FilterWrapper.styles.ts +110 -110
  74. package/src/components/FiltersPane/FilterWrapper/FilterWrapper.tsx +360 -360
  75. package/src/components/FiltersPane/FilterWrapper/index.ts +1 -1
  76. package/src/components/FiltersPane/FiltersPane.stories.tsx +308 -308
  77. package/src/components/FiltersPane/FiltersPane.styles.ts +71 -71
  78. package/src/components/FiltersPane/FiltersPane.tsx +193 -193
  79. package/src/components/FiltersPane/FiltersPaneSearch/FiltersPaneSearch.styles.ts +109 -109
  80. package/src/components/FiltersPane/FiltersPaneSearch/FiltersPaneSearch.tsx +175 -175
  81. package/src/components/FiltersPane/FiltersPaneSearch/index.ts +1 -1
  82. package/src/components/FiltersPane/index.ts +20 -20
  83. package/src/components/FiltersPane/locales.ts +107 -107
  84. package/src/components/FiltersPane/types.ts +126 -126
  85. package/src/components/Flag/Flag.stories.tsx +29 -29
  86. package/src/components/Flag/Flag.styles.ts +18 -18
  87. package/src/components/Flag/Flag.tsx +28 -28
  88. package/src/components/Flag/augment.d.ts +1 -1
  89. package/src/components/Flag/index.ts +2 -2
  90. package/src/components/FlexibleTable/FlexibleTable.stories.tsx +86 -86
  91. package/src/components/FlexibleTable/FlexibleTable.styles.ts +131 -131
  92. package/src/components/FlexibleTable/FlexibleTable.tsx +243 -243
  93. package/src/components/FlexibleTable/TableRow.tsx +171 -171
  94. package/src/components/FlexibleTable/TableValue.tsx +81 -83
  95. package/src/components/FlexibleTable/fixture-test.ts +254 -254
  96. package/src/components/FlexibleTable/index.ts +3 -3
  97. package/src/components/FlexibleTable/types.ts +58 -58
  98. package/src/components/Icon/ComplexIconBoilerplate.tsx +17 -17
  99. package/src/components/Icon/Icon.stories.tsx +88 -88
  100. package/src/components/Icon/Icon.styles.ts +10 -10
  101. package/src/components/Icon/Icon.tsx +34 -34
  102. package/src/components/Icon/IconBoilerplate.tsx +42 -42
  103. package/src/components/Icon/complexIcons/augment.d.ts +1 -1
  104. package/src/components/Icon/complexIcons/avatarGreen.svg +57 -57
  105. package/src/components/Icon/complexIcons/icons.ts +7 -7
  106. package/src/components/Icon/complexIcons/index.ts +1 -1
  107. package/src/components/Icon/icons/icons.ts +838 -838
  108. package/src/components/Icon/icons/index.ts +1 -1
  109. package/src/components/Icon/index.ts +4 -4
  110. package/src/components/IncrementInput/ChangeButton.tsx +34 -34
  111. package/src/components/IncrementInput/IncrementInput.stories.tsx +34 -34
  112. package/src/components/IncrementInput/IncrementInput.styles.ts +77 -77
  113. package/src/components/IncrementInput/IncrementInput.tsx +95 -95
  114. package/src/components/IncrementInput/index.ts +2 -2
  115. package/src/components/Input/Input.stories.tsx +92 -92
  116. package/src/components/Input/Input.styles.ts +305 -305
  117. package/src/components/Input/Input.tsx +318 -318
  118. package/src/components/Input/index.ts +2 -2
  119. package/src/components/List/List.stories.tsx +62 -62
  120. package/src/components/List/List.styles.ts +52 -52
  121. package/src/components/List/List.tsx +82 -82
  122. package/src/components/List/index.ts +2 -2
  123. package/src/components/Modal/Modal.stories.tsx +113 -113
  124. package/src/components/Modal/Modal.styles.ts +308 -308
  125. package/src/components/Modal/Modal.tsx +210 -210
  126. package/src/components/Modal/index.ts +2 -2
  127. package/src/components/MoreMenu/MoreMenu.stories.tsx +46 -46
  128. package/src/components/MoreMenu/MoreMenu.styles.ts +70 -70
  129. package/src/components/MoreMenu/MoreMenu.tsx +102 -102
  130. package/src/components/MoreMenu/index.ts +2 -2
  131. package/src/components/MultiSelect/MultiSelect.stories.tsx +46 -46
  132. package/src/components/MultiSelect/MultiSelect.styles.ts +55 -55
  133. package/src/components/MultiSelect/MultiSelect.tsx +98 -98
  134. package/src/components/MultiSelect/MultiSelectInput/MultiSelectInput.styles.ts +73 -73
  135. package/src/components/MultiSelect/MultiSelectInput/MultiSelectInput.tsx +62 -62
  136. package/src/components/MultiSelect/MultiSelectInput/index.ts +1 -1
  137. package/src/components/MultiSelect/index.ts +3 -3
  138. package/src/components/MultiSelectList/MultiSelectList.styles.ts +125 -125
  139. package/src/components/MultiSelectList/MultiSelectList.tsx +519 -519
  140. package/src/components/MultiSelectList/index.ts +2 -2
  141. package/src/components/MultiSelectList/locales.ts +37 -37
  142. package/src/components/Notification/Notification.stories.tsx +51 -51
  143. package/src/components/Notification/Notification.styles.ts +50 -50
  144. package/src/components/Notification/Notification.tsx +84 -84
  145. package/src/components/Notification/index.ts +2 -2
  146. package/src/components/NumberInput/NumberInput.stories.tsx +36 -36
  147. package/src/components/NumberInput/NumberInput.tsx +154 -154
  148. package/src/components/NumberInput/helpers.ts +87 -87
  149. package/src/components/NumberInput/index.ts +1 -1
  150. package/src/components/PhoneInput/PhoneInput.stories.tsx +71 -71
  151. package/src/components/PhoneInput/PhoneInput.styles.ts +84 -84
  152. package/src/components/PhoneInput/PhoneInput.tsx +223 -223
  153. package/src/components/PhoneInput/PhoneInputCountryList/PhoneInputCountryList.stories.tsx +21 -21
  154. package/src/components/PhoneInput/PhoneInputCountryList/PhoneInputCountryList.styles.ts +100 -100
  155. package/src/components/PhoneInput/PhoneInputCountryList/PhoneInputCountryList.tsx +171 -171
  156. package/src/components/PhoneInput/PhoneInputCountryList/index.ts +2 -2
  157. package/src/components/PhoneInput/index.ts +6 -6
  158. package/src/components/PhoneInput/phone-info.ts +2167 -2167
  159. package/src/components/PhoneInput/types.ts +16 -16
  160. package/src/components/RadioButton/RadioButton.stories.tsx +46 -46
  161. package/src/components/RadioButton/RadioButton.styles.ts +37 -37
  162. package/src/components/RadioButton/RadioButton.tsx +56 -56
  163. package/src/components/RadioButton/index.ts +2 -2
  164. package/src/components/ScrollIntoViewIfNeeded/ScrollIntoViewIfNeeded.ts +66 -66
  165. package/src/components/ScrollIntoViewIfNeeded/index.ts +1 -1
  166. package/src/components/SearchInput/SearchInput.stories.tsx +24 -24
  167. package/src/components/SearchInput/SearchInput.styles.ts +50 -50
  168. package/src/components/SearchInput/SearchInput.tsx +63 -63
  169. package/src/components/SearchInput/index.ts +2 -2
  170. package/src/components/Select/Select.stories.tsx +258 -258
  171. package/src/components/Select/Select.styles.ts +85 -85
  172. package/src/components/Select/Select.tsx +510 -510
  173. package/src/components/Select/SelectList/SelectList.styles.ts +68 -68
  174. package/src/components/Select/SelectList/SelectList.tsx +139 -139
  175. package/src/components/Select/SelectList/index.ts +1 -1
  176. package/src/components/Select/helpers.ts +21 -21
  177. package/src/components/Select/index.ts +3 -3
  178. package/src/components/SmartInput/SmartInput.stories.tsx +63 -63
  179. package/src/components/SmartInput/SmartInput.tsx +180 -180
  180. package/src/components/SmartInput/helpers.ts +85 -85
  181. package/src/components/SmartInput/index.ts +1 -1
  182. package/src/components/Switch/Switch.stories.tsx +40 -40
  183. package/src/components/Switch/Switch.styles.ts +75 -75
  184. package/src/components/Switch/Switch.tsx +89 -89
  185. package/src/components/Switch/index.ts +2 -2
  186. package/src/components/TextArea/TextArea.stories.tsx +35 -35
  187. package/src/components/TextArea/TextArea.styles.ts +153 -153
  188. package/src/components/TextArea/TextArea.tsx +178 -178
  189. package/src/components/TextArea/index.ts +2 -2
  190. package/src/components/TextWithInfo/TextWithInfo.stories.tsx +53 -53
  191. package/src/components/TextWithInfo/TextWithInfo.styles.ts +60 -60
  192. package/src/components/TextWithInfo/TextWithInfo.tsx +67 -67
  193. package/src/components/TextWithInfo/index.ts +2 -2
  194. package/src/components/TextWithTooltip/TextWithTooltip.stories.tsx +58 -58
  195. package/src/components/TextWithTooltip/TextWithTooltip.styles.ts +19 -19
  196. package/src/components/TextWithTooltip/TextWithTooltip.tsx +163 -163
  197. package/src/components/TextWithTooltip/index.ts +2 -2
  198. package/src/components/ThemedPreloader/ThemedPreloader.stories.tsx +41 -41
  199. package/src/components/ThemedPreloader/ThemedPreloader.styles.ts +21 -21
  200. package/src/components/ThemedPreloader/ThemedPreloader.tsx +56 -56
  201. package/src/components/ThemedPreloader/components/DefaultPreloader/DefaultPreloader.tsx +34 -34
  202. package/src/components/ThemedPreloader/components/DefaultPreloader/index.ts +1 -1
  203. package/src/components/ThemedPreloader/components/DotsPreloader/DotsPreloader.styles.ts +54 -54
  204. package/src/components/ThemedPreloader/components/DotsPreloader/DotsPreloader.tsx +18 -18
  205. package/src/components/ThemedPreloader/components/DotsPreloader/index.ts +2 -2
  206. package/src/components/ThemedPreloader/components/SvgPreloader/SvgPreloader.styles.ts +11 -11
  207. package/src/components/ThemedPreloader/components/SvgPreloader/SvgPreloader.tsx +32 -32
  208. package/src/components/ThemedPreloader/components/SvgPreloader/index.ts +2 -2
  209. package/src/components/ThemedPreloader/components/index.ts +2 -2
  210. package/src/components/ThemedPreloader/index.ts +2 -2
  211. package/src/components/Toaster/Toaster.stories.tsx +34 -34
  212. package/src/components/Toaster/Toaster.styles.ts +59 -59
  213. package/src/components/Toaster/Toaster.tsx +113 -113
  214. package/src/components/Toaster/index.ts +2 -2
  215. package/src/components/Tooltip/Tooltip.stories.tsx +21 -21
  216. package/src/components/Tooltip/Tooltip.styles.ts +45 -45
  217. package/src/components/Tooltip/Tooltip.tsx +40 -40
  218. package/src/components/Tooltip/index.ts +3 -3
  219. package/src/components/Tooltip/types.ts +1 -1
  220. package/src/components/index.ts +36 -36
  221. package/src/helpers/colors.ts +2 -2
  222. package/src/helpers/dateHelpers/date-helpers.ts +9 -9
  223. package/src/helpers/index.ts +4 -4
  224. package/src/helpers/phone.ts +106 -106
  225. package/src/helpers/popper-helpers.ts +17 -17
  226. package/src/helpers/snippets.tsx +5 -5
  227. package/src/helpers/utils.ts +219 -219
  228. package/src/hooks/index.ts +6 -6
  229. package/src/hooks/use-did-mount-effect.ts +21 -21
  230. package/src/hooks/use-dropdown.ts +85 -85
  231. package/src/hooks/use-is-mounted.ts +15 -15
  232. package/src/hooks/use-on-click-outside.ts +92 -92
  233. package/src/hooks/use-theme.ts +36 -36
  234. package/src/hooks/use-tweak-styles.ts +14 -14
  235. package/src/index.ts +6 -6
  236. package/src/theme.ts +155 -155
  237. package/src/types.ts +106 -106
  238. package/src/vite-env.d.ts +1 -1
@@ -1,219 +1,219 @@
1
- import { IDataAttributes } from '../types';
2
- import { HTMLAttributes, KeyboardEvent, MouseEvent } from 'react';
3
-
4
- export const transformToKebab = (string: string): string => {
5
- let result = '';
6
- string.split('').forEach((char) => {
7
- if (char.toLowerCase() === char) {
8
- result += char;
9
- } else {
10
- result += `-${char.toLowerCase()}`;
11
- }
12
- });
13
-
14
- return result;
15
- };
16
-
17
- export const hasExactParent = (element: Element, parent: Element): boolean => {
18
- if (element === parent) {
19
- return true; // Found the exact parent
20
- }
21
-
22
- const parentNode = getParentNode(element);
23
-
24
- if (parentNode === element) {
25
- return false; // Reached the top-level HTML element or Shadow DOM host
26
- }
27
-
28
- return hasExactParent(parentNode, parent);
29
- };
30
-
31
- export const getParentNode = (element: Element | ShadowRoot): Element =>
32
- element.nodeName === 'HTML'
33
- ? (element as Element)
34
- : (element.parentNode as Element) ?? (element as ShadowRoot).host;
35
-
36
- export const getStyleComputedProperty = (
37
- element: Element,
38
- ): Partial<CSSStyleDeclaration> =>
39
- element.nodeType !== 1 ? {} : getComputedStyle(element, null);
40
-
41
- export const getScrollParent = (element: Element | Document): Element => {
42
- if (!element) {
43
- return document.body;
44
- }
45
-
46
- switch (element.nodeName) {
47
- case 'HTML':
48
- case 'BODY':
49
- return (element as Element).ownerDocument.body;
50
- case '#document':
51
- return (element as Document).body;
52
- }
53
-
54
- const { overflow, overflowX, overflowY } =
55
- getStyleComputedProperty(element as Element) ?? {};
56
- if (
57
- /(auto|scroll|overlay)/.test(
58
- (overflow ?? '') + (overflowY ?? '') + (overflowX ?? ''),
59
- )
60
- ) {
61
- return element as Element;
62
- }
63
- return getScrollParent(getParentNode(element as Element));
64
- };
65
-
66
- export const isElementOffScreen = (
67
- element: HTMLElement,
68
- input?: HTMLElement,
69
- ): boolean => {
70
- const el = element;
71
- const scrollParent = getScrollParent(element);
72
-
73
- const { scrollHeight: scrollHeightWithElement } = scrollParent;
74
- el.hidden = true;
75
- const { scrollHeight: scrollHeightWithoutElement } = scrollParent;
76
- el.hidden = false;
77
-
78
- const isOffscreen = scrollHeightWithElement !== scrollHeightWithoutElement;
79
-
80
- if (isOffscreen && input !== undefined) {
81
- const elRect = el.getBoundingClientRect();
82
- const scrollParentRect = scrollParent.getBoundingClientRect();
83
- const topOffset = elRect.top - scrollParentRect.top;
84
- if (input.clientHeight + el.clientHeight > topOffset) {
85
- return false;
86
- }
87
- }
88
-
89
- return isOffscreen;
90
- };
91
-
92
- export const getNumberInRange = (
93
- value: number,
94
- min = -Infinity,
95
- max = Infinity,
96
- ): number => Math.min(max, Math.max(min, value));
97
-
98
- const DEFAULT_THOUSANDS_SEPARATOR = '\u2009';
99
-
100
- export const formatStringNumber = (
101
- val?: string,
102
- separator = DEFAULT_THOUSANDS_SEPARATOR,
103
- ): string => {
104
- if (val === undefined) {
105
- return '';
106
- }
107
- const parts = val.split('.');
108
-
109
- parts[0] = parts[0]
110
- // убрать лидирующие нули
111
- .replace(/^0+(?=\d)/, '')
112
- // проставить сепараторы тысяч
113
- .replace(/\B(?=(\d{3})+(?!\d))/g, separator);
114
- return (parts[1] ?? '').length > 0 ? parts.join('.') : parts[0];
115
- };
116
-
117
- export const formatNumber = (
118
- val?: number,
119
- separator = DEFAULT_THOUSANDS_SEPARATOR,
120
- ): string => {
121
- if (val === undefined || isNaN(val)) {
122
- return '';
123
- }
124
- return formatStringNumber(String(val), separator);
125
- };
126
-
127
- export const removeStringFormat = (val?: string): string =>
128
- (val ?? '').replace(',', '.').replace(/\s/g, '');
129
-
130
- export const stringToNumber = (val?: string): number | undefined => {
131
- const trimmed = removeStringFormat(val);
132
- if (trimmed === '') {
133
- return undefined;
134
- }
135
- const num = Number(trimmed);
136
- return isNaN(num) ? undefined : num;
137
- };
138
-
139
- export const setCaretPosition = (
140
- elem: HTMLInputElement,
141
- caretPos: number | null,
142
- ): void => {
143
- if (caretPos === null || elem === null) {
144
- return;
145
- }
146
- if (elem.selectionStart) {
147
- elem.focus();
148
- elem.setSelectionRange(caretPos, caretPos);
149
- } else {
150
- elem.focus();
151
- }
152
- };
153
-
154
- export const isSpaceChar = (char?: string): boolean =>
155
- char !== undefined && char.match(/\s/) !== null;
156
-
157
- export const isInt = (n: number): boolean => n % 1 === 0;
158
-
159
- export const getNumberLength = (n?: number): number =>
160
- n === undefined || isNaN(n) ? 0 : n.toString().length;
161
-
162
- /**
163
- * Проверяет, что `val` не `null`, не `undefined` и не пустая строка
164
- */
165
- export const isNotEmpty = <T>(val: T | null | undefined): val is T =>
166
- typeof val === 'string'
167
- ? val.trim() !== ''
168
- : val !== null && val !== undefined;
169
-
170
- export const trimStringToMaxLength = (val: string, maxLength: number) =>
171
- val.length > maxLength ? val.slice(0, maxLength) : val;
172
-
173
- export const addDataAttributes = (
174
- data: IDataAttributes = {},
175
- ): IDataAttributes =>
176
- Object.fromEntries(
177
- Object.entries(data).map(([key, value]) =>
178
- isNotEmpty(value) ? [`data-${transformToKebab(key)}`, value] : [],
179
- ),
180
- );
181
-
182
- export const addDataTestId = (
183
- ...args: Parameters<typeof getTestId>
184
- ): { 'data-testid': string } | undefined => {
185
- const testId = getTestId(...args);
186
- return isNotEmpty(testId) ? { 'data-testid': testId } : undefined;
187
- };
188
-
189
- export const getTestId = (
190
- testId: string | undefined,
191
- postfix?: string | number,
192
- ): string | undefined => {
193
- if (!isNotEmpty(testId)) {
194
- return undefined;
195
- }
196
- return isNotEmpty(postfix) ? `${testId}-${postfix}` : testId;
197
- };
198
-
199
- export const getSelectKeyHandler =
200
- (cb: (e: KeyboardEvent) => void): ((e: KeyboardEvent) => void) =>
201
- (e) => {
202
- if (e.code === 'Enter' || e.code === 'NumpadEnter') {
203
- cb(e);
204
- }
205
- };
206
-
207
- export const addClickHandler = (
208
- cb?: (e: MouseEvent | KeyboardEvent) => void,
209
- hasAction = true,
210
- ): HTMLAttributes<unknown> =>
211
- hasAction && isNotEmpty(cb)
212
- ? {
213
- tabIndex: 0,
214
- onClick: cb as (e: MouseEvent) => void,
215
- onKeyDown: getSelectKeyHandler(cb),
216
- }
217
- : {
218
- tabIndex: -1,
219
- };
1
+ import { IDataAttributes } from '../types';
2
+ import { HTMLAttributes, KeyboardEvent, MouseEvent } from 'react';
3
+
4
+ export const transformToKebab = (string: string): string => {
5
+ let result = '';
6
+ string.split('').forEach((char) => {
7
+ if (char.toLowerCase() === char) {
8
+ result += char;
9
+ } else {
10
+ result += `-${char.toLowerCase()}`;
11
+ }
12
+ });
13
+
14
+ return result;
15
+ };
16
+
17
+ export const hasExactParent = (element: Element, parent: Element): boolean => {
18
+ if (element === parent) {
19
+ return true; // Found the exact parent
20
+ }
21
+
22
+ const parentNode = getParentNode(element);
23
+
24
+ if (parentNode === element) {
25
+ return false; // Reached the top-level HTML element or Shadow DOM host
26
+ }
27
+
28
+ return hasExactParent(parentNode, parent);
29
+ };
30
+
31
+ export const getParentNode = (element: Element | ShadowRoot): Element =>
32
+ element.nodeName === 'HTML'
33
+ ? (element as Element)
34
+ : (element.parentNode as Element) ?? (element as ShadowRoot).host;
35
+
36
+ export const getStyleComputedProperty = (
37
+ element: Element,
38
+ ): Partial<CSSStyleDeclaration> =>
39
+ element.nodeType !== 1 ? {} : getComputedStyle(element, null);
40
+
41
+ export const getScrollParent = (element: Element | Document): Element => {
42
+ if (!element) {
43
+ return document.body;
44
+ }
45
+
46
+ switch (element.nodeName) {
47
+ case 'HTML':
48
+ case 'BODY':
49
+ return (element as Element).ownerDocument.body;
50
+ case '#document':
51
+ return (element as Document).body;
52
+ }
53
+
54
+ const { overflow, overflowX, overflowY } =
55
+ getStyleComputedProperty(element as Element) ?? {};
56
+ if (
57
+ /(auto|scroll|overlay)/.test(
58
+ (overflow ?? '') + (overflowY ?? '') + (overflowX ?? ''),
59
+ )
60
+ ) {
61
+ return element as Element;
62
+ }
63
+ return getScrollParent(getParentNode(element as Element));
64
+ };
65
+
66
+ export const isElementOffScreen = (
67
+ element: HTMLElement,
68
+ input?: HTMLElement,
69
+ ): boolean => {
70
+ const el = element;
71
+ const scrollParent = getScrollParent(element);
72
+
73
+ const { scrollHeight: scrollHeightWithElement } = scrollParent;
74
+ el.hidden = true;
75
+ const { scrollHeight: scrollHeightWithoutElement } = scrollParent;
76
+ el.hidden = false;
77
+
78
+ const isOffscreen = scrollHeightWithElement !== scrollHeightWithoutElement;
79
+
80
+ if (isOffscreen && input !== undefined) {
81
+ const elRect = el.getBoundingClientRect();
82
+ const scrollParentRect = scrollParent.getBoundingClientRect();
83
+ const topOffset = elRect.top - scrollParentRect.top;
84
+ if (input.clientHeight + el.clientHeight > topOffset) {
85
+ return false;
86
+ }
87
+ }
88
+
89
+ return isOffscreen;
90
+ };
91
+
92
+ export const getNumberInRange = (
93
+ value: number,
94
+ min = -Infinity,
95
+ max = Infinity,
96
+ ): number => Math.min(max, Math.max(min, value));
97
+
98
+ const DEFAULT_THOUSANDS_SEPARATOR = '\u2009';
99
+
100
+ export const formatStringNumber = (
101
+ val?: string,
102
+ separator = DEFAULT_THOUSANDS_SEPARATOR,
103
+ ): string => {
104
+ if (val === undefined) {
105
+ return '';
106
+ }
107
+ const parts = val.split('.');
108
+
109
+ parts[0] = parts[0]
110
+ // убрать лидирующие нули
111
+ .replace(/^0+(?=\d)/, '')
112
+ // проставить сепараторы тысяч
113
+ .replace(/\B(?=(\d{3})+(?!\d))/g, separator);
114
+ return (parts[1] ?? '').length > 0 ? parts.join('.') : parts[0];
115
+ };
116
+
117
+ export const formatNumber = (
118
+ val?: number,
119
+ separator = DEFAULT_THOUSANDS_SEPARATOR,
120
+ ): string => {
121
+ if (val === undefined || isNaN(val)) {
122
+ return '';
123
+ }
124
+ return formatStringNumber(String(val), separator);
125
+ };
126
+
127
+ export const removeStringFormat = (val?: string): string =>
128
+ (val ?? '').replace(',', '.').replace(/\s/g, '');
129
+
130
+ export const stringToNumber = (val?: string): number | undefined => {
131
+ const trimmed = removeStringFormat(val);
132
+ if (trimmed === '') {
133
+ return undefined;
134
+ }
135
+ const num = Number(trimmed);
136
+ return isNaN(num) ? undefined : num;
137
+ };
138
+
139
+ export const setCaretPosition = (
140
+ elem: HTMLInputElement,
141
+ caretPos: number | null,
142
+ ): void => {
143
+ if (caretPos === null || elem === null) {
144
+ return;
145
+ }
146
+ if (elem.selectionStart) {
147
+ elem.focus();
148
+ elem.setSelectionRange(caretPos, caretPos);
149
+ } else {
150
+ elem.focus();
151
+ }
152
+ };
153
+
154
+ export const isSpaceChar = (char?: string): boolean =>
155
+ char !== undefined && char.match(/\s/) !== null;
156
+
157
+ export const isInt = (n: number): boolean => n % 1 === 0;
158
+
159
+ export const getNumberLength = (n?: number): number =>
160
+ n === undefined || isNaN(n) ? 0 : n.toString().length;
161
+
162
+ /**
163
+ * Проверяет, что `val` не `null`, не `undefined` и не пустая строка
164
+ */
165
+ export const isNotEmpty = <T>(val: T | null | undefined): val is T =>
166
+ typeof val === 'string'
167
+ ? val.trim() !== ''
168
+ : val !== null && val !== undefined;
169
+
170
+ export const trimStringToMaxLength = (val: string, maxLength: number) =>
171
+ val.length > maxLength ? val.slice(0, maxLength) : val;
172
+
173
+ export const addDataAttributes = (
174
+ data: IDataAttributes = {},
175
+ ): IDataAttributes =>
176
+ Object.fromEntries(
177
+ Object.entries(data).map(([key, value]) =>
178
+ isNotEmpty(value) ? [`data-${transformToKebab(key)}`, value] : [],
179
+ ),
180
+ );
181
+
182
+ export const addDataTestId = (
183
+ ...args: Parameters<typeof getTestId>
184
+ ): { 'data-testid': string } | undefined => {
185
+ const testId = getTestId(...args);
186
+ return isNotEmpty(testId) ? { 'data-testid': testId } : undefined;
187
+ };
188
+
189
+ export const getTestId = (
190
+ testId: string | undefined,
191
+ postfix?: string | number,
192
+ ): string | undefined => {
193
+ if (!isNotEmpty(testId)) {
194
+ return undefined;
195
+ }
196
+ return isNotEmpty(postfix) ? `${testId}-${postfix}` : testId;
197
+ };
198
+
199
+ export const getSelectKeyHandler =
200
+ (cb: (e: KeyboardEvent) => void): ((e: KeyboardEvent) => void) =>
201
+ (e) => {
202
+ if (e.code === 'Enter' || e.code === 'NumpadEnter') {
203
+ cb(e);
204
+ }
205
+ };
206
+
207
+ export const addClickHandler = (
208
+ cb?: (e: MouseEvent | KeyboardEvent) => void,
209
+ hasAction = true,
210
+ ): HTMLAttributes<unknown> =>
211
+ hasAction && isNotEmpty(cb)
212
+ ? {
213
+ tabIndex: 0,
214
+ onClick: cb as (e: MouseEvent) => void,
215
+ onKeyDown: getSelectKeyHandler(cb),
216
+ }
217
+ : {
218
+ tabIndex: -1,
219
+ };
@@ -1,6 +1,6 @@
1
- export * from './use-is-mounted';
2
- export * from './use-on-click-outside';
3
- export * from './use-theme';
4
- export * from './use-dropdown';
5
- export * from './use-tweak-styles';
6
- export * from './use-did-mount-effect';
1
+ export * from './use-is-mounted';
2
+ export * from './use-on-click-outside';
3
+ export * from './use-theme';
4
+ export * from './use-dropdown';
5
+ export * from './use-tweak-styles';
6
+ export * from './use-did-mount-effect';
@@ -1,21 +1,21 @@
1
- import { DependencyList, EffectCallback, useEffect, useRef } from 'react';
2
-
3
- export const useDidMountEffect = (
4
- effect: EffectCallback,
5
- dependencies: DependencyList,
6
- ): void => {
7
- const isMountedRef = useRef(false);
8
- useEffect(() => {
9
- let unmount: ReturnType<EffectCallback>;
10
- if (isMountedRef.current) {
11
- unmount = effect();
12
- } else {
13
- isMountedRef.current = true;
14
- }
15
- return () => {
16
- if (unmount !== undefined) {
17
- unmount();
18
- }
19
- };
20
- }, dependencies);
21
- };
1
+ import { DependencyList, EffectCallback, useEffect, useRef } from 'react';
2
+
3
+ export const useDidMountEffect = (
4
+ effect: EffectCallback,
5
+ dependencies: DependencyList,
6
+ ): void => {
7
+ const isMountedRef = useRef(false);
8
+ useEffect(() => {
9
+ let unmount: ReturnType<EffectCallback>;
10
+ if (isMountedRef.current) {
11
+ unmount = effect();
12
+ } else {
13
+ isMountedRef.current = true;
14
+ }
15
+ return () => {
16
+ if (unmount !== undefined) {
17
+ unmount();
18
+ }
19
+ };
20
+ }, dependencies);
21
+ };
@@ -1,85 +1,85 @@
1
- import { DependencyList, useEffect } from 'react';
2
- import usePopper, { VirtualElement } from 'react-overlays/usePopper';
3
-
4
- import { IDropdownWithPopperOptions } from '../types';
5
- import { getScrollParent, minWidthModifier } from '../helpers';
6
-
7
- export const useDropdown = ({
8
- isOpen,
9
- onDropdownClose,
10
- referenceElement,
11
- dropdownElement,
12
- options,
13
- dependenciesForPositionUpdating = [],
14
- }: {
15
- isOpen: boolean;
16
- referenceElement: VirtualElement | null | undefined;
17
- dropdownElement: HTMLElement | null | undefined;
18
- options?: IDropdownWithPopperOptions;
19
- dependenciesForPositionUpdating?: DependencyList;
20
- onDropdownClose(event: Event): void;
21
- }): ReturnType<typeof usePopper> | undefined => {
22
- const {
23
- shouldUsePopper = false,
24
- shouldRenderInBody = false,
25
- shouldHideOnScroll = false,
26
- scrollParent = 'document',
27
- canBeFlipped = false,
28
- modifiers = [],
29
- placement = 'bottom-start',
30
- flipOptions,
31
- } = options ?? {};
32
-
33
- useEffect(() => {
34
- if (!shouldHideOnScroll || !isOpen) {
35
- return;
36
- }
37
-
38
- const scrollParentEl =
39
- scrollParent === 'auto'
40
- ? getScrollParent(referenceElement as Element)
41
- : scrollParent === 'document'
42
- ? document
43
- : scrollParent;
44
- scrollParentEl.addEventListener('scroll', onDropdownClose);
45
-
46
- return () => {
47
- scrollParentEl.removeEventListener('scroll', onDropdownClose);
48
- };
49
- }, [shouldHideOnScroll, isOpen]);
50
-
51
- let popperData: ReturnType<typeof usePopper> | undefined;
52
- if (shouldUsePopper) {
53
- popperData = usePopper(referenceElement, dropdownElement, {
54
- enabled: isOpen,
55
- placement,
56
- modifiers: [
57
- ...(shouldRenderInBody ? [minWidthModifier] : []),
58
- {
59
- name: 'offset',
60
- options: {
61
- offset: [0, 6],
62
- },
63
- },
64
- {
65
- name: 'flip',
66
- options: {
67
- fallbackPlacements: canBeFlipped
68
- ? ['bottom-start', 'top-start']
69
- : ['bottom-start'],
70
- ...flipOptions,
71
- },
72
- },
73
- ...modifiers,
74
- ],
75
- });
76
- }
77
-
78
- useEffect(() => {
79
- if (dependenciesForPositionUpdating.length !== 0) {
80
- popperData?.update();
81
- }
82
- }, dependenciesForPositionUpdating);
83
-
84
- return popperData;
85
- };
1
+ import { DependencyList, useEffect } from 'react';
2
+ import usePopper, { VirtualElement } from 'react-overlays/usePopper';
3
+
4
+ import { IDropdownWithPopperOptions } from '../types';
5
+ import { getScrollParent, minWidthModifier } from '../helpers';
6
+
7
+ export const useDropdown = ({
8
+ isOpen,
9
+ onDropdownClose,
10
+ referenceElement,
11
+ dropdownElement,
12
+ options,
13
+ dependenciesForPositionUpdating = [],
14
+ }: {
15
+ isOpen: boolean;
16
+ referenceElement: VirtualElement | null | undefined;
17
+ dropdownElement: HTMLElement | null | undefined;
18
+ options?: IDropdownWithPopperOptions;
19
+ dependenciesForPositionUpdating?: DependencyList;
20
+ onDropdownClose(event: Event): void;
21
+ }): ReturnType<typeof usePopper> | undefined => {
22
+ const {
23
+ shouldUsePopper = false,
24
+ shouldRenderInBody = false,
25
+ shouldHideOnScroll = false,
26
+ scrollParent = 'document',
27
+ canBeFlipped = false,
28
+ modifiers = [],
29
+ placement = 'bottom-start',
30
+ flipOptions,
31
+ } = options ?? {};
32
+
33
+ useEffect(() => {
34
+ if (!shouldHideOnScroll || !isOpen) {
35
+ return;
36
+ }
37
+
38
+ const scrollParentEl =
39
+ scrollParent === 'auto'
40
+ ? getScrollParent(referenceElement as Element)
41
+ : scrollParent === 'document'
42
+ ? document
43
+ : scrollParent;
44
+ scrollParentEl.addEventListener('scroll', onDropdownClose);
45
+
46
+ return () => {
47
+ scrollParentEl.removeEventListener('scroll', onDropdownClose);
48
+ };
49
+ }, [shouldHideOnScroll, isOpen]);
50
+
51
+ let popperData: ReturnType<typeof usePopper> | undefined;
52
+ if (shouldUsePopper) {
53
+ popperData = usePopper(referenceElement, dropdownElement, {
54
+ enabled: isOpen,
55
+ placement,
56
+ modifiers: [
57
+ ...(shouldRenderInBody ? [minWidthModifier] : []),
58
+ {
59
+ name: 'offset',
60
+ options: {
61
+ offset: [0, 6],
62
+ },
63
+ },
64
+ {
65
+ name: 'flip',
66
+ options: {
67
+ fallbackPlacements: canBeFlipped
68
+ ? ['bottom-start', 'top-start']
69
+ : ['bottom-start'],
70
+ ...flipOptions,
71
+ },
72
+ },
73
+ ...modifiers,
74
+ ],
75
+ });
76
+ }
77
+
78
+ useEffect(() => {
79
+ if (dependenciesForPositionUpdating.length !== 0) {
80
+ popperData?.update();
81
+ }
82
+ }, dependenciesForPositionUpdating);
83
+
84
+ return popperData;
85
+ };