@true-engineering/true-react-common-ui-kit 3.40.0 → 3.42.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 (95) hide show
  1. package/LICENSE +201 -201
  2. package/README.md +16 -1
  3. package/dist/components/FiltersPane/FiltersPane.stories.d.ts +2 -1
  4. package/dist/components/FiltersPane/components/FilterWrapper/FilterWrapper.styles.d.ts +1 -1
  5. package/dist/components/FiltersPane/components/FiltersPaneSearch/FiltersPaneSearch.styles.d.ts +1 -1
  6. package/dist/components/FiltersPane/types.d.ts +1 -0
  7. package/dist/components/Input/Input.styles.d.ts +1 -1
  8. package/dist/components/SearchInput/SearchInput.stories.d.ts +1 -1
  9. package/dist/components/Select/Select.styles.d.ts +3 -3
  10. package/dist/components/SmartInput/SmartInput.d.ts +8 -6
  11. package/dist/components/SmartInput/constants.d.ts +4 -6
  12. package/dist/components/SmartInput/helpers.d.ts +2 -4
  13. package/dist/components/SmartInput/types.d.ts +5 -0
  14. package/dist/true-react-common-ui-kit.js +1154 -998
  15. package/dist/true-react-common-ui-kit.js.map +1 -1
  16. package/dist/true-react-common-ui-kit.umd.cjs +1154 -998
  17. package/dist/true-react-common-ui-kit.umd.cjs.map +1 -1
  18. package/package.json +3 -2
  19. package/src/components/AccountInfo/AccountInfo.stories.tsx +32 -32
  20. package/src/components/AccountInfo/AccountInfo.tsx +80 -80
  21. package/src/components/AddButton/AddButton.stories.tsx +21 -21
  22. package/src/components/AddButton/AddButton.tsx +52 -52
  23. package/src/components/Colors/Colors.stories.tsx +7 -7
  24. package/src/components/DateInput/DateInput.tsx +90 -90
  25. package/src/components/DateInput/constants.ts +2 -2
  26. package/src/components/Description/Description.stories.tsx +27 -27
  27. package/src/components/Description/Description.tsx +61 -61
  28. package/src/components/FiltersPane/FiltersPane.stories.tsx +12 -0
  29. package/src/components/FiltersPane/components/FilterValueView/FilterValueView.tsx +166 -166
  30. package/src/components/FiltersPane/components/FilterWithDates/FilterWithDates.tsx +210 -210
  31. package/src/components/FiltersPane/components/FilterWithPeriod/FilterWithPeriod.tsx +177 -177
  32. package/src/components/FiltersPane/components/FilterWrapper/FilterWrapper.styles.ts +4 -0
  33. package/src/components/FiltersPane/components/FilterWrapper/FilterWrapper.tsx +20 -2
  34. package/src/components/FiltersPane/types.ts +1 -0
  35. package/src/components/Flag/Flag.stories.tsx +29 -29
  36. package/src/components/Flag/Flag.tsx +26 -26
  37. package/src/components/Flag/augment.d.ts +1 -1
  38. package/src/components/FlexibleTable/components/FlexibleTableRow/FlexibleTableRow.styles.ts +25 -25
  39. package/src/components/Icon/Icon.stories.tsx +86 -86
  40. package/src/components/Icon/complexIcons/augment.d.ts +1 -1
  41. package/src/components/Icon/complexIcons/avatarGreen.svg +57 -57
  42. package/src/components/Icon/complexIcons/index.ts +1 -1
  43. package/src/components/IncrementInput/IncrementInput.tsx +105 -105
  44. package/src/components/Input/types.ts +32 -32
  45. package/src/components/Modal/Modal.stories.tsx +105 -105
  46. package/src/components/MultiSelect/MultiSelect.stories.tsx +46 -46
  47. package/src/components/MultiSelect/MultiSelect.tsx +106 -106
  48. package/src/components/MultiSelect/components/MultiSelectInput/MultiSelectInput.tsx +53 -53
  49. package/src/components/Notification/Notification.stories.tsx +55 -55
  50. package/src/components/Notification/Notification.styles.ts +57 -57
  51. package/src/components/Notification/Notification.tsx +77 -77
  52. package/src/components/Notification/types.ts +1 -1
  53. package/src/components/NumberInput/NumberInput.tsx +137 -137
  54. package/src/components/NumberInput/index.ts +1 -1
  55. package/src/components/PhoneInput/PhoneInput.tsx +214 -214
  56. package/src/components/PhoneInput/components/PhoneInputCountryList/PhoneInputCountryList.tsx +155 -155
  57. package/src/components/PhoneInput/types.ts +16 -16
  58. package/src/components/RadioButton/RadioButton.stories.tsx +46 -46
  59. package/src/components/RadioButton/RadioButton.tsx +57 -57
  60. package/src/components/ScrollIntoViewIfNeeded/index.ts +1 -1
  61. package/src/components/Select/constants.ts +2 -2
  62. package/src/components/Select/types.ts +1 -1
  63. package/src/components/Selector/Selector.stories.tsx +62 -62
  64. package/src/components/Selector/Selector.tsx +115 -115
  65. package/src/components/Selector/index.ts +2 -2
  66. package/src/components/Selector/types.ts +12 -12
  67. package/src/components/Skeleton/Skeleton.stories.tsx +19 -19
  68. package/src/components/SmartInput/SmartInput.stories.tsx +2 -1
  69. package/src/components/SmartInput/SmartInput.tsx +76 -94
  70. package/src/components/SmartInput/constants.ts +9 -2
  71. package/src/components/SmartInput/helpers.ts +26 -13
  72. package/src/components/SmartInput/types.ts +7 -0
  73. package/src/components/Status/Status.stories.tsx +73 -73
  74. package/src/components/Status/Status.styles.ts +143 -143
  75. package/src/components/Status/Status.tsx +49 -49
  76. package/src/components/Status/constants.ts +11 -11
  77. package/src/components/Status/index.ts +3 -3
  78. package/src/components/Status/types.ts +5 -5
  79. package/src/components/Switch/Switch.stories.tsx +40 -40
  80. package/src/components/Switch/Switch.tsx +75 -75
  81. package/src/components/TextWithInfo/TextWithInfo.stories.tsx +53 -53
  82. package/src/components/TextWithInfo/TextWithInfo.tsx +62 -62
  83. package/src/components/TextWithTooltip/TextWithTooltip.stories.tsx +58 -58
  84. package/src/components/ThemedPreloader/ThemedPreloader.stories.tsx +41 -41
  85. package/src/components/ThemedPreloader/ThemedPreloader.tsx +54 -54
  86. package/src/components/ThemedPreloader/components/DefaultPreloader/index.ts +1 -1
  87. package/src/components/Toaster/Toaster.stories.tsx +30 -30
  88. package/src/components/Tooltip/Tooltip.stories.tsx +19 -19
  89. package/src/components/Tooltip/Tooltip.tsx +35 -35
  90. package/src/components/Tooltip/types.ts +1 -1
  91. package/src/helpers/popper-helpers.ts +17 -17
  92. package/src/hooks/use-dropdown.ts +84 -84
  93. package/src/hooks/use-is-mounted.ts +15 -15
  94. package/src/theme/helpers.ts +76 -76
  95. package/src/vite-env.d.ts +1 -1
