@true-engineering/true-react-common-ui-kit 2.1.1 → 2.3.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 (258) hide show
  1. package/LICENSE +201 -201
  2. package/README.md +16 -0
  3. package/dist/components/Button/Button.d.ts +3 -4
  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 +3 -4
  7. package/dist/components/List/List.d.ts +3 -4
  8. package/dist/components/Modal/Modal.d.ts +3 -3
  9. package/dist/components/ScrollIntoViewIfNeeded/ScrollIntoViewIfNeeded.d.ts +4 -4
  10. package/dist/components/Select/Select.d.ts +2 -3
  11. package/dist/helpers/deprecated.d.ts +12 -0
  12. package/dist/helpers/index.d.ts +2 -2
  13. package/dist/helpers/misc.d.ts +19 -0
  14. package/dist/helpers/snippets.d.ts +3 -3
  15. package/dist/true-react-common-ui-kit.js +389 -405
  16. package/dist/true-react-common-ui-kit.js.map +1 -1
  17. package/dist/true-react-common-ui-kit.umd.cjs +389 -406
  18. package/dist/true-react-common-ui-kit.umd.cjs.map +1 -1
  19. package/dist/types.d.ts +3 -1
  20. package/dist/vite-env.d.ts +1 -1
  21. package/package.json +93 -92
  22. package/src/components/AccountInfo/AccountInfo.stories.tsx +32 -32
  23. package/src/components/AccountInfo/AccountInfo.styles.ts +55 -55
  24. package/src/components/AccountInfo/AccountInfo.tsx +77 -77
  25. package/src/components/AccountInfo/index.ts +2 -2
  26. package/src/components/AddButton/AddButton.stories.tsx +21 -21
  27. package/src/components/AddButton/AddButton.styles.ts +34 -34
  28. package/src/components/AddButton/AddButton.tsx +48 -48
  29. package/src/components/AddButton/index.ts +2 -2
  30. package/src/components/Button/Button.stories.tsx +52 -52
  31. package/src/components/Button/Button.styles.ts +196 -196
  32. package/src/components/Button/Button.tsx +167 -187
  33. package/src/components/Button/index.ts +2 -2
  34. package/src/components/Checkbox/Checkbox.stories.tsx +32 -32
  35. package/src/components/Checkbox/Checkbox.styles.ts +62 -62
  36. package/src/components/Checkbox/Checkbox.tsx +105 -100
  37. package/src/components/Checkbox/index.ts +2 -2
  38. package/src/components/CloseButton/CloseButton.styles.ts +34 -34
  39. package/src/components/CloseButton/CloseButton.tsx +35 -35
  40. package/src/components/CloseButton/index.ts +2 -2
  41. package/src/components/Colors/Colors.stories.tsx +7 -7
  42. package/src/components/Colors/Colors.styles.ts +38 -38
  43. package/src/components/Colors/Colors.tsx +26 -26
  44. package/src/components/Colors/index.ts +2 -2
  45. package/src/components/CssBaseline/CssBaseline.styles.ts +15 -15
  46. package/src/components/CssBaseline/CssBaseline.tsx +15 -15
  47. package/src/components/CssBaseline/index.ts +2 -2
  48. package/src/components/DateInput/DateInput.stories.tsx +61 -61
  49. package/src/components/DateInput/DateInput.styles.ts +14 -14
  50. package/src/components/DateInput/DateInput.tsx +82 -82
  51. package/src/components/DateInput/constants.ts +2 -2
  52. package/src/components/DateInput/index.ts +3 -3
  53. package/src/components/DatePicker/DatePicker.stories.tsx +87 -87
  54. package/src/components/DatePicker/DatePicker.styles.ts +44 -44
  55. package/src/components/DatePicker/DatePicker.tsx +327 -322
  56. package/src/components/DatePicker/DatePickerHeader/DatePickerHeader.styles.ts +84 -84
  57. package/src/components/DatePicker/DatePickerHeader/DatePickerHeader.tsx +79 -79
  58. package/src/components/DatePicker/DatePickerHeader/index.ts +2 -2
  59. package/src/components/DatePicker/constants.ts +1 -1
  60. package/src/components/DatePicker/helpers.ts +23 -19
  61. package/src/components/DatePicker/index.ts +3 -3
  62. package/src/components/DatePicker/types.ts +45 -45
  63. package/src/components/Description/Description.stories.tsx +27 -27
  64. package/src/components/Description/Description.styles.ts +31 -31
  65. package/src/components/Description/Description.tsx +60 -60
  66. package/src/components/Description/index.ts +2 -2
  67. package/src/components/FiltersPane/FilterInterval/FilterInterval.styles.ts +64 -64
  68. package/src/components/FiltersPane/FilterInterval/FilterInterval.tsx +143 -143
  69. package/src/components/FiltersPane/FilterInterval/index.ts +1 -1
  70. package/src/components/FiltersPane/FilterMultiSelect/FilterMultiSelect.tsx +10 -10
  71. package/src/components/FiltersPane/FilterMultiSelect/index.ts +1 -1
  72. package/src/components/FiltersPane/FilterSelect/FilterSelect.styles.ts +143 -143
  73. package/src/components/FiltersPane/FilterSelect/FilterSelect.tsx +348 -347
  74. package/src/components/FiltersPane/FilterSelect/index.ts +1 -1
  75. package/src/components/FiltersPane/FilterSelect/locales.ts +37 -37
  76. package/src/components/FiltersPane/FilterValueView/FilterValueView.styles.tsx +15 -15
  77. package/src/components/FiltersPane/FilterValueView/FilterValueView.tsx +166 -166
  78. package/src/components/FiltersPane/FilterValueView/index.tsx +1 -1
  79. package/src/components/FiltersPane/FilterWithDates/FilterWithDates.styles.ts +60 -60
  80. package/src/components/FiltersPane/FilterWithDates/FilterWithDates.tsx +184 -183
  81. package/src/components/FiltersPane/FilterWithDates/index.ts +1 -1
  82. package/src/components/FiltersPane/FilterWithPeriod/FilterWithPeriod.styles.ts +17 -17
  83. package/src/components/FiltersPane/FilterWithPeriod/FilterWithPeriod.tsx +198 -198
  84. package/src/components/FiltersPane/FilterWithPeriod/index.ts +1 -1
  85. package/src/components/FiltersPane/FilterWrapper/FilterWrapper.styles.ts +110 -110
  86. package/src/components/FiltersPane/FilterWrapper/FilterWrapper.tsx +346 -346
  87. package/src/components/FiltersPane/FilterWrapper/index.ts +1 -1
  88. package/src/components/FiltersPane/FiltersPane.stories.tsx +295 -295
  89. package/src/components/FiltersPane/FiltersPane.styles.ts +71 -71
  90. package/src/components/FiltersPane/FiltersPane.tsx +151 -151
  91. package/src/components/FiltersPane/FiltersPaneSearch/FiltersPaneSearch.styles.ts +109 -109
  92. package/src/components/FiltersPane/FiltersPaneSearch/FiltersPaneSearch.tsx +155 -155
  93. package/src/components/FiltersPane/FiltersPaneSearch/index.ts +1 -1
  94. package/src/components/FiltersPane/index.ts +20 -20
  95. package/src/components/FiltersPane/locales.ts +107 -107
  96. package/src/components/FiltersPane/types.ts +112 -112
  97. package/src/components/Flag/Flag.stories.tsx +29 -29
  98. package/src/components/Flag/Flag.styles.ts +18 -18
  99. package/src/components/Flag/Flag.tsx +27 -27
  100. package/src/components/Flag/augment.d.ts +1 -1
  101. package/src/components/Flag/index.ts +2 -2
  102. package/src/components/FlexibleTable/FlexibleTable.stories.tsx +84 -84
  103. package/src/components/FlexibleTable/FlexibleTable.styles.ts +131 -131
  104. package/src/components/FlexibleTable/FlexibleTable.tsx +205 -205
  105. package/src/components/FlexibleTable/TableRow.tsx +152 -152
  106. package/src/components/FlexibleTable/TableValue.tsx +75 -75
  107. package/src/components/FlexibleTable/fixture-test.ts +254 -254
  108. package/src/components/FlexibleTable/index.ts +3 -3
  109. package/src/components/FlexibleTable/types.ts +52 -52
  110. package/src/components/Icon/ComplexIconBoilerplate.tsx +17 -17
  111. package/src/components/Icon/Icon.stories.tsx +85 -85
  112. package/src/components/Icon/Icon.styles.ts +10 -10
  113. package/src/components/Icon/Icon.tsx +32 -32
  114. package/src/components/Icon/IconBoilerplate.tsx +42 -42
  115. package/src/components/Icon/complexIcons/augment.d.ts +1 -1
  116. package/src/components/Icon/complexIcons/avatarGreen.svg +57 -57
  117. package/src/components/Icon/complexIcons/icons.ts +7 -7
  118. package/src/components/Icon/complexIcons/index.ts +1 -1
  119. package/src/components/Icon/icons/icons.ts +838 -838
  120. package/src/components/Icon/icons/index.ts +1 -1
  121. package/src/components/Icon/index.ts +4 -4
  122. package/src/components/IncrementInput/ChangeButton.tsx +33 -33
  123. package/src/components/IncrementInput/IncrementInput.stories.tsx +31 -31
  124. package/src/components/IncrementInput/IncrementInput.styles.ts +77 -77
  125. package/src/components/IncrementInput/IncrementInput.tsx +80 -79
  126. package/src/components/IncrementInput/index.ts +2 -2
  127. package/src/components/Input/Input.stories.tsx +86 -86
  128. package/src/components/Input/Input.styles.ts +307 -307
  129. package/src/components/Input/Input.tsx +311 -306
  130. package/src/components/Input/index.ts +2 -2
  131. package/src/components/List/List.stories.tsx +62 -62
  132. package/src/components/List/List.styles.ts +52 -52
  133. package/src/components/List/List.tsx +64 -61
  134. package/src/components/List/index.ts +2 -2
  135. package/src/components/Modal/Modal.stories.tsx +105 -105
  136. package/src/components/Modal/Modal.styles.ts +305 -305
  137. package/src/components/Modal/Modal.tsx +200 -181
  138. package/src/components/Modal/index.ts +2 -2
  139. package/src/components/MoreMenu/MoreMenu.stories.tsx +46 -46
  140. package/src/components/MoreMenu/MoreMenu.styles.ts +70 -70
  141. package/src/components/MoreMenu/MoreMenu.tsx +90 -89
  142. package/src/components/MoreMenu/index.ts +2 -2
  143. package/src/components/MultiSelect/MultiSelect.stories.tsx +46 -46
  144. package/src/components/MultiSelect/MultiSelect.styles.ts +55 -55
  145. package/src/components/MultiSelect/MultiSelect.tsx +89 -89
  146. package/src/components/MultiSelect/MultiSelectInput/MultiSelectInput.styles.ts +73 -73
  147. package/src/components/MultiSelect/MultiSelectInput/MultiSelectInput.tsx +51 -51
  148. package/src/components/MultiSelect/MultiSelectInput/index.ts +1 -1
  149. package/src/components/MultiSelect/index.ts +3 -3
  150. package/src/components/MultiSelectList/MultiSelectList.styles.ts +124 -124
  151. package/src/components/MultiSelectList/MultiSelectList.tsx +451 -451
  152. package/src/components/MultiSelectList/index.ts +2 -2
  153. package/src/components/MultiSelectList/locales.ts +37 -37
  154. package/src/components/Notification/Notification.stories.tsx +46 -46
  155. package/src/components/Notification/Notification.styles.ts +50 -50
  156. package/src/components/Notification/Notification.tsx +79 -75
  157. package/src/components/Notification/index.ts +2 -2
  158. package/src/components/NumberInput/NumberInput.stories.tsx +35 -35
  159. package/src/components/NumberInput/NumberInput.tsx +133 -133
  160. package/src/components/NumberInput/helpers.ts +86 -85
  161. package/src/components/NumberInput/index.ts +1 -1
  162. package/src/components/PhoneInput/PhoneInput.stories.tsx +70 -70
  163. package/src/components/PhoneInput/PhoneInput.styles.ts +84 -84
  164. package/src/components/PhoneInput/PhoneInput.tsx +194 -194
  165. package/src/components/PhoneInput/PhoneInputCountryList/PhoneInputCountryList.stories.tsx +21 -21
  166. package/src/components/PhoneInput/PhoneInputCountryList/PhoneInputCountryList.styles.ts +100 -100
  167. package/src/components/PhoneInput/PhoneInputCountryList/PhoneInputCountryList.tsx +147 -147
  168. package/src/components/PhoneInput/PhoneInputCountryList/index.ts +2 -2
  169. package/src/components/PhoneInput/index.ts +6 -6
  170. package/src/components/PhoneInput/phone-info.ts +2147 -2147
  171. package/src/components/PhoneInput/types.ts +16 -16
  172. package/src/components/RadioButton/RadioButton.stories.tsx +46 -46
  173. package/src/components/RadioButton/RadioButton.styles.ts +37 -37
  174. package/src/components/RadioButton/RadioButton.tsx +55 -54
  175. package/src/components/RadioButton/index.ts +2 -2
  176. package/src/components/ScrollIntoViewIfNeeded/ScrollIntoViewIfNeeded.ts +61 -61
  177. package/src/components/ScrollIntoViewIfNeeded/index.ts +1 -1
  178. package/src/components/SearchInput/SearchInput.stories.tsx +23 -23
  179. package/src/components/SearchInput/SearchInput.styles.ts +50 -50
  180. package/src/components/SearchInput/SearchInput.tsx +50 -49
  181. package/src/components/SearchInput/index.ts +2 -2
  182. package/src/components/Select/MultiSelect.stories.tsx +240 -240
  183. package/src/components/Select/Select.stories.tsx +235 -235
  184. package/src/components/Select/Select.styles.ts +96 -96
  185. package/src/components/Select/Select.tsx +575 -575
  186. package/src/components/Select/SelectList/SelectList.styles.ts +72 -72
  187. package/src/components/Select/SelectList/SelectList.tsx +158 -152
  188. package/src/components/Select/SelectList/index.ts +1 -1
  189. package/src/components/Select/SelectListItem/SelectListItem.styles.ts +14 -14
  190. package/src/components/Select/SelectListItem/SelectListItem.tsx +68 -67
  191. package/src/components/Select/constants.ts +2 -2
  192. package/src/components/Select/helpers.ts +26 -26
  193. package/src/components/Select/index.ts +4 -4
  194. package/src/components/Select/types.ts +1 -1
  195. package/src/components/SmartInput/SmartInput.stories.tsx +50 -50
  196. package/src/components/SmartInput/SmartInput.tsx +147 -147
  197. package/src/components/SmartInput/helpers.ts +85 -85
  198. package/src/components/SmartInput/index.ts +1 -1
  199. package/src/components/Switch/Switch.stories.tsx +40 -40
  200. package/src/components/Switch/Switch.styles.ts +75 -75
  201. package/src/components/Switch/Switch.tsx +83 -82
  202. package/src/components/Switch/index.ts +2 -2
  203. package/src/components/TextArea/TextArea.stories.tsx +35 -35
  204. package/src/components/TextArea/TextArea.styles.ts +153 -153
  205. package/src/components/TextArea/TextArea.tsx +165 -165
  206. package/src/components/TextArea/index.ts +2 -2
  207. package/src/components/TextWithInfo/TextWithInfo.stories.tsx +53 -53
  208. package/src/components/TextWithInfo/TextWithInfo.styles.ts +60 -60
  209. package/src/components/TextWithInfo/TextWithInfo.tsx +60 -60
  210. package/src/components/TextWithInfo/index.ts +2 -2
  211. package/src/components/TextWithTooltip/TextWithTooltip.stories.tsx +58 -58
  212. package/src/components/TextWithTooltip/TextWithTooltip.styles.ts +19 -19
  213. package/src/components/TextWithTooltip/TextWithTooltip.tsx +143 -143
  214. package/src/components/TextWithTooltip/index.ts +2 -2
  215. package/src/components/ThemedPreloader/ThemedPreloader.stories.tsx +41 -41
  216. package/src/components/ThemedPreloader/ThemedPreloader.styles.ts +21 -21
  217. package/src/components/ThemedPreloader/ThemedPreloader.tsx +50 -50
  218. package/src/components/ThemedPreloader/components/DefaultPreloader/DefaultPreloader.tsx +29 -29
  219. package/src/components/ThemedPreloader/components/DefaultPreloader/index.ts +1 -1
  220. package/src/components/ThemedPreloader/components/DotsPreloader/DotsPreloader.styles.ts +54 -54
  221. package/src/components/ThemedPreloader/components/DotsPreloader/DotsPreloader.tsx +18 -18
  222. package/src/components/ThemedPreloader/components/DotsPreloader/index.ts +2 -2
  223. package/src/components/ThemedPreloader/components/SvgPreloader/SvgPreloader.styles.ts +11 -11
  224. package/src/components/ThemedPreloader/components/SvgPreloader/SvgPreloader.tsx +25 -25
  225. package/src/components/ThemedPreloader/components/SvgPreloader/index.ts +2 -2
  226. package/src/components/ThemedPreloader/components/index.ts +2 -2
  227. package/src/components/ThemedPreloader/index.ts +2 -2
  228. package/src/components/Toaster/Toaster.stories.tsx +30 -30
  229. package/src/components/Toaster/Toaster.styles.ts +59 -59
  230. package/src/components/Toaster/Toaster.tsx +110 -105
  231. package/src/components/Toaster/index.ts +2 -2
  232. package/src/components/Tooltip/Tooltip.stories.tsx +19 -19
  233. package/src/components/Tooltip/Tooltip.styles.ts +45 -45
  234. package/src/components/Tooltip/Tooltip.tsx +35 -35
  235. package/src/components/Tooltip/index.ts +3 -3
  236. package/src/components/Tooltip/types.ts +1 -1
  237. package/src/components/index.ts +36 -36
  238. package/src/helpers/deprecated.ts +22 -0
  239. package/src/helpers/index.ts +4 -4
  240. package/src/helpers/{utils.ts → misc.ts} +158 -231
  241. package/src/helpers/phone.ts +87 -87
  242. package/src/helpers/popper-helpers.ts +17 -17
  243. package/src/helpers/snippets.tsx +6 -5
  244. package/src/hooks/index.ts +6 -6
  245. package/src/hooks/use-did-mount-effect.ts +18 -18
  246. package/src/hooks/use-dropdown.ts +82 -82
  247. package/src/hooks/use-is-mounted.ts +15 -15
  248. package/src/hooks/use-on-click-outside.ts +77 -77
  249. package/src/hooks/use-theme.ts +32 -32
  250. package/src/hooks/use-tweak-styles.ts +13 -13
  251. package/src/index.ts +6 -6
  252. package/src/theme.ts +149 -149
  253. package/src/types.ts +107 -98
  254. package/src/vite-env.d.ts +1 -1
  255. package/dist/helpers/colors.d.ts +0 -2
  256. package/dist/helpers/utils.d.ts +0 -43
  257. package/src/helpers/colors.ts +0 -3
  258. package/src/helpers/dateHelpers/date-helpers.ts +0 -9
