@true-engineering/true-react-common-ui-kit 3.25.1 → 3.26.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 (93) hide show
  1. package/LICENSE +201 -201
  2. package/README.md +11 -0
  3. package/dist/components/Button/Button.d.ts +2 -2
  4. package/dist/components/WithPopup/WithPopup.d.ts +13 -14
  5. package/dist/components/WithPopup/WithPopup.styles.d.ts +1 -1
  6. package/dist/components/WithPopup/types.d.ts +11 -0
  7. package/dist/true-react-common-ui-kit.js +156 -105
  8. package/dist/true-react-common-ui-kit.js.map +1 -1
  9. package/dist/true-react-common-ui-kit.umd.cjs +154 -103
  10. package/dist/true-react-common-ui-kit.umd.cjs.map +1 -1
  11. package/dist/types.d.ts +20 -0
  12. package/package.json +1 -1
  13. package/src/components/AccountInfo/AccountInfo.stories.tsx +32 -32
  14. package/src/components/AccountInfo/AccountInfo.tsx +80 -80
  15. package/src/components/AddButton/AddButton.stories.tsx +21 -21
  16. package/src/components/AddButton/AddButton.tsx +52 -52
  17. package/src/components/Button/Button.tsx +129 -129
  18. package/src/components/Colors/Colors.stories.tsx +7 -7
  19. package/src/components/DateInput/DateInput.tsx +90 -90
  20. package/src/components/DateInput/constants.ts +2 -2
  21. package/src/components/Description/Description.stories.tsx +27 -27
  22. package/src/components/Description/Description.tsx +61 -61
  23. package/src/components/FiltersPane/components/FilterValueView/FilterValueView.tsx +166 -166
  24. package/src/components/FiltersPane/components/FilterWithDates/FilterWithDates.tsx +210 -210
  25. package/src/components/FiltersPane/components/FilterWithPeriod/FilterWithPeriod.tsx +177 -177
  26. package/src/components/Flag/Flag.stories.tsx +29 -29
  27. package/src/components/Flag/Flag.tsx +26 -26
  28. package/src/components/Flag/augment.d.ts +1 -1
  29. package/src/components/FlexibleTable/components/FlexibleTableCell/FlexibleTableCell.styles.ts +38 -38
  30. package/src/components/FlexibleTable/components/FlexibleTableRow/FlexibleTableRow.styles.ts +25 -25
  31. package/src/components/FlexibleTable/helpers.ts +13 -13
  32. package/src/components/Icon/Icon.stories.tsx +86 -86
  33. package/src/components/Icon/complexIcons/augment.d.ts +1 -1
  34. package/src/components/Icon/complexIcons/avatarGreen.svg +57 -57
  35. package/src/components/Icon/complexIcons/index.ts +1 -1
  36. package/src/components/IncrementInput/IncrementInput.tsx +105 -105
  37. package/src/components/Input/Input.tsx +297 -297
  38. package/src/components/Input/types.ts +32 -32
  39. package/src/components/List/List.stories.tsx +70 -70
  40. package/src/components/List/List.tsx +33 -33
  41. package/src/components/List/components/ListItem/ListItem.tsx +57 -57
  42. package/src/components/Modal/Modal.stories.tsx +105 -105
  43. package/src/components/MultiSelect/MultiSelect.stories.tsx +46 -46
  44. package/src/components/MultiSelect/MultiSelect.tsx +106 -106
  45. package/src/components/MultiSelect/components/MultiSelectInput/MultiSelectInput.tsx +53 -53
  46. package/src/components/NewMoreMenu/NewMoreMenu.tsx +8 -9
  47. package/src/components/Notification/Notification.stories.tsx +46 -46
  48. package/src/components/Notification/Notification.tsx +69 -69
  49. package/src/components/NumberInput/NumberInput.tsx +137 -137
  50. package/src/components/NumberInput/index.ts +1 -1
  51. package/src/components/PhoneInput/PhoneInput.tsx +214 -214
  52. package/src/components/PhoneInput/components/PhoneInputCountryList/PhoneInputCountryList.tsx +155 -155
  53. package/src/components/PhoneInput/types.ts +16 -16
  54. package/src/components/RadioButton/RadioButton.stories.tsx +46 -46
  55. package/src/components/RadioButton/RadioButton.tsx +57 -57
  56. package/src/components/ScrollIntoViewIfNeeded/index.ts +1 -1
  57. package/src/components/Select/Select.stories.tsx +235 -235
  58. package/src/components/Select/constants.ts +2 -2
  59. package/src/components/Select/types.ts +1 -1
  60. package/src/components/Selector/Selector.stories.tsx +62 -62
  61. package/src/components/Selector/Selector.tsx +115 -115
  62. package/src/components/Selector/index.ts +2 -2
  63. package/src/components/Selector/types.ts +12 -12
  64. package/src/components/Skeleton/Skeleton.stories.tsx +19 -19
  65. package/src/components/SmartInput/SmartInput.tsx +134 -134
  66. package/src/components/Status/Status.stories.tsx +73 -73
  67. package/src/components/Status/Status.styles.ts +143 -143
  68. package/src/components/Status/Status.tsx +49 -49
  69. package/src/components/Status/constants.ts +11 -11
  70. package/src/components/Status/index.ts +3 -3
  71. package/src/components/Status/types.ts +5 -5
  72. package/src/components/Switch/Switch.stories.tsx +40 -40
  73. package/src/components/Switch/Switch.tsx +75 -75
  74. package/src/components/TextWithInfo/TextWithInfo.stories.tsx +53 -53
  75. package/src/components/TextWithInfo/TextWithInfo.tsx +62 -62
  76. package/src/components/TextWithTooltip/TextWithTooltip.stories.tsx +58 -58
  77. package/src/components/ThemedPreloader/ThemedPreloader.stories.tsx +41 -41
  78. package/src/components/ThemedPreloader/ThemedPreloader.tsx +54 -54
  79. package/src/components/ThemedPreloader/components/DefaultPreloader/index.ts +1 -1
  80. package/src/components/Toaster/Toaster.stories.tsx +30 -30
  81. package/src/components/Tooltip/Tooltip.stories.tsx +19 -19
  82. package/src/components/Tooltip/Tooltip.tsx +35 -35
  83. package/src/components/Tooltip/types.ts +1 -1
  84. package/src/components/WithPopup/WithPopup.stories.tsx +36 -15
  85. package/src/components/WithPopup/WithPopup.styles.ts +6 -8
  86. package/src/components/WithPopup/WithPopup.tsx +64 -35
  87. package/src/components/WithPopup/types.ts +14 -0
  88. package/src/helpers/popper-helpers.ts +17 -17
  89. package/src/hooks/use-dropdown.ts +84 -84
  90. package/src/hooks/use-is-mounted.ts +15 -15
  91. package/src/theme/helpers.ts +76 -76
  92. package/src/types.ts +28 -0
  93. package/src/vite-env.d.ts +1 -1