@@ -1,62 +1,62 @@
1
- import { useState } from 'react';
2
- import { ComponentStory } from '@storybook/react';
3
- import { Status } from '../Status';
4
- import { Selector } from './Selector';
5
- import { ISelectorOption } from './types';
6
-
7
- const options: Array<ISelectorOption<string>> = [
8
- {
9
- label: 'Label 1',
10
- value: '1',
11
- icon: 'eye',
12
- },
13
- {
14
- label: 'My label 2',
15
- value: '2',
16
- },
17
- {
18
- label: (
19
- <>
20
- This is Label 3{' '}
21
- <Status size="xs" color="teal">
22
- P
23
- </Status>
24
- </>
25
- ),
26
- value: '3',
27
- },
28
- {
29
- label: 'This is disabled',
30
- value: '4',
31
- isDisabled: true,
32
- },
33
- ];
34
-
35
- export default {
36
- title: 'Controls/Selector',
37
- component: Selector,
38
- };
39
-
40
- const Template: ComponentStory<typeof Selector> = (args) => {
41
- const [value, setValue] = useState(options[0].value);
42
- return <Selector {...args} value={value} options={options} onChange={setValue} />;
43
- };
44
-
45
- export const Default = Template.bind({});
46
-
47
- Default.args = {
48
- size: 'l',
49
- iconPosition: 'left',
50
- hasSameOptionsWidth: true,
51
- isDisabled: false,
52
- isInvalid: false,
53
- isRequired: false,
54
- };
55
-
56
- Default.argTypes = {};
57
-
58
- Default.parameters = {
59
- controls: {
60
- exclude: ['options', 'value', 'testId', 'onChange'],
61
- },
62
- };
1
+ import { useState } from 'react';
2
+ import { ComponentStory } from '@storybook/react';
3
+ import { Status } from '../Status';
4
+ import { Selector } from './Selector';
5
+ import { ISelectorOption } from './types';
6
+
7
+ const options: Array<ISelectorOption<string>> = [
8
+ {
9
+ label: 'Label 1',
10
+ value: '1',
11
+ icon: 'eye',
12
+ },
13
+ {
14
+ label: 'My label 2',
15
+ value: '2',
16
+ },
17
+ {
18
+ label: (
19
+ <>
20
+ This is Label 3{' '}
21
+ <Status size="xs" color="teal">
22
+ P
23
+ </Status>
24
+ </>
25
+ ),
26
+ value: '3',
27
+ },
28
+ {
29
+ label: 'This is disabled',
30
+ value: '4',
31
+ isDisabled: true,
32
+ },
33
+ ];
34
+
35
+ export default {
36
+ title: 'Controls/Selector',
37
+ component: Selector,
38
+ };
39
+
40
+ const Template: ComponentStory<typeof Selector> = (args) => {
41
+ const [value, setValue] = useState(options[0].value);
42
+ return <Selector {...args} value={value} options={options} onChange={setValue} />;
43
+ };
44
+
45
+ export const Default = Template.bind({});
46
+
47
+ Default.args = {
48
+ size: 'l',
49
+ iconPosition: 'left',
50
+ hasSameOptionsWidth: true,
51
+ isDisabled: false,
52
+ isInvalid: false,
53
+ isRequired: false,
54
+ };
55
+
56
+ Default.argTypes = {};
57
+
58
+ Default.parameters = {
59
+ controls: {
60
+ exclude: ['options', 'value', 'testId', 'onChange'],
61
+ },
62
+ };
@@ -1,115 +1,115 @@
1
- import { useEffect, useRef, useState } from 'react';
2
- import clsx from 'clsx';
3
- import {
4
- addDataTestId,
5
- getTestId,
6
- hasDuplicates,
7
- isNotEmpty,
8
- isReactNodeNotEmpty,
9
- } from '@true-engineering/true-react-platform-helpers';
10
- import { addDataAttributes } from '../../helpers';
11
- import { ICommonProps } from '../../types';
12
- import { renderIcon } from '../Icon';
13
- import { ISelectorOption, ISelectorValue } from './types';
14
- import { getSelectorLineStyle, useStyles, ISelectorStyles } from './Selector.styles';
15
-
16
- export interface ISelectorProps<V extends ISelectorValue> extends ICommonProps<ISelectorStyles> {
17
- options: Array<ISelectorOption<V>>;
18
- value?: V;
19
- /** @default false */
20
- isDisabled?: boolean;
21
- /** @default false */
22
- isRequired?: boolean;
23
- /** @default false */
24
- isInvalid?: boolean;
25
- /** @default true */
26
- hasSameOptionsWidth?: boolean;
27
- /** @default 'left' */
28
- iconPosition?: 'left' | 'right';
29
- /** @default 'l' */
30
- size?: 's' | 'm' | 'l';
31
- onChange: (value: V) => void;
32
- }
33
-
34
- export function Selector<V extends ISelectorValue>({
35
- options,
36
- value,
37
- isDisabled = false,
38
- isRequired = false,
39
- isInvalid = false,
40
- hasSameOptionsWidth = true,
41
- iconPosition = 'left',
42
- size = 'l',
43
- testId,
44
- data,
45
- tweakStyles,
46
- onChange,
47
- }: ISelectorProps<V>): JSX.Element {
48
- const classes = useStyles({ theme: tweakStyles });
49
- const optionsValues = options.map((opt) => opt.value);
50
-
51
- const [elementsData, setElementsData] = useState<HTMLElement[]>([]);
52
- const listRef = useRef<HTMLDivElement>(null);
53
-
54
- if (hasDuplicates(optionsValues)) {
55
- // eslint-disable-next-line no-console
56
- console.error('Selector: Значения options.value должны быть уникальными');
57
- }
58
-
59
- useEffect(() => {
60
- const listEl = listRef.current;
61
- if (listEl === null) {
62
- return;
63
- }
64
-
65
- setElementsData([...listEl.querySelectorAll<HTMLElement>(`.${classes.optionWrapper}`)]);
66
- }, [options, size]);
67
-
68
- const activeElementData = isNotEmpty(value)
69
- ? elementsData[optionsValues.indexOf(value)]
70
- : undefined;
71
-
72
- return (
73
- <div
74
- className={clsx(classes.root, classes.selector, {
75
- [classes.iconFromRight]: iconPosition === 'right',
76
- [classes.invalid]: isInvalid,
77
- [classes.required]: isRequired,
78
- [classes.autoWidth]: hasSameOptionsWidth,
79
- })}
80
- ref={listRef}
81
- {...addDataTestId(testId)}
82
- {...addDataAttributes(data)}
83
- >
84
- {isNotEmpty(activeElementData) && (
85
- <div className={classes.line} style={getSelectorLineStyle(activeElementData)} />
86
- )}
87
-
88
- {options.map((option) => {
89
- const optionId = option.value.toString();
90
- const isDisabledOption = option.isDisabled ?? isDisabled;
91
- const isActiveOption = option.value === value;
92
-
93
- return (
94
- <div key={optionId} className={classes.optionWrapper} {...addDataTestId(testId)}>
95
- <button
96
- type="button"
97
- className={clsx(classes.option, classes[size], {
98
- [classes.active]: isActiveOption,
99
- [classes.disabled]: isDisabledOption,
100
- })}
101
- disabled={isDisabledOption}
102
- onClick={!isDisabledOption ? () => onChange(option.value) : undefined}
103
- {...addDataTestId(getTestId(testId, optionId))}
104
- >
105
- {isReactNodeNotEmpty(option.icon) && (
106
- <div className={classes.optionIcon}>{renderIcon(option.icon)}</div>
107
- )}
108
- <div className={classes.optionText}>{option.label}</div>
109
- </button>
110
- </div>
111
- );
112
- })}
113
- </div>
114
- );
115
- }
1
+ import { useEffect, useRef, useState } from 'react';
2
+ import clsx from 'clsx';
3
+ import {
4
+ addDataTestId,
5
+ getTestId,
6
+ hasDuplicates,
7
+ isNotEmpty,
8
+ isReactNodeNotEmpty,
9
+ } from '@true-engineering/true-react-platform-helpers';
10
+ import { addDataAttributes } from '../../helpers';
11
+ import { ICommonProps } from '../../types';
12
+ import { renderIcon } from '../Icon';
13
+ import { ISelectorOption, ISelectorValue } from './types';
14
+ import { getSelectorLineStyle, useStyles, ISelectorStyles } from './Selector.styles';
15
+
16
+ export interface ISelectorProps<V extends ISelectorValue> extends ICommonProps<ISelectorStyles> {
17
+ options: Array<ISelectorOption<V>>;
18
+ value?: V;
19
+ /** @default false */
20
+ isDisabled?: boolean;
21
+ /** @default false */
22
+ isRequired?: boolean;
23
+ /** @default false */
24
+ isInvalid?: boolean;
25
+ /** @default true */
26
+ hasSameOptionsWidth?: boolean;
27
+ /** @default 'left' */
28
+ iconPosition?: 'left' | 'right';
29
+ /** @default 'l' */
30
+ size?: 's' | 'm' | 'l';
31
+ onChange: (value: V) => void;
32
+ }
33
+
34
+ export function Selector<V extends ISelectorValue>({
35
+ options,
36
+ value,
37
+ isDisabled = false,
38
+ isRequired = false,
39
+ isInvalid = false,
40
+ hasSameOptionsWidth = true,
41
+ iconPosition = 'left',
42
+ size = 'l',
43
+ testId,
44
+ data,
45
+ tweakStyles,
46
+ onChange,
47
+ }: ISelectorProps<V>): JSX.Element {
48
+ const classes = useStyles({ theme: tweakStyles });
49
+ const optionsValues = options.map((opt) => opt.value);
50
+
51
+ const [elementsData, setElementsData] = useState<HTMLElement[]>([]);
52
+ const listRef = useRef<HTMLDivElement>(null);
53
+
54
+ if (hasDuplicates(optionsValues)) {
55
+ // eslint-disable-next-line no-console
56
+ console.error('Selector: Значения options.value должны быть уникальными');
57
+ }
58
+
59
+ useEffect(() => {
60
+ const listEl = listRef.current;
61
+ if (listEl === null) {
62
+ return;
63
+ }
64
+
65
+ setElementsData([...listEl.querySelectorAll<HTMLElement>(`.${classes.optionWrapper}`)]);
66
+ }, [options, size]);
67
+
68
+ const activeElementData = isNotEmpty(value)
69
+ ? elementsData[optionsValues.indexOf(value)]
70
+ : undefined;
71
+
72
+ return (
73
+ <div
74
+ className={clsx(classes.root, classes.selector, {
75
+ [classes.iconFromRight]: iconPosition === 'right',
76
+ [classes.invalid]: isInvalid,
77
+ [classes.required]: isRequired,
78
+ [classes.autoWidth]: hasSameOptionsWidth,
79
+ })}
80
+ ref={listRef}
81
+ {...addDataTestId(testId)}
82
+ {...addDataAttributes(data)}
83
+ >
84
+ {isNotEmpty(activeElementData) && (
85
+ <div className={classes.line} style={getSelectorLineStyle(activeElementData)} />
86
+ )}
87
+
88
+ {options.map((option) => {
89
+ const optionId = option.value.toString();
90
+ const isDisabledOption = option.isDisabled ?? isDisabled;
91
+ const isActiveOption = option.value === value;
92
+
93
+ return (
94
+ <div key={optionId} className={classes.optionWrapper} {...addDataTestId(testId)}>
95
+ <button
96
+ type="button"
97
+ className={clsx(classes.option, classes[size], {
98
+ [classes.active]: isActiveOption,
99
+ [classes.disabled]: isDisabledOption,
100
+ })}
101
+ disabled={isDisabledOption}
102
+ onClick={!isDisabledOption ? () => onChange(option.value) : undefined}
103
+ {...addDataTestId(getTestId(testId, optionId))}
104
+ >
105
+ {isReactNodeNotEmpty(option.icon) && (
106
+ <div className={classes.optionIcon}>{renderIcon(option.icon)}</div>
107
+ )}
108
+ <div className={classes.optionText}>{option.label}</div>
109
+ </button>
110
+ </div>
111
+ );
112
+ })}
113
+ </div>
114
+ );
115
+ }
@@ -1,2 +1,2 @@
1
- export * from './Selector';
2
- export type { ISelectorStyles } from './Selector.styles';
1
+ export * from './Selector';
2
+ export type { ISelectorStyles } from './Selector.styles';
@@ -1,12 +1,12 @@
1
- import { ReactNode } from 'react';
2
- import { IIcon } from '../Icon';
3
-
4
- export type ISelectorValue = string | number | boolean;
5
-
6
- export interface ISelectorOption<V> {
7
- label: ReactNode;
8
- icon?: IIcon;
9
- value: V;
10
- /** @default false */
11
- isDisabled?: boolean;
12
- }
1
+ import { ReactNode } from 'react';
2
+ import { IIcon } from '../Icon';
3
+
4
+ export type ISelectorValue = string | number | boolean;
5
+
6
+ export interface ISelectorOption<V> {
7
+ label: ReactNode;
8
+ icon?: IIcon;
9
+ value: V;
10
+ /** @default false */
11
+ isDisabled?: boolean;
12
+ }
@@ -1,19 +1,19 @@
1
- import { ComponentStory } from '@storybook/react';
2
- import { Skeleton } from './Skeleton';
3
-
4
- export default {
5
- title: 'Feedback/Skeleton',
6
- component: Skeleton,
7
- };
8
-
9
- const Template: ComponentStory<typeof Skeleton> = (args) => (
10
- <div style={{ padding: 32, backgroundColor: '#fff', width: 200 }}>
11
- <Skeleton {...args} />
12
- </div>
13
- );
14
-
15
- export const Default = Template.bind({});
16
-
17
- Default.args = {
18
- height: '20px',
19
- };
1
+ import { ComponentStory } from '@storybook/react';
2
+ import { Skeleton } from './Skeleton';
3
+
4
+ export default {
5
+ title: 'Feedback/Skeleton',
6
+ component: Skeleton,
7
+ };
8
+
9
+ const Template: ComponentStory<typeof Skeleton> = (args) => (
10
+ <div style={{ padding: 32, backgroundColor: '#fff', width: 200 }}>
11
+ <Skeleton {...args} />
12
+ </div>
13
+ );
14
+
15
+ export const Default = Template.bind({});
16
+
17
+ Default.args = {
18
+ height: '20px',
19
+ };
@@ -33,7 +33,7 @@ const Template: ComponentStory<typeof SmartInput> = (args) => {
33
33
  <li>Трансформация значения в UPPERCASE</li>
34
34
  </ul>
35
35
 
36
- <SmartInput {...args} value={value} onChange={(v) => setValue(v)} />
36
+ <SmartInput {...args} value={value} onChange={setValue} />
37
37
  </div>
38
38
  );