@@ -1,231 +1,158 @@
1
- import { HTMLAttributes, KeyboardEvent, MouseEvent } from 'react';
2
- import { IDataAttributes } from '../types';
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 | Document): Element =>
32
- element.nodeName === 'HTML' || element === document
33
- ? (element as Element)
34
- : (element.parentNode as Element) ?? (element as ShadowRoot).host;
35
-
36
- export const getStyleComputedProperty = (element: Element): Partial<CSSStyleDeclaration> =>
37
- element.nodeType !== 1 ? {} : getComputedStyle(element, null);
38
-
39
- export const getScrollParent = (element: Element | Document): Element => {
40
- if (!element) {
41
- return document.body;
42
- }
43
-
44
- switch (element.nodeName) {
45
- case 'HTML':
46
- case 'BODY':
47
- return (element as Element).ownerDocument.body;
48
- case '#document':
49
- return (element as Document).body;
50
- }
51
-
52
- const { overflow, overflowX, overflowY } = getStyleComputedProperty(element as Element) ?? {};
53
- if (/(auto|scroll|overlay)/.test((overflow ?? '') + (overflowY ?? '') + (overflowX ?? ''))) {
54
- return element as Element;
55
- }
56
- return getScrollParent(getParentNode(element as Element));
57
- };
58
-
59
- export const isElementOffScreen = (element: HTMLElement, input?: HTMLElement): boolean => {
60
- const el = element;
61
- const scrollParent = getScrollParent(element);
62
-
63
- const { scrollHeight: scrollHeightWithElement } = scrollParent;
64
- el.hidden = true;
65
- const { scrollHeight: scrollHeightWithoutElement } = scrollParent;
66
- el.hidden = false;
67
-
68
- const isOffscreen = scrollHeightWithElement !== scrollHeightWithoutElement;
69
-
70
- if (isOffscreen && input !== undefined) {
71
- const elRect = el.getBoundingClientRect();
72
- const scrollParentRect = scrollParent.getBoundingClientRect();
73
- const topOffset = elRect.top - scrollParentRect.top;
74
- if (input.clientHeight + el.clientHeight > topOffset) {
75
- return false;
76
- }
77
- }
78
-
79
- return isOffscreen;
80
- };
81
-
82
- export const getNumberInRange = (value: number, min = -Infinity, max = Infinity): number =>
83
- Math.min(max, Math.max(min, value));
84
-
85
- const DEFAULT_THOUSANDS_SEPARATOR = '\u2009';
86
-
87
- export const formatStringNumber = (
88
- val?: string,
89
- separator = DEFAULT_THOUSANDS_SEPARATOR,
90
- ): string => {
91
- if (val === undefined) {
92
- return '';
93
- }
94
- const parts = val.split('.');
95
-
96
- parts[0] = parts[0]
97
- // убрать лидирующие нули
98
- .replace(/^0+(?=\d)/, '')
99
- // проставить сепараторы тысяч
100
- .replace(/\B(?=(\d{3})+(?!\d))/g, separator);
101
- return (parts[1] ?? '').length > 0 ? parts.join('.') : parts[0];
102
- };
103
-
104
- export const formatNumber = (val?: number, separator = DEFAULT_THOUSANDS_SEPARATOR): string => {
105
- if (val === undefined || isNaN(val)) {
106
- return '';
107
- }
108
- return formatStringNumber(String(val), separator);
109
- };
110
-
111
- export const removeStringFormat = (val?: string): string =>
112
- (val ?? '').replace(',', '.').replace(/\s/g, '');
113
-
114
- export const stringToNumber = (val?: string): number | undefined => {
115
- const trimmed = removeStringFormat(val);
116
- if (trimmed === '') {
117
- return undefined;
118
- }
119
- const num = Number(trimmed);
120
- return isNaN(num) ? undefined : num;
121
- };
122
-
123
- export const setCaretPosition = (elem: HTMLInputElement, caretPos: number | null): void => {
124
- if (caretPos === null || elem === null) {
125
- return;
126
- }
127
- if (elem.selectionStart) {
128
- elem.focus();
129
- elem.setSelectionRange(caretPos, caretPos);
130
- } else {
131
- elem.focus();
132
- }
133
- };
134
-
135
- export const isSpaceChar = (char?: string): boolean =>
136
- char !== undefined && char.match(/\s/) !== null;
137
-
138
- export const isInt = (n: number): boolean => n % 1 === 0;
139
-
140
- export const getNumberLength = (n?: number): number =>
141
- n === undefined || isNaN(n) ? 0 : n.toString().length;
142
-
143
- // TODO: Не должен проверять строку, т.к. под это есть отдельный хелпер
144
- /**
145
- * Проверяет, что `val` не `null`, не `undefined` и не пустая строка
146
- */
147
- export const isNotEmpty = <T>(val: T | null | undefined): val is T =>
148
- typeof val === 'string' ? val.trim() !== '' : val !== null && val !== undefined;
149
-
150
- /**
151
- * Проверяет, что переданное значение `null` или `undefined`
152
- */
153
- export const isEmpty = <T>(val: T | null | undefined): val is null | undefined =>
154
- val === null || val === undefined;
155
-
156
- /**
157
- * Проверяет, что передана непустая строка
158
- */
159
- export const isStringNotEmpty = <T extends string>(value: T | undefined | null): value is T =>
160
- (value ?? '').trim() !== '';
161
-
162
- export const trimStringToMaxLength = (val: string, maxLength: number) =>
163
- val.length > maxLength ? val.slice(0, maxLength) : val;
164
-
165
- export const addDataAttributes = (data: IDataAttributes = {}): IDataAttributes =>
166
- Object.fromEntries(
167
- Object.entries(data).map(([key, value]) =>
168
- isNotEmpty(value) ? [`data-${transformToKebab(key)}`, value] : [],
169
- ),
170
- );
171
-
172
- export const addDataTestId = (
173
- ...args: Parameters<typeof getTestId>
174
- ): { 'data-testid': string } | undefined => {
175
- const testId = getTestId(...args);
176
- return isNotEmpty(testId) ? { 'data-testid': testId } : undefined;
177
- };
178
-
179
- export const getTestId = (
180
- testId: string | undefined,
181
- postfix?: string | number,
182
- ): string | undefined => {
183
- if (!isNotEmpty(testId)) {
184
- return undefined;
185
- }
186
- return isNotEmpty(postfix) ? `${testId}-${postfix}` : testId;
187
- };
188
-
189
- export const getSelectKeyHandler =
190
- (cb: (e: KeyboardEvent) => void): ((e: KeyboardEvent) => void) =>
191
- (e) => {
192
- if (e.code === 'Enter' || e.code === 'NumpadEnter') {
193
- cb(e);
194
- }
195
- };
196
-
197
- export const addClickHandler = (
198
- cb?: (e: MouseEvent | KeyboardEvent) => void,
199
- hasAction = true,
200
- ): HTMLAttributes<unknown> =>
201
- hasAction && isNotEmpty(cb)
202
- ? {
203
- tabIndex: 0,
204
- onClick: cb as (e: MouseEvent) => void,
205
- onKeyDown: getSelectKeyHandler(cb),
206
- }
207
- : {
208
- tabIndex: -1,
209
- };
210
-
211
- /**
212
- * Позволяет создать текстовый фильтр для набора items
213
- * @param getter - функция возвращающая набор строковых значений из каждого item,
214
- * по которым должен осуществляться поиск
215
- */
216
- export const createFilter =
217
- <T>(
218
- getter: (item: T) => Array<string | undefined>,
219
- compareFn?: (item: string, query: string) => boolean,
220
- ): ((items: T[], query: string) => T[]) =>
221
- (items, query) =>
222
- items.filter((item) => {
223
- const possibleValues = getter(item).reduce(
224
- (acc, cur) => [...acc, ...(isStringNotEmpty(cur) ? [cur.toLowerCase()] : [])],
225
- [] as string[],
226
- );
227
- const queryString = query.toLowerCase().trim();
228
- return possibleValues.some((v) =>
229
- (compareFn ?? (() => v?.includes(queryString)))(v, queryString),
230
- );
231
- });
1
+ import hexToRgba from 'hex-to-rgba';
2
+ import {
3
+ isNotEmpty,
4
+ isString,
5
+ isStringNotEmpty,
6
+ } from '@true-engineering/true-react-platform-helpers';
7
+ import { IDataAttributes } from '../types';
8
+
9
+ export const rgba = hexToRgba;
10
+
11
+ export const transformToKebab = (string: string): string => {
12
+ let result = '';
13
+ string.split('').forEach((char) => {
14
+ if (char.toLowerCase() === char) {
15
+ result += char;
16
+ } else {
17
+ result += `-${char.toLowerCase()}`;
18
+ }
19
+ });
20
+
21
+ return result;
22
+ };
23
+
24
+ export const hasExactParent = (element: Element, parent: Element): boolean => {
25
+ if (element === parent) {
26
+ return true; // Found the exact parent
27
+ }
28
+
29
+ const parentNode = getParentNode(element);
30
+
31
+ if (parentNode === element) {
32
+ return false; // Reached the top-level HTML element or Shadow DOM host
33
+ }
34
+
35
+ return hasExactParent(parentNode, parent);
36
+ };
37
+
38
+ export const getParentNode = (element: Element | ShadowRoot | Document): Element =>
39
+ element.nodeName === 'HTML' || element === document
40
+ ? (element as Element)
41
+ : (element.parentNode as Element) ?? (element as ShadowRoot).host;
42
+
43
+ export const getStyleComputedProperty = (element: Element): Partial<CSSStyleDeclaration> =>
44
+ element.nodeType !== 1 ? {} : getComputedStyle(element, null);
45
+
46
+ export const getScrollParent = (element: Element | Document): Element => {
47
+ if (!element) {
48
+ return document.body;
49
+ }
50
+
51
+ switch (element.nodeName) {
52
+ case 'HTML':
53
+ case 'BODY':
54
+ return (element as Element).ownerDocument.body;
55
+ case '#document':
56
+ return (element as Document).body;
57
+ }
58
+
59
+ const { overflow, overflowX, overflowY } = getStyleComputedProperty(element as Element) ?? {};
60
+ if (/(auto|scroll|overlay)/.test((overflow ?? '') + (overflowY ?? '') + (overflowX ?? ''))) {
61
+ return element as Element;
62
+ }
63
+ return getScrollParent(getParentNode(element as Element));
64
+ };
65
+
66
+ export const isElementOffScreen = (element: HTMLElement, input?: HTMLElement): 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 = (value: number, min = -Infinity, max = Infinity): number =>
90
+ Math.min(max, Math.max(min, value));
91
+
92
+ const DEFAULT_THOUSANDS_SEPARATOR = '\u2009';
93
+
94
+ export const formatStringNumber = (
95
+ val?: string,
96
+ separator = DEFAULT_THOUSANDS_SEPARATOR,
97
+ ): string => {
98
+ if (val === undefined) {
99
+ return '';
100
+ }
101
+ const parts = val.split('.');
102
+
103
+ parts[0] = parts[0]
104
+ // убрать лидирующие нули
105
+ .replace(/^0+(?=\d)/, '')
106
+ // проставить сепараторы тысяч
107
+ .replace(/\B(?=(\d{3})+(?!\d))/g, separator);
108
+ return (parts[1] ?? '').length > 0 ? parts.join('.') : parts[0];
109
+ };
110
+
111
+ export const formatNumber = (val?: number, separator = DEFAULT_THOUSANDS_SEPARATOR): string => {
112
+ if (val === undefined || isNaN(val)) {
113
+ return '';
114
+ }
115
+ return formatStringNumber(String(val), separator);
116
+ };
117
+
118
+ export const removeStringFormat = (val?: string): string =>
119
+ (val ?? '').replace(',', '.').replace(/\s/g, '');
120
+
121
+ export const stringToNumber = (val?: string): number | undefined => {
122
+ const trimmed = removeStringFormat(val);
123
+ if (trimmed === '') {
124
+ return undefined;
125
+ }
126
+ const num = Number(trimmed);
127
+ return isNaN(num) ? undefined : num;
128
+ };
129
+
130
+ export const setCaretPosition = (elem: HTMLInputElement, caretPos: number | null): void => {
131
+ if (caretPos === null || elem === null) {
132
+ return;
133
+ }
134
+ if (elem.selectionStart) {
135
+ elem.focus();
136
+ elem.setSelectionRange(caretPos, caretPos);
137
+ } else {
138
+ elem.focus();
139
+ }
140
+ };
141
+
142
+ export const isSpaceChar = (char?: string): boolean =>
143
+ char !== undefined && char.match(/\s/) !== null;
144
+
145
+ export const getNumberLength = (n?: number): number =>
146
+ n === undefined || isNaN(n) ? 0 : n.toString().length;
147
+
148
+ export const trimStringToMaxLength = (val: string, maxLength: number) =>
149
+ val.length > maxLength ? val.slice(0, maxLength) : val;
150
+
151
+ export const addDataAttributes = (data: IDataAttributes = {}): IDataAttributes =>
152
+ Object.fromEntries(
153
+ Object.entries(data).map(([key, value]) =>
154
+ (isString(value) && isStringNotEmpty(value)) || isNotEmpty(value)
155
+ ? [`data-${transformToKebab(key)}`, value]
156
+ : [],
157
+ ),
158
+ );
@@ -1,87 +1,87 @@
1
- import { phoneInfo, IPhoneInfo, IPhoneValue } from '../components';
2
- import { isNotEmpty } from './utils';
3
-
4
- export const findCountryByCode = (
5
- countryCode?: string,
6
- countriesList = phoneInfo,
7
- ): IPhoneInfo | undefined =>
8
- countryCode !== undefined
9
- ? countriesList.find((info) => info.countryCode === countryCode)
10
- : undefined;
11
-
12
- export const findCountryIndexByCode = (countryCode?: string, countriesList = phoneInfo): number =>
13
- countryCode !== undefined
14
- ? countriesList.findIndex((info) => info.countryCode === countryCode)
15
- : -1;
16
-
17
- export const getCountryTextByLocale = (country: IPhoneInfo, locale: string): string =>
18
- locale.toLowerCase() === 'ru' ? country.countryRu : country.countryEn;
19
-
20
- export const sortCountriesByLocale = (countriesList: IPhoneInfo[], locale: string) =>
21
- countriesList.sort((countryA, countryB) =>
22
- getCountryTextByLocale(countryA, locale).localeCompare(
23
- getCountryTextByLocale(countryB, locale),
24
- ),
25
- );
26
-
27
- export const checkSearchStringInCountry = (country: IPhoneInfo, searchString: string): boolean => {
28
- const checkValue = searchString.toLowerCase().trim().replace('+', '');
29
-
30
- const possibleValues = [
31
- country.countryRu,
32
- country.countryEn,
33
- ...country.countryRu.split(' '),
34
- ...country.countryEn.split(' '),
35
- country.dialCode,
36
- ].map((part) => part.toLowerCase());
37
-
38
- return possibleValues.some((v) => v.startsWith(checkValue));
39
- };
40
-
41
- export const getFullPhone = (phone?: IPhoneValue): string =>
42
- (phone?.dialCode ?? '') + (phone?.phoneNumber ?? '');
43
-
44
- export const getCountryCodeFromPhone = (phoneWithCode: string): string | undefined => {
45
- // попробуем найти уникальный код страны fullCode (dialCode + arealCode)
46
- let countryCode = phoneInfo.find((info) =>
47
- info.fullCodes.some((code) => phoneWithCode.startsWith(code)),
48
- )?.countryCode;
49
-
50
- if (countryCode === undefined && isNotEmpty(phoneWithCode)) {
51
- // если не нашли уникальный fullCode (dialCode + arealCode),
52
- // то пробуем найти dialCode и выбираем с наименьшим Priority
53
- countryCode = phoneInfo
54
- .filter((info) => phoneWithCode.startsWith(info.dialCode))
55
- .sort(
56
- (infoA, infoB) => (infoA.dialCodePriority ?? 1000) - (infoB.dialCodePriority ?? 1000),
57
- )[0]?.countryCode;
58
- }
59
-
60
- return countryCode;
61
- };
62
-
63
- export const getPhoneObjFromString = (fullPhone: string, countryCode?: string): IPhoneValue => {
64
- const newCountryCode = countryCode ?? getCountryCodeFromPhone(fullPhone);
65
- const dialCode = findCountryByCode(newCountryCode)?.dialCode;
66
- const phoneNumber = fullPhone.slice(dialCode?.length);
67
-
68
- return { dialCode, phoneNumber, countryCode: newCountryCode };
69
- };
70
-
71
- export const DEFAULT_PHONE_MASK = '+ 9 999 999 999 999 99';
72
-
73
- export const getPhoneMask = (countryCode?: string): string => {
74
- let mask = DEFAULT_PHONE_MASK;
75
- if (countryCode !== undefined) {
76
- const countrySettings = findCountryByCode(countryCode);
77
-
78
- if (countrySettings !== undefined) {
79
- const dialCodeLength = countrySettings.dialCode.length;
80
- const areaAndPhoneMask =
81
- countrySettings.phoneMask ?? mask.substring(2, mask.length - dialCodeLength);
82
- mask = `+ ${'9'.repeat(dialCodeLength)} ${areaAndPhoneMask}`;
83
- }
84
- }
85
-
86
- return mask;
87
- };
1
+ import { isNotEmpty, isEmpty } from '@true-engineering/true-react-platform-helpers';
2
+ import { phoneInfo, IPhoneInfo, IPhoneValue } from '../components';
3
+
4
+ export const findCountryByCode = (
5
+ countryCode?: string,
6
+ countriesList = phoneInfo,
7
+ ): IPhoneInfo | undefined =>
8
+ isNotEmpty(countryCode)
9
+ ? countriesList.find((info) => info.countryCode === countryCode)
10
+ : undefined;
11
+
12
+ export const findCountryIndexByCode = (countryCode?: string, countriesList = phoneInfo): number =>
13
+ isNotEmpty(countryCode)
14
+ ? countriesList.findIndex((info) => info.countryCode === countryCode)
15
+ : -1;
16
+
17
+ export const getCountryTextByLocale = (country: IPhoneInfo, locale: string): string =>
18
+ locale.toLowerCase() === 'ru' ? country.countryRu : country.countryEn;
19
+
20
+ export const sortCountriesByLocale = (countriesList: IPhoneInfo[], locale: string) =>
21
+ countriesList.sort((countryA, countryB) =>
22
+ getCountryTextByLocale(countryA, locale).localeCompare(
23
+ getCountryTextByLocale(countryB, locale),
24
+ ),
25
+ );
26
+
27
+ export const checkSearchStringInCountry = (country: IPhoneInfo, searchString: string): boolean => {
28
+ const checkValue = searchString.toLowerCase().trim().replace('+', '');
29
+
30
+ const possibleValues = [
31
+ country.countryRu,
32
+ country.countryEn,
33
+ ...country.countryRu.split(' '),
34
+ ...country.countryEn.split(' '),
35
+ country.dialCode,
36
+ ].map((part) => part.toLowerCase());
37
+
38
+ return possibleValues.some((v) => v.startsWith(checkValue));
39
+ };
40
+
41
+ export const getFullPhone = (phone?: IPhoneValue): string =>
42
+ (phone?.dialCode ?? '') + (phone?.phoneNumber ?? '');
43
+
44
+ export const getCountryCodeFromPhone = (phoneWithCode: string): string | undefined => {
45
+ // попробуем найти уникальный код страны fullCode (dialCode + arealCode)
46
+ let countryCode = phoneInfo.find((info) =>
47
+ info.fullCodes.some((code) => phoneWithCode.startsWith(code)),
48
+ )?.countryCode;
49
+
50
+ if (isEmpty(countryCode) && isNotEmpty(phoneWithCode)) {
51
+ // если не нашли уникальный fullCode (dialCode + arealCode),
52
+ // то пробуем найти dialCode и выбираем с наименьшим Priority
53
+ countryCode = phoneInfo
54
+ .filter((info) => phoneWithCode.startsWith(info.dialCode))
55
+ .sort(
56
+ (infoA, infoB) => (infoA.dialCodePriority ?? 1000) - (infoB.dialCodePriority ?? 1000),
57
+ )[0]?.countryCode;
58
+ }
59
+
60
+ return countryCode;
61
+ };
62
+
63
+ export const getPhoneObjFromString = (fullPhone: string, countryCode?: string): IPhoneValue => {
64
+ const newCountryCode = countryCode ?? getCountryCodeFromPhone(fullPhone);
65
+ const dialCode = findCountryByCode(newCountryCode)?.dialCode;
66
+ const phoneNumber = fullPhone.slice(dialCode?.length);
67
+
68
+ return { dialCode, phoneNumber, countryCode: newCountryCode };
69
+ };
70
+
71
+ export const DEFAULT_PHONE_MASK = '+ 9 999 999 999 999 99';
72
+
73
+ export const getPhoneMask = (countryCode?: string): string => {
74
+ let mask = DEFAULT_PHONE_MASK;
75
+ if (isNotEmpty(countryCode)) {
76
+ const countrySettings = findCountryByCode(countryCode);
77
+
78
+ if (countrySettings !== undefined) {
79
+ const dialCodeLength = countrySettings.dialCode.length;
80
+ const areaAndPhoneMask =
81
+ countrySettings.phoneMask ?? mask.substring(2, mask.length - dialCodeLength);
82
+ mask = `+ ${'9'.repeat(dialCodeLength)} ${areaAndPhoneMask}`;
83
+ }
84
+ }
85
+
86
+ return mask;
87
+ };
@@ -1,17 +1,17 @@
1
- import { Modifier } from 'react-overlays/usePopper';
2
-
3
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
4
- export const minWidthModifier: Partial<Modifier<any, any>> = {
5
- name: 'minWidth',
6
- enabled: true,
7
- phase: 'beforeWrite',
8
- requires: ['computeStyles'],
9
- fn: ({ state }) => {
10
- state.styles.popper.minWidth = `${state.rects.reference.width}px`;
11
- },
12
- effect: ({ state }) => {
13
- state.elements.popper.style.minWidth = `${
14
- (state.elements.reference as HTMLElement).offsetWidth
15
- }px`;
16
- },
17
- };
1
+ import { Modifier } from 'react-overlays/usePopper';
2
+
3
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
4
+ export const minWidthModifier: Partial<Modifier<any, any>> = {
5
+ name: 'minWidth',
6
+ enabled: true,
7
+ phase: 'beforeWrite',
8
+ requires: ['computeStyles'],
9
+ fn: ({ state }) => {
10
+ state.styles.popper.minWidth = `${state.rects.reference.width}px`;
11
+ },
12
+ effect: ({ state }) => {
13
+ state.elements.popper.style.minWidth = `${
14
+ (state.elements.reference as HTMLElement).offsetWidth
15
+ }px`;
16
+ },
17
+ };
@@ -1,5 +1,6 @@
1
- import { ReactElement, ReactNode } from 'react';
2
- import { Icon, IIconType } from '../components';
3
-
4
- export const renderIcon = (icon: IIconType | ReactElement): ReactNode =>
5
- typeof icon === 'string' ? <Icon type={icon} /> : icon;
1
+ import { ReactNode } from 'react';
2
+ import { Icon } from '../components';
3
+ import { IIcon } from '../types';
4
+
5
+ export const renderIcon = (icon: IIcon): ReactNode =>
6
+ typeof icon === 'string' ? <Icon type={icon} /> : icon;
@@ -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';