@true-engineering/true-react-common-ui-kit 1.5.2 → 1.6.0

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