@true-engineering/true-react-common-ui-kit 1.6.0 → 1.8.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 (239) 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 +2 -2
  4. package/dist/components/Icon/complexIcons/augment.d.ts +1 -1
  5. package/dist/components/Select/Select.d.ts +1 -1
  6. package/dist/true-react-common-ui-kit.js +64 -64
  7. package/dist/true-react-common-ui-kit.js.map +1 -1
  8. package/dist/true-react-common-ui-kit.umd.cjs +64 -64
  9. package/dist/true-react-common-ui-kit.umd.cjs.map +1 -1
  10. package/dist/vite-env.d.ts +1 -1
  11. package/package.json +91 -91
  12. package/src/components/AccountInfo/AccountInfo.stories.tsx +35 -35
  13. package/src/components/AccountInfo/AccountInfo.styles.ts +55 -55
  14. package/src/components/AccountInfo/AccountInfo.tsx +106 -106
  15. package/src/components/AccountInfo/index.ts +2 -2
  16. package/src/components/AddButton/AddButton.stories.tsx +21 -21
  17. package/src/components/AddButton/AddButton.styles.ts +34 -34
  18. package/src/components/AddButton/AddButton.tsx +49 -49
  19. package/src/components/AddButton/index.ts +2 -2
  20. package/src/components/Button/Button.stories.tsx +61 -61
  21. package/src/components/Button/Button.styles.ts +196 -196
  22. package/src/components/Button/Button.tsx +195 -195
  23. package/src/components/Button/index.ts +2 -2
  24. package/src/components/Checkbox/Checkbox.stories.tsx +35 -35
  25. package/src/components/Checkbox/Checkbox.styles.ts +62 -62
  26. package/src/components/Checkbox/Checkbox.tsx +106 -106
  27. package/src/components/Checkbox/index.ts +2 -2
  28. package/src/components/CloseButton/CloseButton.styles.ts +34 -34
  29. package/src/components/CloseButton/CloseButton.tsx +37 -37
  30. package/src/components/CloseButton/index.ts +2 -2
  31. package/src/components/Colors/Colors.stories.tsx +7 -7
  32. package/src/components/Colors/Colors.styles.ts +38 -38
  33. package/src/components/Colors/Colors.tsx +34 -34
  34. package/src/components/Colors/index.ts +2 -2
  35. package/src/components/CssBaseline/CssBaseline.styles.ts +15 -15
  36. package/src/components/CssBaseline/CssBaseline.tsx +17 -17
  37. package/src/components/CssBaseline/index.ts +2 -2
  38. package/src/components/DateInput/DateInput.stories.tsx +63 -63
  39. package/src/components/DateInput/DateInput.styles.ts +14 -14
  40. package/src/components/DateInput/DateInput.tsx +60 -60
  41. package/src/components/DateInput/index.ts +2 -2
  42. package/src/components/DatePicker/DatePicker.stories.tsx +96 -96
  43. package/src/components/DatePicker/DatePicker.styles.ts +54 -54
  44. package/src/components/DatePicker/DatePicker.tsx +358 -358
  45. package/src/components/DatePicker/DatePickerHeader/DatePickerHeader.styles.ts +84 -84
  46. package/src/components/DatePicker/DatePickerHeader/DatePickerHeader.tsx +94 -94
  47. package/src/components/DatePicker/DatePickerHeader/index.ts +1 -1
  48. package/src/components/DatePicker/DatePickerInput/DatePickerInput.styles.ts +25 -25
  49. package/src/components/DatePicker/DatePickerInput/DatePickerInput.tsx +31 -31
  50. package/src/components/DatePicker/DatePickerInput/index.ts +1 -1
  51. package/src/components/DatePicker/index.ts +4 -4
  52. package/src/components/Description/Description.stories.tsx +29 -29
  53. package/src/components/Description/Description.styles.ts +31 -31
  54. package/src/components/Description/Description.tsx +69 -69
  55. package/src/components/Description/index.ts +2 -2
  56. package/src/components/FiltersPane/FilterInterval/FilterInterval.styles.ts +64 -64
  57. package/src/components/FiltersPane/FilterInterval/FilterInterval.tsx +162 -162
  58. package/src/components/FiltersPane/FilterInterval/index.ts +1 -1
  59. package/src/components/FiltersPane/FilterMultiSelect/FilterMultiSelect.tsx +14 -14
  60. package/src/components/FiltersPane/FilterMultiSelect/index.ts +1 -1
  61. package/src/components/FiltersPane/FilterSelect/FilterSelect.styles.ts +144 -144
  62. package/src/components/FiltersPane/FilterSelect/FilterSelect.tsx +397 -397
  63. package/src/components/FiltersPane/FilterSelect/index.ts +1 -1
  64. package/src/components/FiltersPane/FilterSelect/locales.ts +37 -37
  65. package/src/components/FiltersPane/FilterValueView/FilterValueView.styles.tsx +15 -15
  66. package/src/components/FiltersPane/FilterValueView/FilterValueView.tsx +186 -186
  67. package/src/components/FiltersPane/FilterValueView/index.tsx +1 -1
  68. package/src/components/FiltersPane/FilterWithDates/FilterWithDates.styles.ts +60 -60
  69. package/src/components/FiltersPane/FilterWithDates/FilterWithDates.tsx +222 -222
  70. package/src/components/FiltersPane/FilterWithDates/index.ts +1 -1
  71. package/src/components/FiltersPane/FilterWithPeriod/FilterWithPeriod.styles.ts +17 -17
  72. package/src/components/FiltersPane/FilterWithPeriod/FilterWithPeriod.tsx +231 -231
  73. package/src/components/FiltersPane/FilterWithPeriod/index.ts +1 -1
  74. package/src/components/FiltersPane/FilterWrapper/FilterWrapper.styles.ts +110 -110
  75. package/src/components/FiltersPane/FilterWrapper/FilterWrapper.tsx +360 -360
  76. package/src/components/FiltersPane/FilterWrapper/index.ts +1 -1
  77. package/src/components/FiltersPane/FiltersPane.stories.tsx +308 -308
  78. package/src/components/FiltersPane/FiltersPane.styles.ts +71 -71
  79. package/src/components/FiltersPane/FiltersPane.tsx +193 -193
  80. package/src/components/FiltersPane/FiltersPaneSearch/FiltersPaneSearch.styles.ts +109 -109
  81. package/src/components/FiltersPane/FiltersPaneSearch/FiltersPaneSearch.tsx +175 -175
  82. package/src/components/FiltersPane/FiltersPaneSearch/index.ts +1 -1
  83. package/src/components/FiltersPane/index.ts +20 -20
  84. package/src/components/FiltersPane/locales.ts +107 -107
  85. package/src/components/FiltersPane/types.ts +126 -126
  86. package/src/components/Flag/Flag.stories.tsx +29 -29
  87. package/src/components/Flag/Flag.styles.ts +18 -18
  88. package/src/components/Flag/Flag.tsx +28 -28
  89. package/src/components/Flag/augment.d.ts +1 -1
  90. package/src/components/Flag/index.ts +2 -2
  91. package/src/components/FlexibleTable/FlexibleTable.stories.tsx +86 -80
  92. package/src/components/FlexibleTable/FlexibleTable.styles.ts +131 -131
  93. package/src/components/FlexibleTable/FlexibleTable.tsx +243 -243
  94. package/src/components/FlexibleTable/TableRow.tsx +171 -171
  95. package/src/components/FlexibleTable/TableValue.tsx +83 -83
  96. package/src/components/FlexibleTable/fixture-test.ts +254 -254
  97. package/src/components/FlexibleTable/index.ts +3 -3
  98. package/src/components/FlexibleTable/types.ts +58 -58
  99. package/src/components/Icon/ComplexIconBoilerplate.tsx +17 -17
  100. package/src/components/Icon/Icon.stories.tsx +88 -88
  101. package/src/components/Icon/Icon.styles.ts +10 -10
  102. package/src/components/Icon/Icon.tsx +34 -34
  103. package/src/components/Icon/IconBoilerplate.tsx +42 -42
  104. package/src/components/Icon/complexIcons/augment.d.ts +1 -1
  105. package/src/components/Icon/complexIcons/avatarGreen.svg +57 -57
  106. package/src/components/Icon/complexIcons/icons.ts +7 -7
  107. package/src/components/Icon/complexIcons/index.ts +1 -1
  108. package/src/components/Icon/icons/icons.ts +838 -838
  109. package/src/components/Icon/icons/index.ts +1 -1
  110. package/src/components/Icon/index.ts +4 -4
  111. package/src/components/IncrementInput/ChangeButton.tsx +34 -34
  112. package/src/components/IncrementInput/IncrementInput.stories.tsx +34 -34
  113. package/src/components/IncrementInput/IncrementInput.styles.ts +77 -77
  114. package/src/components/IncrementInput/IncrementInput.tsx +95 -95
  115. package/src/components/IncrementInput/index.ts +2 -2
  116. package/src/components/Input/Input.stories.tsx +92 -92
  117. package/src/components/Input/Input.styles.ts +305 -305
  118. package/src/components/Input/Input.tsx +318 -318
  119. package/src/components/Input/index.ts +2 -2
  120. package/src/components/List/List.stories.tsx +62 -62
  121. package/src/components/List/List.styles.ts +52 -52
  122. package/src/components/List/List.tsx +82 -82
  123. package/src/components/List/index.ts +2 -2
  124. package/src/components/Modal/Modal.stories.tsx +113 -113
  125. package/src/components/Modal/Modal.styles.ts +308 -308
  126. package/src/components/Modal/Modal.tsx +210 -210
  127. package/src/components/Modal/index.ts +2 -2
  128. package/src/components/MoreMenu/MoreMenu.stories.tsx +46 -46
  129. package/src/components/MoreMenu/MoreMenu.styles.ts +70 -70
  130. package/src/components/MoreMenu/MoreMenu.tsx +102 -102
  131. package/src/components/MoreMenu/index.ts +2 -2
  132. package/src/components/MultiSelect/MultiSelect.stories.tsx +46 -46
  133. package/src/components/MultiSelect/MultiSelect.styles.ts +55 -55
  134. package/src/components/MultiSelect/MultiSelect.tsx +98 -98
  135. package/src/components/MultiSelect/MultiSelectInput/MultiSelectInput.styles.ts +73 -73
  136. package/src/components/MultiSelect/MultiSelectInput/MultiSelectInput.tsx +62 -62
  137. package/src/components/MultiSelect/MultiSelectInput/index.ts +1 -1
  138. package/src/components/MultiSelect/index.ts +3 -3
  139. package/src/components/MultiSelectList/MultiSelectList.styles.ts +125 -125
  140. package/src/components/MultiSelectList/MultiSelectList.tsx +519 -519
  141. package/src/components/MultiSelectList/index.ts +2 -2
  142. package/src/components/MultiSelectList/locales.ts +37 -37
  143. package/src/components/Notification/Notification.stories.tsx +51 -51
  144. package/src/components/Notification/Notification.styles.ts +50 -50
  145. package/src/components/Notification/Notification.tsx +84 -84
  146. package/src/components/Notification/index.ts +2 -2
  147. package/src/components/NumberInput/NumberInput.stories.tsx +36 -36
  148. package/src/components/NumberInput/NumberInput.tsx +154 -154
  149. package/src/components/NumberInput/helpers.ts +87 -87
  150. package/src/components/NumberInput/index.ts +1 -1
  151. package/src/components/PhoneInput/PhoneInput.stories.tsx +71 -71
  152. package/src/components/PhoneInput/PhoneInput.styles.ts +84 -84
  153. package/src/components/PhoneInput/PhoneInput.tsx +223 -223
  154. package/src/components/PhoneInput/PhoneInputCountryList/PhoneInputCountryList.stories.tsx +21 -21
  155. package/src/components/PhoneInput/PhoneInputCountryList/PhoneInputCountryList.styles.ts +100 -100
  156. package/src/components/PhoneInput/PhoneInputCountryList/PhoneInputCountryList.tsx +171 -171
  157. package/src/components/PhoneInput/PhoneInputCountryList/index.ts +2 -2
  158. package/src/components/PhoneInput/index.ts +6 -6
  159. package/src/components/PhoneInput/phone-info.ts +2167 -2167
  160. package/src/components/PhoneInput/types.ts +16 -16
  161. package/src/components/RadioButton/RadioButton.stories.tsx +46 -46
  162. package/src/components/RadioButton/RadioButton.styles.ts +37 -37
  163. package/src/components/RadioButton/RadioButton.tsx +56 -56
  164. package/src/components/RadioButton/index.ts +2 -2
  165. package/src/components/ScrollIntoViewIfNeeded/ScrollIntoViewIfNeeded.ts +66 -66
  166. package/src/components/ScrollIntoViewIfNeeded/index.ts +1 -1
  167. package/src/components/SearchInput/SearchInput.stories.tsx +24 -24
  168. package/src/components/SearchInput/SearchInput.styles.ts +50 -50
  169. package/src/components/SearchInput/SearchInput.tsx +63 -63
  170. package/src/components/SearchInput/index.ts +2 -2
  171. package/src/components/Select/Select.stories.tsx +258 -258
  172. package/src/components/Select/Select.styles.ts +85 -85
  173. package/src/components/Select/Select.tsx +510 -514
  174. package/src/components/Select/SelectList/SelectList.styles.ts +68 -68
  175. package/src/components/Select/SelectList/SelectList.tsx +139 -139
  176. package/src/components/Select/SelectList/index.ts +1 -1
  177. package/src/components/Select/helpers.ts +21 -21
  178. package/src/components/Select/index.ts +3 -3
  179. package/src/components/SmartInput/SmartInput.stories.tsx +63 -63
  180. package/src/components/SmartInput/SmartInput.tsx +180 -180
  181. package/src/components/SmartInput/helpers.ts +85 -85
  182. package/src/components/SmartInput/index.ts +1 -1
  183. package/src/components/Switch/Switch.stories.tsx +40 -40
  184. package/src/components/Switch/Switch.styles.ts +75 -75
  185. package/src/components/Switch/Switch.tsx +89 -89
  186. package/src/components/Switch/index.ts +2 -2
  187. package/src/components/TextArea/TextArea.stories.tsx +35 -35
  188. package/src/components/TextArea/TextArea.styles.ts +153 -153
  189. package/src/components/TextArea/TextArea.tsx +178 -178
  190. package/src/components/TextArea/index.ts +2 -2
  191. package/src/components/TextWithInfo/TextWithInfo.stories.tsx +53 -53
  192. package/src/components/TextWithInfo/TextWithInfo.styles.ts +60 -60
  193. package/src/components/TextWithInfo/TextWithInfo.tsx +67 -67
  194. package/src/components/TextWithInfo/index.ts +2 -2
  195. package/src/components/TextWithTooltip/TextWithTooltip.stories.tsx +58 -58
  196. package/src/components/TextWithTooltip/TextWithTooltip.styles.ts +19 -19
  197. package/src/components/TextWithTooltip/TextWithTooltip.tsx +163 -163
  198. package/src/components/TextWithTooltip/index.ts +2 -2
  199. package/src/components/ThemedPreloader/ThemedPreloader.stories.tsx +41 -41
  200. package/src/components/ThemedPreloader/ThemedPreloader.styles.ts +21 -21
  201. package/src/components/ThemedPreloader/ThemedPreloader.tsx +56 -56
  202. package/src/components/ThemedPreloader/components/DefaultPreloader/DefaultPreloader.tsx +34 -34
  203. package/src/components/ThemedPreloader/components/DefaultPreloader/index.ts +1 -1
  204. package/src/components/ThemedPreloader/components/DotsPreloader/DotsPreloader.styles.ts +54 -54
  205. package/src/components/ThemedPreloader/components/DotsPreloader/DotsPreloader.tsx +18 -18
  206. package/src/components/ThemedPreloader/components/DotsPreloader/index.ts +2 -2
  207. package/src/components/ThemedPreloader/components/SvgPreloader/SvgPreloader.styles.ts +11 -11
  208. package/src/components/ThemedPreloader/components/SvgPreloader/SvgPreloader.tsx +32 -32
  209. package/src/components/ThemedPreloader/components/SvgPreloader/index.ts +2 -2
  210. package/src/components/ThemedPreloader/components/index.ts +2 -2
  211. package/src/components/ThemedPreloader/index.ts +2 -2
  212. package/src/components/Toaster/Toaster.stories.tsx +34 -34
  213. package/src/components/Toaster/Toaster.styles.ts +59 -59
  214. package/src/components/Toaster/Toaster.tsx +113 -113
  215. package/src/components/Toaster/index.ts +2 -2
  216. package/src/components/Tooltip/Tooltip.stories.tsx +21 -21
  217. package/src/components/Tooltip/Tooltip.styles.ts +45 -45
  218. package/src/components/Tooltip/Tooltip.tsx +40 -40
  219. package/src/components/Tooltip/index.ts +3 -3
  220. package/src/components/Tooltip/types.ts +1 -1
  221. package/src/components/index.ts +36 -36
  222. package/src/helpers/colors.ts +2 -2
  223. package/src/helpers/dateHelpers/date-helpers.ts +9 -9
  224. package/src/helpers/index.ts +4 -4
  225. package/src/helpers/phone.ts +106 -106
  226. package/src/helpers/popper-helpers.ts +17 -17
  227. package/src/helpers/snippets.tsx +5 -5
  228. package/src/helpers/utils.ts +219 -219
  229. package/src/hooks/index.ts +6 -6
  230. package/src/hooks/use-did-mount-effect.ts +21 -21
  231. package/src/hooks/use-dropdown.ts +85 -85
  232. package/src/hooks/use-is-mounted.ts +15 -15
  233. package/src/hooks/use-on-click-outside.ts +92 -92
  234. package/src/hooks/use-theme.ts +36 -36
  235. package/src/hooks/use-tweak-styles.ts +14 -14
  236. package/src/index.ts +6 -6
  237. package/src/theme.ts +155 -155
  238. package/src/types.ts +106 -106
  239. 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
+ };