39
39
  };
@@ -47,5 +47,6 @@ Default.args = {
47
47
  smartType: 'default',
48
48
  isUpperCase: false,
49
49
  isDisabled: false,
50
+ isTransliterationEnabled: true,
50
51
  hasFloatingLabel: true,
51
52
  };
@@ -1,133 +1,115 @@
1
- import { useState, useEffect, forwardRef, FormEvent, ClipboardEvent } from 'react';
2
- import { Input, IInputProps } from '../Input';
3
- import { CharactersMap, SMART_INPUT_REGEX_MAP, TransliterationMap } from './constants';
4
- import { transformCaseSensitive } from './helpers';
5
- import { ISmartType } from './types';
6
-
7
- export interface ISmartInputProps extends IInputProps {
8
- /** @default false */
9
- isUpperCase?: boolean;
1
+ import { ClipboardEvent, FormEvent, forwardRef, useRef } from 'react';
2
+ import { isEmpty, isNotEmpty } from '@true-engineering/true-react-platform-helpers';
3
+ import { useMergedRefs } from '../../hooks';
4
+ import { IInputProps, Input } from '../Input';
5
+ import {
6
+ CharactersMap,
7
+ EmailCharactersMap,
8
+ SMART_INPUT_REGEX_MAP,
9
+ TransliterationMap,
10
+ } from './constants';
11
+ import { mapSymbols } from './helpers';
12
+ import { IInputRawValue, ISmartType } from './types';
13
+
14
+ export interface ISmartInputProps extends Omit<IInputProps, 'onChange'> {
10
15
  /** @default 'default' */
11
16
  smartType?: ISmartType;
12
17
  regExp?: RegExp;
13
- onChange: (value: string) => void;
18
+ /** @default false */
19
+ isUpperCase?: boolean;
20
+ /** @default true */
21
+ isTransliterationEnabled?: boolean;
22
+ onChange: (
23
+ value: string,
24
+ rawValue: IInputRawValue,
25
+ event?: FormEvent<HTMLInputElement> | ClipboardEvent<HTMLInputElement>,
26
+ ) => void;
14
27
  }
15
28
 
16
29
  export const SmartInput = forwardRef<HTMLInputElement, ISmartInputProps>(
17
30
  (
18
31
  {
19
- onChange,
20
- isUpperCase = false,
32
+ value = '',
21
33
  smartType = 'default',
22
34
  regExp,
23
- value = '',
24
35
  maxLength,
36
+ isUpperCase = false,
37
+ isTransliterationEnabled = true,
38
+ onChange,
25
39
  ...rest
26
40
  },
27
41
  ref,
28
42
  ) => {
29
- const [currentValue, setCurrentValue] = useState<string>(getUpperCaseIfNeeded(value));
30
- const [caretPosition, setCaretPosition] = useState<number | null>(null);
31
- const [input, setInput] = useState<HTMLInputElement | null>(null);
43
+ const inputRef = useRef<HTMLInputElement>(null);
44
+ const mergedRef = useMergedRefs([ref, inputRef]);
45
+
32
46
  const regex = regExp || SMART_INPUT_REGEX_MAP[smartType];
33
47
 
34
- useEffect(() => {
35
- if (input && input.type !== 'email' && input.selectionStart !== caretPosition) {
36
- input.selectionStart = caretPosition;
37
- input.selectionEnd = caretPosition;
38
- }
39
- }, [caretPosition]);
40
-
41
- useEffect(() => {
42
- setCurrentValue(getUpperCaseIfNeeded(value));
43
- }, [value]);
44
-
45
- function getUpperCaseIfNeeded(str: string) {
46
- return isUpperCase ? str.toUpperCase() : str;
47
- }
48
-
49
- const handleChange = (str: string, event?: FormEvent<HTMLInputElement>) => {
50
- const mappedValue = str
51
- .split('')
52
- .map((symbol) =>
53
- regex.test(symbol)
54
- ? symbol
55
- : transformCaseSensitive(
56
- smartType,
57
- smartType !== 'email' ? CharactersMap : { ...CharactersMap, '"': '@' },
58
- symbol,
59
- ),
60
- )
61
- .filter((symbol) => regex.test(symbol))
62
- .join('');
63
- const domElement = event?.currentTarget;
64
-
65
- if (domElement) {
66
- if (!input) {
67
- setInput(domElement);
68
- }
69
-
70
- setCurrentValue(getUpperCaseIfNeeded(mappedValue));
71
- onChange(getUpperCaseIfNeeded(mappedValue));
72
-
73
- if (mappedValue !== currentValue) {
74
- setCaretPosition(domElement.selectionStart);
75
- } else {
76
- setCaretPosition(domElement.selectionStart ? domElement.selectionStart - 1 : null);
77
- }
48
+ const getUpperCaseIfNeeded = (str: string) => (isUpperCase ? str.toUpperCase() : str);
49
+
50
+ const updateCaretPosition = (position: number | null, newValue: string) => {
51
+ const input = inputRef.current;
52
+ if (isNotEmpty(input) && input.type !== 'email') {
53
+ // Нужно для того, чтобы после ререндера позиция каретки не сбросилась
54
+ input.value = newValue;
55
+ input.setSelectionRange(position, position);
78
56
  }
79
57
  };
80
58
 
81
- const handlePaste = (event: ClipboardEvent<HTMLInputElement>) => {
82
- const str = event.clipboardData.getData('text/plain').split('').join('');
83
- const domElement = event.currentTarget;
59
+ const handleChange = async (str: string, event: FormEvent<HTMLInputElement>) => {
60
+ const isValid = regex.test(str);
84
61
 
85
- if (!input) {
86
- setInput(domElement);
62
+ const charactersMap = smartType === 'email' ? EmailCharactersMap : CharactersMap;
63
+ const newValue = getUpperCaseIfNeeded(
64
+ isValid ? str : mapSymbols(str, regex, charactersMap, isTransliterationEnabled),
65
+ );
66
+
67
+ onChange(newValue, { rawValue: str, isValid }, event);
68
+
69
+ const input = inputRef.current;
70
+ if (isNotEmpty(input)) {
71
+ const start = input.selectionStart;
72
+ updateCaretPosition(isEmpty(start) || newValue !== value ? start : start - 1, newValue);
87
73
  }
74
+ };
88
75
 
76
+ const handlePaste = async (event: ClipboardEvent<HTMLInputElement>) => {
89
77
  event.preventDefault();
90
- const selectionStart = domElement.selectionStart ?? 0;
91
- const selectionEnd = domElement.selectionEnd ?? 0;
92
-
93
- let mappedValue = str
94
- .split('')
95
- .map((symbol) =>
96
- regex.test(symbol)
97
- ? symbol
98
- : transformCaseSensitive(smartType, TransliterationMap, symbol),
99
- )
100
- .filter((letter) => regex.test(letter))
101
- .join('');
102
-
103
- const newValueLength =
104
- mappedValue.length + currentValue.length - (selectionEnd - selectionStart);
105
-
106
- if (maxLength !== undefined && maxLength >= 0 && newValueLength > maxLength) {
107
- const validMappedValueLength = mappedValue.length - (newValueLength - maxLength);
108
-
109
- mappedValue = mappedValue.substring(0, validMappedValueLength);
78
+
79
+ const str = event.clipboardData.getData('text/plain').split('').join('');
80
+
81
+ const input = event.currentTarget;
82
+ const selectionStart = input.selectionStart ?? 0;
83
+ const selectionEnd = input.selectionEnd ?? 0;
84
+
85
+ const isValid = regex.test(str);
86
+
87
+ let mappedValue = isValid
88
+ ? str
89
+ : mapSymbols(str, regex, TransliterationMap, isTransliterationEnabled);
90
+
91
+ const newValueLength = mappedValue.length + value.length - (selectionEnd - selectionStart);
92
+
93
+ if (isNotEmpty(maxLength) && maxLength >= 0 && newValueLength > maxLength) {
94
+ mappedValue = mappedValue.substring(0, mappedValue.length - (newValueLength - maxLength));
110
95
  }
111
96
 
112
97
  const newValue = getUpperCaseIfNeeded(
113
- `${currentValue?.substring(0, selectionStart)}${mappedValue}${currentValue?.substring(
114
- selectionEnd,
115
- )}`,
98
+ `${value.substring(0, selectionStart)}${mappedValue}${value.substring(selectionEnd)}`,
116
99
  );
117
100
 
118
- setCaretPosition(selectionStart + mappedValue.length);
119
- setCurrentValue(newValue);
120
- onChange(newValue);
101
+ onChange(newValue, { rawValue: str, isValid }, event);
102
+ updateCaretPosition(selectionStart + mappedValue.length, newValue);
121
103
  };
122
104
 
123
105
  return (
124
106
  <Input
125
107
  {...rest}
126
- ref={ref}
108
+ ref={mergedRef}
109
+ value={value}
127
110
  maxLength={maxLength}
128
- onChange={handleChange}
129
111
  onPaste={handlePaste}
130
- value={currentValue}
112
+ onChange={handleChange}
131
113
  />
132
114
  );
133
115
  },