@centreon/ui 24.4.57 → 24.4.59

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 (147) hide show
  1. package/package.json +31 -38
  2. package/public/mockServiceWorker.js +1 -1
  3. package/src/Button/Icon/index.tsx +1 -1
  4. package/src/Button/Save/StartIcon.tsx +3 -3
  5. package/src/Button/Save/index.tsx +5 -9
  6. package/src/Checkbox/Checkbox.tsx +2 -2
  7. package/src/Checkbox/CheckboxGroup/index.tsx +2 -2
  8. package/src/Dashboard/Item.tsx +1 -1
  9. package/src/Dashboard/Layout.tsx +2 -2
  10. package/src/Dialog/Confirm/index.tsx +2 -10
  11. package/src/Dialog/index.tsx +2 -9
  12. package/src/FallbackPage/FallbackPage.tsx +3 -3
  13. package/src/FileDropZone/index.tsx +1 -3
  14. package/src/Form/Inputs/index.tsx +1 -3
  15. package/src/Form/Inputs/models.ts +1 -9
  16. package/src/Graph/Gauge/Gauge.tsx +1 -1
  17. package/src/Graph/HeatMap/HeatMap.stories.tsx +25 -0
  18. package/src/Graph/HeatMap/ResponsiveHeatMap.tsx +2 -8
  19. package/src/Graph/LineChart/BasicComponents/Lines/Threshold/Circle.tsx +2 -2
  20. package/src/Graph/LineChart/BasicComponents/Thresholds.tsx +2 -2
  21. package/src/Graph/LineChart/BasicComponents/useFilterLines.ts +1 -1
  22. package/src/Graph/LineChart/InteractiveComponents/AnchorPoint/GuidingLines.tsx +2 -2
  23. package/src/Graph/LineChart/InteractiveComponents/Annotations/EventAnnotations.tsx +1 -1
  24. package/src/Graph/LineChart/Legend/Legend.styles.ts +1 -1
  25. package/src/Graph/LineChart/Legend/LegendHeader.tsx +1 -1
  26. package/src/Graph/LineChart/Legend/useInteractiveValues.ts +2 -2
  27. package/src/Graph/LineChart/Legend/useLegend.ts +3 -3
  28. package/src/Graph/LineChart/helpers/doc.ts +13 -16
  29. package/src/Graph/LineChart/helpers/index.ts +1 -1
  30. package/src/Graph/LineChart/index.stories.tsx +2 -4
  31. package/src/Graph/LineChart/index.tsx +1 -1
  32. package/src/Graph/SingleBar/Thresholds.tsx +2 -2
  33. package/src/Graph/Text/Text.stories.tsx +4 -60
  34. package/src/Graph/Text/Text.tsx +1 -1
  35. package/src/Graph/common/timeSeries/index.ts +14 -22
  36. package/src/Graph/common/utils.ts +0 -19
  37. package/src/Graph/index.ts +0 -3
  38. package/src/InputField/Select/Autocomplete/Connected/index.tsx +7 -10
  39. package/src/InputField/Select/Autocomplete/Draggable/SortableList.tsx +1 -1
  40. package/src/InputField/Select/Autocomplete/Draggable/SortableListContent.tsx +1 -1
  41. package/src/InputField/Select/Autocomplete/Draggable/index.tsx +1 -1
  42. package/src/InputField/Select/Autocomplete/Multi/index.tsx +2 -4
  43. package/src/InputField/Select/Autocomplete/index.tsx +116 -129
  44. package/src/InputField/Select/IconPopover/index.tsx +2 -2
  45. package/src/InputField/Select/index.tsx +1 -12
  46. package/src/InputField/Text/index.tsx +2 -2
  47. package/src/Listing/ActionBar/index.tsx +8 -9
  48. package/src/Listing/Cell/DataCell.styles.ts +0 -3
  49. package/src/Listing/Cell/DataCell.tsx +5 -23
  50. package/src/Listing/Header/ListingHeader.tsx +1 -1
  51. package/src/Listing/Listing.cypress.spec.tsx +4 -80
  52. package/src/Listing/Listing.styles.ts +7 -4
  53. package/src/Listing/index.stories.tsx +3 -37
  54. package/src/Listing/index.test.tsx +1 -1
  55. package/src/Listing/index.tsx +3 -4
  56. package/src/Listing/models.ts +0 -1
  57. package/src/Module/index.tsx +4 -2
  58. package/src/PopoverMenu/index.tsx +5 -6
  59. package/src/RichTextEditor/RichTextEditor.tsx +1 -12
  60. package/src/SortableItems/index.tsx +7 -2
  61. package/src/ThemeProvider/index.tsx +0 -24
  62. package/src/TimePeriods/CustomTimePeriod/CompactCustomTimePeriod.styles.ts +7 -6
  63. package/src/TimePeriods/CustomTimePeriod/PopoverCustomTimePeriod/PickersStartEndDate.tsx +3 -8
  64. package/src/TimePeriods/CustomTimePeriod/PopoverCustomTimePeriod/models.ts +2 -0
  65. package/src/TimePeriods/DateTimePickerInput.tsx +19 -56
  66. package/src/TimePeriods/ResolutionTimePeriod.cypress.spec.tsx +9 -12
  67. package/src/TimePeriods/TimePeriods.cypress.spec.tsx +33 -9
  68. package/src/TimePeriods/helpers/index.ts +1 -1
  69. package/src/TimePeriods/index.stories.tsx +4 -12
  70. package/src/TimePeriods/index.tsx +2 -2
  71. package/src/Typography/FluidTypography/index.stories.tsx +2 -2
  72. package/src/Typography/FluidTypography/index.tsx +28 -21
  73. package/src/Typography/FluidTypography/useFluidResizeObserver.ts +56 -0
  74. package/src/api/QueryProvider.tsx +1 -1
  75. package/src/api/TestQueryProvider.tsx +1 -1
  76. package/src/api/index.ts +3 -3
  77. package/src/api/useFetchQuery/index.ts +23 -27
  78. package/src/api/useMutationQuery/index.test.ts +4 -4
  79. package/src/api/useMutationQuery/index.ts +25 -60
  80. package/src/components/Button/Icon/IconButton.tsx +2 -6
  81. package/src/components/DataTable/DataTable.stories.tsx +0 -40
  82. package/src/components/DataTable/DataTable.styles.ts +0 -3
  83. package/src/components/DataTable/DataTable.tsx +3 -3
  84. package/src/components/DataTable/Item/DataTableItem.styles.ts +2 -7
  85. package/src/components/DataTable/Item/DataTableItem.tsx +28 -9
  86. package/src/components/DataTable/index.ts +1 -3
  87. package/src/components/Form/AccessRights/ShareInput/ContactSwitch.tsx +3 -3
  88. package/src/components/Form/AccessRights/ShareInput/ShareInput.tsx +0 -1
  89. package/src/components/Form/Dashboard/DashboardForm.tsx +12 -15
  90. package/src/components/Layout/PageLayout/PageLayout.tsx +1 -1
  91. package/src/components/Layout/PageLayout/PageLayoutActions.tsx +0 -1
  92. package/src/components/Layout/PageLayout/PageLayoutBody.tsx +0 -1
  93. package/src/components/Layout/PageLayout/PageLayoutHeader.tsx +1 -5
  94. package/src/components/Layout/PageLayout/index.ts +1 -3
  95. package/src/components/List/Item/ListItem.tsx +3 -3
  96. package/src/components/Modal/Modal.styles.ts +3 -8
  97. package/src/components/Tooltip/ConfirmationTooltip/ConfirmationTooltip.stories.tsx +3 -3
  98. package/src/components/Tooltip/ConfirmationTooltip/ConfirmationTooltip.tsx +1 -1
  99. package/src/components/Tooltip/ConfirmationTooltip/models.ts +1 -1
  100. package/src/index.ts +2 -2
  101. package/src/queryParameters/url/index.ts +1 -5
  102. package/src/screens/dashboard/DashboardsDetail.stories.tsx +108 -0
  103. package/src/screens/dashboard/DashboardsOverview.stories.tsx +281 -0
  104. package/src/utils/index.ts +1 -1
  105. package/src/utils/useDateTimePickerAdapter.ts +309 -0
  106. package/src/utils/useFullscreen/Fullscreen.cypress.spec.tsx +3 -1
  107. package/src/utils/useFullscreen/useFullscreenListener.ts +7 -10
  108. package/src/utils/useInfiniteScrollListing.ts +1 -4
  109. package/src/utils/{useLicenseExpirationWarning.test.tsx → useLicenseExpirationWarning.cypress.spec.tsx} +37 -48
  110. package/src/utils/useLicenseExpirationWarning.ts +18 -18
  111. package/src/Form/Form.cypress.spec.tsx +0 -133
  112. package/src/Form/Inputs/List/Content.tsx +0 -62
  113. package/src/Form/Inputs/List/List.styles.ts +0 -29
  114. package/src/Form/Inputs/List/List.tsx +0 -58
  115. package/src/Form/Inputs/List/useList.ts +0 -81
  116. package/src/Graph/BarStack/BarStack.cypress.spec.tsx +0 -154
  117. package/src/Graph/BarStack/BarStack.stories.tsx +0 -123
  118. package/src/Graph/BarStack/BarStack.styles.ts +0 -37
  119. package/src/Graph/BarStack/BarStack.tsx +0 -14
  120. package/src/Graph/BarStack/ResponsiveBarStack.tsx +0 -222
  121. package/src/Graph/BarStack/index.ts +0 -1
  122. package/src/Graph/BarStack/models.ts +0 -20
  123. package/src/Graph/BarStack/useResponsiveBarStack.ts +0 -137
  124. package/src/Graph/Gauge/Gauge.cypress.spec.tsx +0 -102
  125. package/src/Graph/HeatMap/HeatMap.cypress.spec.tsx +0 -145
  126. package/src/Graph/Legend/Legend.tsx +0 -21
  127. package/src/Graph/Legend/index.ts +0 -1
  128. package/src/Graph/Legend/models.ts +0 -11
  129. package/src/Graph/PieChart/PieChart.cypress.spec.tsx +0 -169
  130. package/src/Graph/PieChart/PieChart.stories.tsx +0 -194
  131. package/src/Graph/PieChart/PieChart.styles.ts +0 -39
  132. package/src/Graph/PieChart/PieChart.tsx +0 -14
  133. package/src/Graph/PieChart/ResponsivePie.tsx +0 -254
  134. package/src/Graph/PieChart/index.ts +0 -1
  135. package/src/Graph/PieChart/models.ts +0 -20
  136. package/src/Graph/PieChart/useResponsivePie.ts +0 -85
  137. package/src/Graph/SingleBar/SingleBar.cypress.spec.tsx +0 -121
  138. package/src/Graph/Text/Text.cypress.spec.tsx +0 -101
  139. package/src/Graph/common/testUtils.ts +0 -71
  140. package/src/Graph/translatedLabels.ts +0 -1
  141. package/src/Module/Module.cypress.spec.tsx +0 -129
  142. package/src/Typography/FluidTypography/FluidTypography.cypress.spec.tsx +0 -27
  143. package/src/components/DataTable/DataListing.tsx +0 -6
  144. package/src/components/DataTable/DataTable.cypress.spec.tsx +0 -193
  145. package/src/components/Layout/PageLayout/PageQuickAccess.tsx +0 -76
  146. package/src/components/Layout/PageLayout.cypress.spec.tsx +0 -66
  147. package/src/utils/usePluralizedTranslation.ts +0 -21