@@ -1,129 +1,129 @@
1
- import { ReactNode, ButtonHTMLAttributes, MouseEvent, forwardRef } from 'react';
2
- import clsx from 'clsx';
3
- import { addDataTestId, isReactNodeNotEmpty } from '@true-engineering/true-react-platform-helpers';
4
- import { addDataAttributes } from '../../helpers';
5
- import { useTweakStyles } from '../../hooks';
6
- import { ICommonProps } from '../../types';
7
- import { renderIcon, IIcon } from '../Icon';
8
- import { ThemedPreloader, IThemedPreloaderProps } from '../ThemedPreloader';
9
- import { IButtonSize, IButtonView } from './types';
10
- import { useStyles, IButtonStyles, getPreloaderStyles } from './Button.styles';
11
-
12
- export interface IButtonProps extends ICommonProps<IButtonStyles> {
13
- children?: ReactNode;
14
- /** @default 'button' */
15
- type?: ButtonHTMLAttributes<unknown>['type'];
16
- /** @default 'l' */
17
- size?: IButtonSize;
18
- /** @default 'primary' */
19
- view?: IButtonView;
20
- /** @default false */
21
- shouldSkipTabNavigation?: boolean;
22
- /** @default false */
23
- isDisabled?: boolean;
24
- /** @default false */
25
- isFullWidth?: boolean;
26
- /**
27
- * Нужно ли убирать скругление у кнопки слева
28
- * @default false
29
- */
30
- isInline?: boolean;
31
- /**
32
- * Нужно ли показать лоадер и заблокировать кнопку
33
- * @default false
34
- */
35
- isLoading?: boolean;
36
- /** @default false */
37
- isActive?: boolean;
38
- icon?: IIcon;
39
- /** @default 'left' */
40
- iconPosition?: 'left' | 'right';
41
- /** @default 'dots' */
42
- preloaderType?: IThemedPreloaderProps['type'];
43
- onClick?: (event: MouseEvent<HTMLButtonElement>) => void | Promise<void>;
44
- onMouseDown?: (event: MouseEvent<HTMLButtonElement>) => void | Promise<void>;
45
- }
46
-
47
- export const Button = forwardRef<HTMLButtonElement, IButtonProps>(
48
- (
49
- {
50
- type = 'button',
51
- children,
52
- size = 'l',
53
- view = 'primary',
54
- isFullWidth = false,
55
- isInline = false,
56
- isDisabled = false,
57
- isActive = false,
58
- isLoading = false,
59
- shouldSkipTabNavigation = false,
60
- data,
61
- testId,
62
- tweakStyles,
63
- icon,
64
- iconPosition = 'left',
65
- preloaderType = 'dots',
66
- onClick,
67
- onMouseDown,
68
- },
69
- ref,
70
- ) => {
71
- const classes = useStyles({ theme: tweakStyles });
72
-
73
- const tweakPreloaderStyles = useTweakStyles({
74
- innerStyles: getPreloaderStyles(size),
75
- tweakStyles,
76
- className: 'tweakPreloader',
77
- currentComponentName: 'Button',
78
- });
79
-
80
- const hasIcon = isReactNodeNotEmpty(icon);
81
- const hasChildren = isReactNodeNotEmpty(children);
82
- const hasNoAction = isDisabled || isLoading;
83
-
84
- return (
85
- <button
86
- ref={ref}
87
- type={type}
88
- className={clsx(classes.root, classes[size], classes[view], {
89
- [classes.disabled]: isDisabled,
90
- [classes.fullWidth]: isFullWidth,
91
- [classes.inline]: isInline,
92
- [classes.active]: isActive,
93
- [classes.loading]: isLoading,
94
- [classes.onlyIcon]: hasIcon && !hasChildren,
95
- })}
96
- tabIndex={shouldSkipTabNavigation ? -1 : undefined}
97
- disabled={hasNoAction}
98
- onClick={!hasNoAction ? onClick : undefined}
99
- onMouseDown={!hasNoAction ? onMouseDown : undefined}
100
- {...addDataTestId(testId)}
101
- {...addDataAttributes(data)}
102
- >
103
- <span
104
- className={clsx(classes.content, {
105
- [classes.iconFromRight]: hasChildren && hasIcon && iconPosition === 'right',
106
- [classes.iconFromLeft]: hasChildren && hasIcon && iconPosition === 'left',
107
- })}
108
- >
109
- {hasIcon && (
110
- <span className={classes.icon}>{isReactNodeNotEmpty(icon) && renderIcon(icon)}</span>
111
- )}
112
- {hasChildren && (
113
- <span className={clsx(classes.children, hasIcon && classes.withIcon)}>{children}</span>
114
- )}
115
- </span>
116
-
117
- {isLoading && (
118
- <span className={classes.loader}>
119
- <ThemedPreloader
120
- type={preloaderType}
121
- useCurrentColor
122
- tweakStyles={tweakPreloaderStyles}
123
- />
124
- </span>
125
- )}
126
- </button>
127
- );
128
- },
129
- );
1
+ import { ReactNode, ButtonHTMLAttributes, MouseEvent, forwardRef } from 'react';
2
+ import clsx from 'clsx';
3
+ import { addDataTestId, isReactNodeNotEmpty } from '@true-engineering/true-react-platform-helpers';
4
+ import { addDataAttributes } from '../../helpers';
5
+ import { useTweakStyles } from '../../hooks';
6
+ import { ICommonProps, IDomElementInteractions } from '../../types';
7
+ import { renderIcon, IIcon } from '../Icon';
8
+ import { ThemedPreloader, IThemedPreloaderProps } from '../ThemedPreloader';
9
+ import { IButtonSize, IButtonView } from './types';
10
+ import { useStyles, IButtonStyles, getPreloaderStyles } from './Button.styles';
11
+
12
+ export interface IButtonProps
13
+ extends ICommonProps<IButtonStyles>,
14
+ IDomElementInteractions<HTMLButtonElement> {
15
+ children?: ReactNode;
16
+ /** @default 'button' */
17
+ type?: ButtonHTMLAttributes<unknown>['type'];
18
+ /** @default 'l' */
19
+ size?: IButtonSize;
20
+ /** @default 'primary' */
21
+ view?: IButtonView;
22
+ /** @default false */
23
+ shouldSkipTabNavigation?: boolean;
24
+ /** @default false */
25
+ isDisabled?: boolean;
26
+ /** @default false */
27
+ isFullWidth?: boolean;
28
+ /**
29
+ * Нужно ли убирать скругление у кнопки слева
30
+ * @default false
31
+ */
32
+ isInline?: boolean;
33
+ /**
34
+ * Нужно ли показать лоадер и заблокировать кнопку
35
+ * @default false
36
+ */
37
+ isLoading?: boolean;
38
+ /** @default false */
39
+ isActive?: boolean;
40
+ icon?: IIcon;
41
+ /** @default 'left' */
42
+ iconPosition?: 'left' | 'right';
43
+ /** @default 'dots' */
44
+ preloaderType?: IThemedPreloaderProps['type'];
45
+ onClick?: (event: MouseEvent<HTMLButtonElement>) => void | Promise<void>;
46
+ onMouseDown?: (event: MouseEvent<HTMLButtonElement>) => void | Promise<void>;
47
+ }
48
+
49
+ export const Button = forwardRef<HTMLButtonElement, IButtonProps>(
50
+ (
51
+ {
52
+ type = 'button',
53
+ children,
54
+ size = 'l',
55
+ view = 'primary',
56
+ isFullWidth = false,
57
+ isInline = false,
58
+ isDisabled = false,
59
+ isActive = false,
60
+ isLoading = false,
61
+ shouldSkipTabNavigation = false,
62
+ data,
63
+ testId,
64
+ tweakStyles,
65
+ icon,
66
+ iconPosition = 'left',
67
+ preloaderType = 'dots',
68
+ ...interactions
69
+ },
70
+ ref,
71
+ ) => {
72
+ const classes = useStyles({ theme: tweakStyles });
73
+
74
+ const tweakPreloaderStyles = useTweakStyles({
75
+ innerStyles: getPreloaderStyles(size),
76
+ tweakStyles,
77
+ className: 'tweakPreloader',
78
+ currentComponentName: 'Button',
79
+ });
80
+
81
+ const hasIcon = isReactNodeNotEmpty(icon);
82
+ const hasChildren = isReactNodeNotEmpty(children);
83
+ const hasNoAction = isDisabled || isLoading;
84
+
85
+ return (
86
+ <button
87
+ ref={ref}
88
+ type={type}
89
+ className={clsx(classes.root, classes[size], classes[view], {
90
+ [classes.disabled]: isDisabled,
91
+ [classes.fullWidth]: isFullWidth,
92
+ [classes.inline]: isInline,
93
+ [classes.active]: isActive,
94
+ [classes.loading]: isLoading,
95
+ [classes.onlyIcon]: hasIcon && !hasChildren,
96
+ })}
97
+ tabIndex={shouldSkipTabNavigation ? -1 : undefined}
98
+ disabled={hasNoAction}
99
+ {...interactions}
100
+ {...addDataTestId(testId)}
101
+ {...addDataAttributes(data)}
102
+ >
103
+ <span
104
+ className={clsx(classes.content, {
105
+ [classes.iconFromRight]: hasChildren && hasIcon && iconPosition === 'right',
106
+ [classes.iconFromLeft]: hasChildren && hasIcon && iconPosition === 'left',
107
+ })}
108
+ >
109
+ {hasIcon && (
110
+ <span className={classes.icon}>{isReactNodeNotEmpty(icon) && renderIcon(icon)}</span>
111
+ )}
112
+ {hasChildren && (
113
+ <span className={clsx(classes.children, hasIcon && classes.withIcon)}>{children}</span>
114
+ )}
115
+ </span>
116
+
117
+ {isLoading && (
118
+ <span className={classes.loader}>
119
+ <ThemedPreloader
120
+ type={preloaderType}
121
+ useCurrentColor
122
+ tweakStyles={tweakPreloaderStyles}
123
+ />
124
+ </span>
125
+ )}
126
+ </button>
127
+ );
128
+ },
129
+ );
@@ -1,7 +1,7 @@
1
- import { Colors } from './Colors';
2
-
3
- export default {
4
- title: 'Data Display/Colors',
5
- };
6
-
7
- export const Default = (): JSX.Element => <Colors />;
1
+ import { Colors } from './Colors';
2
+
3
+ export default {
4
+ title: 'Data Display/Colors',
5
+ };
6
+
7
+ export const Default = (): JSX.Element => <Colors />;
@@ -1,90 +1,90 @@
1
- import { FormEvent, MouseEvent, forwardRef, ChangeEvent } from 'react';
2
- import clsx from 'clsx';
3
- import { addDataAttributes } from '../../helpers';
4
- import { useTweakStyles } from '../../hooks';
5
- import { ICommonProps } from '../../types';
6
- import { IInputProps, Input } from '../Input';
7
- import { EMPTY_DATE_INPUT_VALUE, EMPTY_DATE_RANGE_INPUT_VALUE } from './constants';
8
- import { useStyles, IDateInputStyles } from './DateInput.styles';
9
-
10
- export interface IDateInputProps
11
- extends Omit<IInputProps, 'value' | 'beforeMaskedStateChange' | 'onChange' | 'tweakStyles'>,
12
- ICommonProps<IDateInputStyles> {
13
- date?: string;
14
- /** @default '' */
15
- startDate?: string;
16
- /** @default '' */
17
- endDate?: string;
18
- className?: string;
19
- /** @default false */
20
- isRange?: boolean;
21
- onClick?: (event: MouseEvent<HTMLDivElement>) => void;
22
- // react-datepicker ожидает event первым аргументом
23
- onChange?: (event: FormEvent<HTMLInputElement>, value: string) => void;
24
- }
25
-
26
- export const DateInput = forwardRef<HTMLInputElement, IDateInputProps>(
27
- (
28
- {
29
- date,
30
- startDate = '',
31
- endDate = '',
32
- mask,
33
- className,
34
- placeholder,
35
- data,
36
- isRange = false,
37
- tweakStyles,
38
- onClick,
39
- onChange,
40
- ...inputProps
41
- },
42
- ref,
43
- ) => {
44
- const classes = useStyles({ theme: tweakStyles });
45
-
46
- const tweakInputStyles = useTweakStyles({
47
- tweakStyles,
48
- className: 'tweakInput',
49
- currentComponentName: 'DateInput',
50
- });
51
-
52
- const beforeMaskedStateChange: IInputProps['beforeMaskedStateChange'] = ({ nextState }) => {
53
- // Если в инпуте отсутствует введенное значение, то выставляем
54
- // пустую строку для корректной работы react-datepicker
55
- if (
56
- nextState.value === EMPTY_DATE_INPUT_VALUE ||
57
- nextState.value === EMPTY_DATE_RANGE_INPUT_VALUE
58
- ) {
59
- return { value: '', selection: { start: 0, end: 0 } };
60
- }
61
- return nextState;
62
- };
63
-
64
- const handleChange: IInputProps['onChange'] = (value, event) => {
65
- // Событие click срабатывает только при нажатии на кнопку очистки
66
- if (event.type === 'click') {
67
- // react-datepicker ожидает пустую строку в event.target.value
68
- (event as ChangeEvent<HTMLInputElement>).target.value = '';
69
- }
70
- onChange?.(event, value);
71
- };
72
-
73
- return (
74
- <div className={clsx(classes.root, className)} onClick={onClick} {...addDataAttributes(data)}>
75
- <Input
76
- {...inputProps}
77
- ref={ref}
78
- value={isRange ? `${startDate}${endDate}` : date}
79
- mask={mask ?? (isRange ? '99.99.9999 - 99.99.9999' : '99.99.9999')}
80
- placeholder={
81
- placeholder ?? (isRange ? EMPTY_DATE_RANGE_INPUT_VALUE : EMPTY_DATE_INPUT_VALUE)
82
- }
83
- tweakStyles={tweakInputStyles}
84
- onChange={handleChange}
85
- beforeMaskedStateChange={beforeMaskedStateChange}
86
- />
87
- </div>
88
- );
89
- },
90
- );
1
+ import { FormEvent, MouseEvent, forwardRef, ChangeEvent } from 'react';
2
+ import clsx from 'clsx';
3
+ import { addDataAttributes } from '../../helpers';
4
+ import { useTweakStyles } from '../../hooks';
5
+ import { ICommonProps } from '../../types';
6
+ import { IInputProps, Input } from '../Input';
7
+ import { EMPTY_DATE_INPUT_VALUE, EMPTY_DATE_RANGE_INPUT_VALUE } from './constants';
8
+ import { useStyles, IDateInputStyles } from './DateInput.styles';
9
+
10
+ export interface IDateInputProps
11
+ extends Omit<IInputProps, 'value' | 'beforeMaskedStateChange' | 'onChange' | 'tweakStyles'>,
12
+ ICommonProps<IDateInputStyles> {
13
+ date?: string;
14
+ /** @default '' */
15
+ startDate?: string;
16
+ /** @default '' */
17
+ endDate?: string;
18
+ className?: string;
19
+ /** @default false */
20
+ isRange?: boolean;
21
+ onClick?: (event: MouseEvent<HTMLDivElement>) => void;
22
+ // react-datepicker ожидает event первым аргументом
23
+ onChange?: (event: FormEvent<HTMLInputElement>, value: string) => void;
24
+ }
25
+
26
+ export const DateInput = forwardRef<HTMLInputElement, IDateInputProps>(
27
+ (
28
+ {
29
+ date,
30
+ startDate = '',
31
+ endDate = '',
32
+ mask,
33
+ className,
34
+ placeholder,
35
+ data,
36
+ isRange = false,
37
+ tweakStyles,
38
+ onClick,
39
+ onChange,
40
+ ...inputProps
41
+ },
42
+ ref,
43
+ ) => {
44
+ const classes = useStyles({ theme: tweakStyles });
45
+
46
+ const tweakInputStyles = useTweakStyles({
47
+ tweakStyles,
48
+ className: 'tweakInput',
49
+ currentComponentName: 'DateInput',
50
+ });
51
+
52
+ const beforeMaskedStateChange: IInputProps['beforeMaskedStateChange'] = ({ nextState }) => {
53
+ // Если в инпуте отсутствует введенное значение, то выставляем
54
+ // пустую строку для корректной работы react-datepicker
55
+ if (
56
+ nextState.value === EMPTY_DATE_INPUT_VALUE ||
57
+ nextState.value === EMPTY_DATE_RANGE_INPUT_VALUE
58
+ ) {
59
+ return { value: '', selection: { start: 0, end: 0 } };
60
+ }
61
+ return nextState;
62
+ };
63
+
64
+ const handleChange: IInputProps['onChange'] = (value, event) => {
65
+ // Событие click срабатывает только при нажатии на кнопку очистки
66
+ if (event.type === 'click') {
67
+ // react-datepicker ожидает пустую строку в event.target.value
68
+ (event as ChangeEvent<HTMLInputElement>).target.value = '';
69
+ }
70
+ onChange?.(event, value);
71
+ };
72
+
73
+ return (
74
+ <div className={clsx(classes.root, className)} onClick={onClick} {...addDataAttributes(data)}>
75
+ <Input
76
+ {...inputProps}
77
+ ref={ref}
78
+ value={isRange ? `${startDate}${endDate}` : date}
79
+ mask={mask ?? (isRange ? '99.99.9999 - 99.99.9999' : '99.99.9999')}
80
+ placeholder={
81
+ placeholder ?? (isRange ? EMPTY_DATE_RANGE_INPUT_VALUE : EMPTY_DATE_INPUT_VALUE)
82
+ }
83
+ tweakStyles={tweakInputStyles}
84
+ onChange={handleChange}
85
+ beforeMaskedStateChange={beforeMaskedStateChange}
86
+ />
87
+ </div>
88
+ );
89
+ },
90
+ );
@@ -1,2 +1,2 @@
1
- export const EMPTY_DATE_INPUT_VALUE = '__.__.____';
2
- export const EMPTY_DATE_RANGE_INPUT_VALUE = `${EMPTY_DATE_INPUT_VALUE} - ${EMPTY_DATE_INPUT_VALUE}`;
1
+ export const EMPTY_DATE_INPUT_VALUE = '__.__.____';
2
+ export const EMPTY_DATE_RANGE_INPUT_VALUE = `${EMPTY_DATE_INPUT_VALUE} - ${EMPTY_DATE_INPUT_VALUE}`;
@@ -1,27 +1,27 @@
1
- import { ComponentStory } from '@storybook/react';
2
- import { Description } from './Description';
3
-
4
- export default {
5
- title: 'Data Display/Description',
6
- component: Description,
7
- argTypes: {
8
- truncateIndex: {
9
- control: { type: 'range', min: 20, max: 600, step: 10 },
10
- },
11
- },
12
- };
13
-
14
- const lorem =
15
- 'Lorem ipsum dolor, sit amet consectetur adipisicing. Alias necessitatibus quidem nobis molestiae dolor sed at, voluptatum nulla qui enim maiores consectetur dolorum atque odit id ratione pariatur ut nihil. http://google.com';
16
-
17
- const Template: ComponentStory<typeof Description> = (args) => <Description {...args} />;
18
-
19
- export const Default = Template.bind({});
20
-
21
- Default.args = {
22
- text: lorem,
23
- moreTitle: 'подробнее',
24
- lessTitle: 'скрыть',
25
- truncateIndex: 150,
26
- isAlwaysOpen: false,
27
- };
1
+ import { ComponentStory } from '@storybook/react';
2
+ import { Description } from './Description';
3
+
4
+ export default {
5
+ title: 'Data Display/Description',
6
+ component: Description,
7
+ argTypes: {
8
+ truncateIndex: {
9
+ control: { type: 'range', min: 20, max: 600, step: 10 },
10
+ },
11
+ },
12
+ };
13
+
14
+ const lorem =
15
+ 'Lorem ipsum dolor, sit amet consectetur adipisicing. Alias necessitatibus quidem nobis molestiae dolor sed at, voluptatum nulla qui enim maiores consectetur dolorum atque odit id ratione pariatur ut nihil. http://google.com';
16
+
17
+ const Template: ComponentStory<typeof Description> = (args) => <Description {...args} />;
18
+
19
+ export const Default = Template.bind({});
20
+
21
+ Default.args = {
22
+ text: lorem,
23
+ moreTitle: 'подробнее',
24
+ lessTitle: 'скрыть',
25
+ truncateIndex: 150,
26
+ isAlwaysOpen: false,
27
+ };
@@ -1,61 +1,61 @@
1
- import { FC, useState } from 'react';
2
- import { addDataTestId } from '@true-engineering/true-react-platform-helpers';
3
- import { addDataAttributes } from '../../helpers';
4
- import { ICommonProps } from '../../types';
5
- import { LINK_REGEXP } from './constants';
6
- import { useStyles, IDescriptionStyles } from './Description.styles';
7
-
8
- export interface IDescriptionProps extends ICommonProps<IDescriptionStyles> {
9
- text: string;
10
- moreTitle?: string;
11
- lessTitle?: string;
12
- /** @default 150 */
13
- truncateIndex?: number;
14
- /** @default true */
15
- isAlwaysOpen?: boolean;
16
- }
17
-
18
- export const Description: FC<IDescriptionProps> = ({
19
- text,
20
- moreTitle,
21
- lessTitle,
22
- truncateIndex = 150,
23
- isAlwaysOpen = true,
24
- testId,
25
- data,
26
- tweakStyles,
27
- }) => {
28
- const classes = useStyles({ theme: tweakStyles });
29
-
30
- const isTooShort = text.length < truncateIndex;
31
- const [isOpen, setIsOpen] = useState(isAlwaysOpen);
32
- const link = text.match(LINK_REGEXP);
33
- const linkText = link && link[0];
34
- const textWithNoLink = text.replace(LINK_REGEXP, '');
35
-
36
- const shortText = text.slice(0, text.slice(0, truncateIndex).lastIndexOf(' '));
37
-
38
- return (
39
- <div className={classes.root} {...addDataTestId(testId)} {...addDataAttributes(data)}>
40
- {isAlwaysOpen || isOpen || isTooShort ? (
41
- <div>
42
- <span>{textWithNoLink} </span>
43
- {!!linkText && (
44
- <a className={classes.link} href={linkText} target="_blank" rel="noopener noreferrer">
45
- {linkText}
46
- </a>
47
- )}
48
- </div>
49
- ) : (
50
- <div>
51
- <span>{shortText}</span>
52
- </div>
53
- )}
54
- {isAlwaysOpen || isTooShort ? undefined : (
55
- <div className={classes.button} onClick={() => setIsOpen(!isOpen)}>
56
- {isOpen ? lessTitle : moreTitle}
57
- </div>
58
- )}
59
- </div>
60
- );
61
- };
1
+ import { FC, useState } from 'react';
2
+ import { addDataTestId } from '@true-engineering/true-react-platform-helpers';
3
+ import { addDataAttributes } from '../../helpers';
4
+ import { ICommonProps } from '../../types';
5
+ import { LINK_REGEXP } from './constants';
6
+ import { useStyles, IDescriptionStyles } from './Description.styles';
7
+
8
+ export interface IDescriptionProps extends ICommonProps<IDescriptionStyles> {
9
+ text: string;
10
+ moreTitle?: string;
11
+ lessTitle?: string;
12
+ /** @default 150 */
13
+ truncateIndex?: number;
14
+ /** @default true */
15
+ isAlwaysOpen?: boolean;
16
+ }
17
+
18
+ export const Description: FC<IDescriptionProps> = ({
19
+ text,
20
+ moreTitle,
21
+ lessTitle,
22
+ truncateIndex = 150,
23
+ isAlwaysOpen = true,
24
+ testId,
25
+ data,
26
+ tweakStyles,
27
+ }) => {
28
+ const classes = useStyles({ theme: tweakStyles });
29
+
30
+ const isTooShort = text.length < truncateIndex;
31
+ const [isOpen, setIsOpen] = useState(isAlwaysOpen);
32
+ const link = text.match(LINK_REGEXP);
33
+ const linkText = link && link[0];
34
+ const textWithNoLink = text.replace(LINK_REGEXP, '');
35
+
36
+ const shortText = text.slice(0, text.slice(0, truncateIndex).lastIndexOf(' '));
37
+
38
+ return (
39
+ <div className={classes.root} {...addDataTestId(testId)} {...addDataAttributes(data)}>
40
+ {isAlwaysOpen || isOpen || isTooShort ? (
41
+ <div>
42
+ <span>{textWithNoLink} </span>
43
+ {!!linkText && (
44
+ <a className={classes.link} href={linkText} target="_blank" rel="noopener noreferrer">
45
+ {linkText}
46
+ </a>
47
+ )}
48
+ </div>
49
+ ) : (
50
+ <div>
51
+ <span>{shortText}</span>
52
+ </div>
53
+ )}
54
+ {isAlwaysOpen || isTooShort ? undefined : (
55
+ <div className={classes.button} onClick={() => setIsOpen(!isOpen)}>
56
+ {isOpen ? lessTitle : moreTitle}
57
+ </div>
58
+ )}
59
+ </div>
60
+ );
61
+ };