@@ -110,7 +110,9 @@ describe('Fullscreen', () => {
110
110
  });
111
111
 
112
112
  ['input', 'textarea'].forEach((tag) => {
113
- it(`cannot toggle fullscreen feature using the shortcut when ${tag === 'input' ? 'an' : 'a'} ${tag} is focused`, () => {
113
+ it(`cannot toggle fullscreen feature using the shortcut when ${
114
+ tag === 'input' ? 'an' : 'a'
115
+ } ${tag} is focused`, () => {
114
116
  initialize();
115
117
 
116
118
  cy.get('#test')
@@ -39,16 +39,13 @@ export const useFullscreenListener = (): boolean => {
39
39
  resetVariables();
40
40
  };
41
41
 
42
- useEffect(
43
- () => {
44
- document.addEventListener('fullscreenchange', changeFullscreen);
45
-
46
- return () => {
47
- document.removeEventListener('fullscreenchange', changeFullscreen);
48
- };
49
- },
50
- useDeepCompare([document.fullscreenElement])
51
- );
42
+ useEffect(() => {
43
+ document.addEventListener('fullscreenchange', changeFullscreen);
44
+
45
+ return () => {
46
+ document.removeEventListener('fullscreenchange', changeFullscreen);
47
+ };
48
+ }, useDeepCompare([document.fullscreenElement]));
52
49
 
53
50
  useEffect(() => {
54
51
  window.addEventListener('keypress', toggle);
@@ -24,7 +24,6 @@ interface UseInfiniteScrollListing<T> {
24
24
  interface UseInfiniteScrollListingProps<T> {
25
25
  customQueryParameters?: Array<QueryParameter>;
26
26
  decoder?: JsonDecoder.Decoder<Listing<T>>;
27
- enabled?: boolean;
28
27
  endpoint: string;
29
28
  limit?: number;
30
29
  pageAtom: PrimitiveAtom<number>;
@@ -41,8 +40,7 @@ export const useInfiniteScrollListing = <T>({
41
40
  suspense = true,
42
41
  parameters,
43
42
  customQueryParameters,
44
- limit = 100,
45
- enabled = true
43
+ limit = 100
46
44
  }: UseInfiniteScrollListingProps<T>): UseInfiniteScrollListing<T> => {
47
45
  const [maxPage, setMaxPage] = useState(1);
48
46
 
@@ -63,7 +61,6 @@ export const useInfiniteScrollListing = <T>({
63
61
  getQueryKey: () => [queryKeyName, page],
64
62
  isPaginated: true,
65
63
  queryOptions: {
66
- enabled,
67
64
  refetchOnMount: false,
68
65
  refetchOnWindowFocus: false,
69
66
  suspense: suspense && equals(page, 1)
@@ -1,13 +1,6 @@
1
1
  import dayjs from 'dayjs';
2
- import { renderHook } from '@testing-library/react';
3
2
 
4
- import {
5
- getFetchCall,
6
- mockResponse,
7
- resetMocks,
8
- waitFor
9
- } from '../../test/testRenderer';
10
- import TestQueryProvider from '../api/TestQueryProvider';
3
+ import { TestQueryProvider, Method, SnackbarProvider } from '@centreon/ui';
11
4
 
12
5
  import { labelLicenseWarning } from './translatedLabel';
13
6
 
@@ -86,63 +79,59 @@ const getMockedResponse = (expirationDate): object => ({
86
79
  });
87
80
 
88
81
  const mockRequest = ({ expirationDate }: { expirationDate }): void => {
89
- resetMocks();
90
- mockResponse({
91
- data: getMockedResponse(expirationDate)
82
+ cy.interceptAPIRequest({
83
+ alias: 'getLicenseInformations',
84
+ method: Method.GET,
85
+ path: '**internal.php?object=centreon_module&action=list',
86
+ response: getMockedResponse(expirationDate)
92
87
  });
93
88
  };
94
89
 
95
- const showMessage = jest.fn();
90
+ const TestComponent = (): JSX.Element => {
91
+ useLicenseExpirationWarning({
92
+ module: 'centreon-autodiscovery-server'
93
+ });
96
94
 
97
- jest.mock('../Snackbar/useSnackbar', () => ({
98
- __esModule: true,
99
- default: jest
100
- .fn()
101
- .mockImplementation(() => ({ showWarningMessage: showMessage }))
102
- }));
95
+ return <div />;
96
+ };
103
97
 
104
- const initialize = (): void => {
105
- renderHook(
106
- () =>
107
- useLicenseExpirationWarning({
108
- module: 'centreon-autodiscovery-server'
109
- }),
110
- {
111
- wrapper: TestQueryProvider
112
- }
98
+ const TestWithQueryProvider = (): JSX.Element => {
99
+ return (
100
+ <TestQueryProvider>
101
+ <SnackbarProvider>
102
+ <TestComponent />
103
+ </SnackbarProvider>
104
+ </TestQueryProvider>
113
105
  );
114
106
  };
115
107
 
108
+ const initialize = (): void => {
109
+ cy.viewport('macbook-13');
110
+
111
+ cy.mount({
112
+ Component: <TestWithQueryProvider />
113
+ });
114
+ };
115
+
116
116
  const currentDate = dayjs();
117
117
 
118
118
  describe('License', () => {
119
+ beforeEach(initialize);
120
+
119
121
  it('does not display any warning message when the license expires in more than 15 days from the current date', () => {
120
122
  mockRequest({ expirationDate: currentDate.add(20, 'day') });
121
- initialize();
122
123
 
123
- waitFor(() => {
124
- expect(getFetchCall(0)).toEqual(
125
- 'internal.php?object=centreon_module&action=list'
126
- );
127
- });
124
+ cy.waitForRequest('@getLicenseInformations');
128
125
 
129
- expect(showMessage).not.toHaveBeenCalled();
126
+ cy.findByText(
127
+ labelLicenseWarning('centreon-autodiscovery-server', 20)
128
+ ).should('not.exist');
130
129
  });
131
-
132
130
  it('displays a warning message when the license expires within 15 days', () => {
133
131
  mockRequest({ expirationDate: currentDate.add(10.5, 'day') });
134
- initialize();
135
-
136
- waitFor(() => {
137
- expect(getFetchCall(0)).toEqual(
138
- 'internal.php?object=centreon_module&action=list'
139
- );
140
- });
141
-
142
- waitFor(() => {
143
- expect(showMessage).toHaveBeenCalledWith(
144
- labelLicenseWarning('centreon-autodiscovery-server', 10)
145
- );
146
- });
132
+
133
+ cy.findByText(labelLicenseWarning('centreon-autodiscovery-server', 10));
134
+
135
+ cy.makeSnapshot();
147
136
  });
148
137
  });
@@ -19,7 +19,7 @@ export const useLicenseExpirationWarning = ({ module }: Props): void => {
19
19
  const { t } = useTranslation();
20
20
  const { showWarningMessage } = useSnackbar();
21
21
 
22
- const { data } = useFetchQuery({
22
+ const { fetchQuery } = useFetchQuery({
23
23
  getEndpoint: () => extensionsEndpoint,
24
24
  getQueryKey: () => [module]
25
25
  });
@@ -28,25 +28,25 @@ export const useLicenseExpirationWarning = ({ module }: Props): void => {
28
28
 
29
29
  const getExpirationDate = pipe(
30
30
  path(['result', 'module', 'entities']),
31
- find(propEq(module, 'id')),
31
+ find(propEq('id', module)),
32
32
  path(['license', 'expiration_date'])
33
33
  ) as (data) => string;
34
34
 
35
35
  useEffect(() => {
36
- if (isNil(data)) {
37
- return;
38
- }
39
-
40
- const expirationDate = getExpirationDate(data);
41
-
42
- if (isNil(expirationDate)) {
43
- return;
44
- }
45
-
46
- const daysUntilExpiration = dayjs(expirationDate).diff(currentDate, 'day');
47
-
48
- if (lt(daysUntilExpiration, 15)) {
49
- showWarningMessage(t(labelLicenseWarning(module, daysUntilExpiration)));
50
- }
51
- }, [data]);
36
+ fetchQuery().then((response) => {
37
+ const expirationDate = getExpirationDate(response);
38
+ if (isNil(expirationDate)) {
39
+ return;
40
+ }
41
+
42
+ const daysUntilExpiration = dayjs(expirationDate).diff(
43
+ currentDate,
44
+ 'day'
45
+ );
46
+
47
+ if (lt(daysUntilExpiration, 15)) {
48
+ showWarningMessage(t(labelLicenseWarning(module, daysUntilExpiration)));
49
+ }
50
+ });
51
+ }, []);
52
52
  };
@@ -1,133 +0,0 @@
1
- import { object } from 'yup';
2
- import { faker } from '@faker-js/faker';
3
- import { useFormikContext } from 'formik';
4
-
5
- import { Typography } from '@mui/material';
6
-
7
- import { Button } from '../components';
8
-
9
- import { Form } from './Form';
10
- import { InputType } from './Inputs/models';
11
-
12
- faker.seed(42);
13
-
14
- const AddItem = ({ addItem }: { addItem: (item) => void }): JSX.Element => {
15
- const { values } = useFormikContext();
16
- const add = (): void => {
17
- addItem({
18
- alias: faker.company.name(),
19
- id: values.list.length,
20
- name: faker.person.firstName()
21
- });
22
- };
23
-
24
- return (
25
- <Button variant="ghost" onClick={add}>
26
- Add item
27
- </Button>
28
- );
29
- };
30
-
31
- const SortContent = ({
32
- name,
33
- alias
34
- }: {
35
- alias: string;
36
- name: string;
37
- }): JSX.Element => (
38
- <Typography>
39
- {name} ({alias})
40
- </Typography>
41
- );
42
-
43
- const initializeFormList = (): void => {
44
- cy.mount({
45
- Component: (
46
- <Form
47
- initialValues={{
48
- list: []
49
- }}
50
- inputs={[
51
- {
52
- fieldName: 'list',
53
- group: '',
54
- label: '',
55
- list: {
56
- AddItem,
57
- SortContent,
58
- addItemLabel: 'Add an item to the list',
59
- itemProps: ['id', 'name', 'alias'],
60
- sortLabel: 'Sort items'
61
- },
62
- type: InputType.List
63
- }
64
- ]}
65
- submit={cy.stub()}
66
- validationSchema={object()}
67
- />
68
- )
69
- });
70
- };
71
-
72
- describe('Form list', () => {
73
- beforeEach(initializeFormList);
74
-
75
- it('adds an element to the list', () => {
76
- cy.contains('Add an item to the list').should('be.visible');
77
- cy.contains('Sort items').should('be.visible');
78
-
79
- cy.contains('Add item').click();
80
-
81
- cy.findByLabelText('sort-0').should('be.visible');
82
- cy.findByLabelText('delete-0').should('be.visible');
83
- cy.contains('Christelle (Schinner - Wiegand)').should('be.visible');
84
-
85
- cy.makeSnapshot();
86
- });
87
-
88
- it('sorts elements in the list', () => {
89
- cy.contains('Add an item to the list').should('be.visible');
90
- cy.contains('Sort items').should('be.visible');
91
-
92
- cy.contains('Add item').click();
93
- cy.contains('Add item').click();
94
-
95
- cy.findByLabelText('sort-0').should('be.visible');
96
- cy.findByLabelText('delete-0').should('be.visible');
97
- cy.contains('Carley (Satterfield, Miller and Metz)').should('be.visible');
98
- cy.findByLabelText('sort-1').should('be.visible');
99
- cy.findByLabelText('delete-1').should('be.visible');
100
- cy.contains('Anderson (Crist - Bradtke)').should('be.visible');
101
-
102
- cy.moveSortableElementUsingAriaLabel({
103
- ariaLabel: 'sort-0',
104
- direction: 'down'
105
- });
106
-
107
- cy.contains('Carley (Satterfield, Miller and Metz)').should('be.visible');
108
- cy.contains('Anderson (Crist - Bradtke)').should('be.visible');
109
-
110
- cy.makeSnapshot();
111
- });
112
-
113
- it('removes an element from the list', () => {
114
- cy.contains('Add an item to the list').should('be.visible');
115
- cy.contains('Sort items').should('be.visible');
116
-
117
- cy.contains('Add item').click();
118
- cy.contains('Add item').click();
119
-
120
- cy.findByLabelText('sort-0').should('be.visible');
121
- cy.findByLabelText('delete-0').should('be.visible');
122
- cy.contains('Lea (Streich - Hartmann)').should('be.visible');
123
- cy.findByLabelText('sort-1').should('be.visible');
124
- cy.findByLabelText('delete-1').should('be.visible');
125
- cy.contains('Akeem (Quigley LLC)').should('be.visible');
126
-
127
- cy.findByLabelText('delete-0').click();
128
-
129
- cy.contains('Lea (Streich - Hartmann)').should('not.exist');
130
-
131
- cy.makeSnapshot();
132
- });
133
- });
@@ -1,62 +0,0 @@
1
- import { ReactNode } from 'react';
2
-
3
- import { DraggableSyntheticListeners } from '@dnd-kit/core';
4
-
5
- import KrilinIndicatorIcon from '@mui/icons-material/DragIndicator';
6
- import DeleteOutlineIcon from '@mui/icons-material/DeleteOutline';
7
-
8
- import { IconButton } from '../../../components';
9
-
10
- import { useListStyles } from './List.styles';
11
-
12
- export interface ContentProps {
13
- attributes;
14
- children: ReactNode;
15
- deleteItem: (id: string) => () => void;
16
- id: string;
17
- isDragging: boolean;
18
- isInDragOverlay?: boolean;
19
- itemRef: React.RefObject<HTMLDivElement>;
20
- listeners: DraggableSyntheticListeners;
21
- name: string;
22
- style;
23
- }
24
-
25
- const Content = ({
26
- listeners,
27
- itemRef,
28
- attributes,
29
- style,
30
- isDragging,
31
- id,
32
- children,
33
- deleteItem
34
- }: ContentProps): JSX.Element => {
35
- const { classes } = useListStyles();
36
-
37
- return (
38
- <div
39
- className={classes.content}
40
- ref={itemRef}
41
- {...attributes}
42
- style={style}
43
- >
44
- <IconButton
45
- data-dragging={isDragging}
46
- size="small"
47
- {...listeners}
48
- aria-label={`sort-${id}`}
49
- icon={<KrilinIndicatorIcon fontSize="small" />}
50
- />
51
- <div className={classes.innerContent}>{children}</div>
52
- <IconButton
53
- aria-label={`delete-${id}`}
54
- icon={<DeleteOutlineIcon color="error" fontSize="small" />}
55
- size="small"
56
- onClick={deleteItem(id)}
57
- />
58
- </div>
59
- );
60
- };
61
-
62
- export default Content;
@@ -1,29 +0,0 @@
1
- import { makeStyles } from 'tss-react/mui';
2
-
3
- export const useListStyles = makeStyles()((theme) => ({
4
- content: {
5
- '& [data-dragging="false"]': {
6
- cursor: 'grab'
7
- },
8
- '& [data-dragging="true"]': {
9
- cursor: 'grabbing'
10
- },
11
- alignItems: 'center',
12
- borderBottom: `1px dashed ${theme.palette.action.disabledBackground}`,
13
- display: 'flex',
14
- flexDirection: 'row',
15
- padding: theme.spacing(1, 0)
16
- },
17
- innerContent: {
18
- flexGrow: 1
19
- },
20
- items: {
21
- maxHeight: theme.spacing(16),
22
- overflowY: 'auto'
23
- },
24
- list: {
25
- display: 'flex',
26
- flexDirection: 'column',
27
- gap: theme.spacing(1)
28
- }
29
- }));
@@ -1,58 +0,0 @@
1
- import { ComponentType } from 'react';
2
-
3
- import { closestCenter } from '@dnd-kit/core';
4
- import { verticalListSortingStrategy } from '@dnd-kit/sortable';
5
- import { useTranslation } from 'react-i18next';
6
-
7
- import { InputPropsWithoutGroup } from '../models';
8
- import { SortableItems, Subtitle } from '../../..';
9
-
10
- import { useList } from './useList';
11
- import { useListStyles } from './List.styles';
12
- import Content, { ContentProps } from './Content';
13
-
14
- const List = ({
15
- list,
16
- fieldName
17
- }: InputPropsWithoutGroup): JSX.Element | null => {
18
- const { t } = useTranslation();
19
- const { classes } = useListStyles();
20
-
21
- const { addItem, sortList, sortedList, deleteItem } = useList({ fieldName });
22
-
23
- const { AddItem, addItemLabel, sortLabel, SortContent, itemProps } = list as {
24
- AddItem: ComponentType<{ addItem }>;
25
- SortContent: ComponentType;
26
- addItemLabel?: string;
27
- itemProps: Array<string>;
28
- sortLabel?: string;
29
- };
30
-
31
- return (
32
- <div className={classes.list}>
33
- {addItemLabel && <Subtitle>{t(addItemLabel)}</Subtitle>}
34
- <AddItem addItem={addItem} />
35
- {sortLabel && <Subtitle>{t(sortLabel)}</Subtitle>}
36
- <div className={classes.items}>
37
- <SortableItems
38
- updateSortableItemsOnItemsChange
39
- // eslint-disable-next-line react/no-unstable-nested-components
40
- Content={(props: Omit<ContentProps, 'children' | 'deleteItem'>) => (
41
- <Content {...props} deleteItem={deleteItem}>
42
- <SortContent {...props} />
43
- </Content>
44
- )}
45
- collisionDetection={closestCenter}
46
- itemProps={itemProps}
47
- items={sortedList}
48
- sortingStrategy={verticalListSortingStrategy}
49
- onDragEnd={({ items }): void => {
50
- sortList(items);
51
- }}
52
- />
53
- </div>
54
- </div>
55
- );
56
- };
57
-
58
- export default List;
@@ -1,81 +0,0 @@
1
- import { useMemo, useRef } from 'react';
2
-
3
- import { FormikValues, useFormikContext } from 'formik';
4
- import {
5
- append,
6
- equals,
7
- inc,
8
- isEmpty,
9
- pluck,
10
- prop,
11
- reject,
12
- sortBy
13
- } from 'ramda';
14
-
15
- import { SelectEntry } from '../../..';
16
-
17
- interface UseListState {
18
- addItem: (newItem: SelectEntry) => void;
19
- deleteItem: (id: string) => () => void;
20
- sortList: (items: Array<string>) => void;
21
- sortedList: Array<unknown>;
22
- }
23
-
24
- export const useList = ({ fieldName }): UseListState => {
25
- const { values, setFieldValue } = useFormikContext<FormikValues>();
26
- const maxOrder = useRef(0);
27
-
28
- const list = values[fieldName];
29
-
30
- const sortedList = useMemo(
31
- () =>
32
- sortBy(prop('order'), list).map(({ id, ...props }) => ({
33
- id: `${id}`,
34
- ...props
35
- })),
36
- [list]
37
- );
38
-
39
- const addItem = (newItem: SelectEntry): void => {
40
- setFieldValue(
41
- fieldName,
42
- append(
43
- {
44
- ...newItem,
45
- id: (newItem as SelectEntry).id as number,
46
- order: inc(maxOrder.current)
47
- },
48
- list
49
- )
50
- );
51
- };
52
-
53
- const deleteItem = (id: string) => (): void => {
54
- const newItems = reject((item) => equals(Number(id), item.id))(list);
55
-
56
- setFieldValue(fieldName, newItems);
57
- };
58
-
59
- const sortList = (items: Array<string>): void => {
60
- const newOrderedList = items.map((itemId, idx) => {
61
- const item = sortedList.find(({ id }) => equals(id, itemId));
62
-
63
- return {
64
- ...item,
65
- id: Number(item?.id),
66
- order: inc(idx)
67
- };
68
- });
69
-
70
- setFieldValue(fieldName, newOrderedList);
71
- };
72
-
73
- maxOrder.current = isEmpty(list) ? 0 : Math.max(...pluck('order', list));
74
-
75
- return {
76
- addItem,
77
- deleteItem,
78
- sortList,
79
- sortedList
80
- };
81
- };