@centreon/ui 24.4.44 → 24.4.45-develop.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.
- package/package.json +19 -14
- package/public/mockServiceWorker.js +1 -1
- package/src/Button/Icon/index.tsx +1 -1
- package/src/Button/Save/StartIcon.tsx +3 -3
- package/src/Button/Save/index.tsx +9 -5
- package/src/Checkbox/Checkbox.tsx +2 -2
- package/src/Checkbox/CheckboxGroup/index.tsx +2 -2
- package/src/Dashboard/Item.tsx +1 -1
- package/src/Dashboard/Layout.tsx +2 -2
- package/src/Dialog/index.tsx +1 -1
- package/src/FallbackPage/FallbackPage.tsx +3 -3
- package/src/FileDropZone/index.tsx +3 -1
- package/src/Form/Form.cypress.spec.tsx +133 -0
- package/src/Form/Inputs/CheckboxGroup.tsx +1 -4
- package/src/Form/Inputs/List/Content.tsx +62 -0
- package/src/Form/Inputs/List/List.styles.ts +29 -0
- package/src/Form/Inputs/List/List.tsx +58 -0
- package/src/Form/Inputs/List/useList.ts +81 -0
- package/src/Form/Inputs/index.tsx +3 -1
- package/src/Form/Inputs/models.ts +9 -1
- package/src/Graph/LineChart/BasicComponents/Lines/Threshold/Circle.tsx +2 -2
- package/src/Graph/LineChart/BasicComponents/Lines/Threshold/index.tsx +5 -4
- package/src/Graph/LineChart/BasicComponents/Thresholds.tsx +2 -2
- package/src/Graph/LineChart/BasicComponents/useFilterLines.ts +1 -1
- package/src/Graph/LineChart/InteractiveComponents/AnchorPoint/GuidingLines.tsx +2 -2
- package/src/Graph/LineChart/InteractiveComponents/Annotations/Annotation/index.tsx +2 -3
- package/src/Graph/LineChart/InteractiveComponents/Annotations/EventAnnotations.tsx +1 -1
- package/src/Graph/LineChart/Legend/useLegend.ts +3 -3
- package/src/Graph/LineChart/helpers/doc.ts +16 -13
- package/src/Graph/LineChart/helpers/index.ts +1 -1
- package/src/Graph/LineChart/index.stories.tsx +4 -2
- package/src/Graph/SingleBar/Thresholds.tsx +2 -2
- package/src/Graph/Text/Text.stories.tsx +60 -4
- package/src/Graph/common/timeSeries/index.ts +3 -3
- package/src/InputField/Select/Autocomplete/Connected/index.tsx +10 -7
- package/src/InputField/Select/Autocomplete/Draggable/SortableList.tsx +1 -1
- package/src/InputField/Select/Autocomplete/Draggable/SortableListContent.tsx +1 -1
- package/src/InputField/Select/Autocomplete/Draggable/index.tsx +1 -1
- package/src/InputField/Select/Autocomplete/index.tsx +121 -115
- package/src/InputField/Select/IconPopover/index.tsx +2 -2
- package/src/InputField/Select/index.tsx +1 -1
- package/src/InputField/Text/index.tsx +2 -2
- package/src/Listing/Cell/DataCell.tsx +15 -1
- package/src/Listing/Header/ListingHeader.tsx +1 -1
- package/src/Listing/Listing.styles.ts +2 -3
- package/src/Listing/index.stories.tsx +12 -1
- package/src/Listing/index.tsx +1 -2
- package/src/Module/Module.cypress.spec.tsx +129 -0
- package/src/Module/index.tsx +2 -4
- package/src/RichTextEditor/RichTextEditor.tsx +12 -1
- package/src/SortableItems/index.tsx +2 -7
- package/src/ThemeProvider/index.tsx +24 -0
- package/src/TimePeriods/CustomTimePeriod/CompactCustomTimePeriod.styles.ts +6 -7
- package/src/TimePeriods/CustomTimePeriod/PopoverCustomTimePeriod/PickersStartEndDate.tsx +8 -3
- package/src/TimePeriods/CustomTimePeriod/PopoverCustomTimePeriod/models.ts +0 -2
- package/src/TimePeriods/DateTimePickerInput.tsx +56 -19
- package/src/TimePeriods/ResolutionTimePeriod.cypress.spec.tsx +12 -9
- package/src/TimePeriods/TimePeriods.cypress.spec.tsx +9 -33
- package/src/TimePeriods/helpers/index.ts +1 -1
- package/src/TimePeriods/index.stories.tsx +12 -4
- package/src/TimePeriods/index.tsx +2 -2
- package/src/api/QueryProvider.tsx +1 -1
- package/src/api/TestQueryProvider.tsx +1 -1
- package/src/api/useFetchQuery/index.ts +27 -23
- package/src/api/useMutationQuery/index.ts +45 -21
- package/src/components/Button/Icon/IconButton.tsx +6 -2
- package/src/components/DataTable/DataListing.tsx +6 -0
- package/src/components/DataTable/DataTable.cypress.spec.tsx +193 -0
- package/src/components/DataTable/DataTable.stories.tsx +40 -0
- package/src/components/DataTable/DataTable.styles.ts +3 -0
- package/src/components/DataTable/DataTable.tsx +3 -3
- package/src/components/DataTable/Item/DataTableItem.styles.ts +7 -2
- package/src/components/DataTable/Item/DataTableItem.tsx +4 -4
- package/src/components/DataTable/index.ts +3 -1
- package/src/components/Form/Dashboard/DashboardForm.tsx +15 -12
- package/src/components/Layout/PageLayout/PageLayout.tsx +1 -1
- package/src/components/Layout/PageLayout/PageLayoutActions.tsx +1 -0
- package/src/components/Layout/PageLayout/PageLayoutBody.tsx +1 -0
- package/src/components/Layout/PageLayout/PageLayoutHeader.tsx +5 -1
- package/src/components/Layout/PageLayout/PageQuickAccess.tsx +76 -0
- package/src/components/Layout/PageLayout/index.ts +3 -1
- package/src/components/Layout/PageLayout.cypress.spec.tsx +66 -0
- package/src/components/Modal/Modal.styles.ts +1 -1
- package/src/components/Tooltip/ConfirmationTooltip/ConfirmationTooltip.stories.tsx +3 -3
- package/src/components/Tooltip/ConfirmationTooltip/ConfirmationTooltip.tsx +1 -1
- package/src/components/Tooltip/ConfirmationTooltip/models.ts +1 -1
- package/src/index.ts +2 -2
- package/src/queryParameters/url/index.ts +5 -1
- package/src/utils/index.ts +2 -1
- package/src/utils/{useLicenseExpirationWarning.cypress.spec.tsx → useLicenseExpirationWarning.test.tsx} +48 -37
- package/src/utils/useLicenseExpirationWarning.ts +18 -18
- package/src/utils/usePluralizedTranslation.ts +21 -0
- package/src/screens/dashboard/DashboardsDetail.stories.tsx +0 -108
- package/src/screens/dashboard/DashboardsOverview.stories.tsx +0 -281
- package/src/utils/useDateTimePickerAdapter.ts +0 -309
|
@@ -3,15 +3,10 @@ import localizedFormatPlugin from 'dayjs/plugin/localizedFormat';
|
|
|
3
3
|
import timezonePlugin from 'dayjs/plugin/timezone';
|
|
4
4
|
import utcPlugin from 'dayjs/plugin/utc';
|
|
5
5
|
import { Provider, createStore } from 'jotai';
|
|
6
|
-
import { equals
|
|
6
|
+
import { equals } from 'ramda';
|
|
7
7
|
import { renderHook } from '@testing-library/react';
|
|
8
8
|
|
|
9
|
-
import {
|
|
10
|
-
|
|
11
|
-
import {
|
|
12
|
-
useLocaleDateTimeFormat,
|
|
13
|
-
useDateTimePickerAdapter
|
|
14
|
-
} from '@centreon/ui';
|
|
9
|
+
import { useLocaleDateTimeFormat } from '@centreon/ui';
|
|
15
10
|
import { ListingVariant, userAtom } from '@centreon/ui-context';
|
|
16
11
|
|
|
17
12
|
import { CustomTimePeriodProperty } from './models';
|
|
@@ -297,10 +292,6 @@ testData.forEach((item) =>
|
|
|
297
292
|
beforeEach(() => {
|
|
298
293
|
cy.viewport('macbook-13');
|
|
299
294
|
|
|
300
|
-
const { result } = renderHook(() => useDateTimePickerAdapter());
|
|
301
|
-
|
|
302
|
-
const { Adapter } = result.current;
|
|
303
|
-
|
|
304
295
|
const store = createStore();
|
|
305
296
|
store.set(userAtom, {
|
|
306
297
|
...retrievedUser,
|
|
@@ -311,31 +302,16 @@ testData.forEach((item) =>
|
|
|
311
302
|
cy.mount({
|
|
312
303
|
Component: (
|
|
313
304
|
<Provider store={store}>
|
|
314
|
-
<
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
318
|
-
|
|
319
|
-
property={CustomTimePeriodProperty.start}
|
|
320
|
-
/>
|
|
321
|
-
</LocalizationProvider>
|
|
305
|
+
<DateTimePickerInput
|
|
306
|
+
changeDate={cy.stub()}
|
|
307
|
+
date={new Date(item.initialDate)}
|
|
308
|
+
property={CustomTimePeriodProperty.start}
|
|
309
|
+
/>
|
|
322
310
|
</Provider>
|
|
323
311
|
)
|
|
324
312
|
});
|
|
325
313
|
});
|
|
326
314
|
|
|
327
|
-
it('checks input calendar value contains correct date', () => {
|
|
328
|
-
const { result } = renderHook(() => useLocaleDateTimeFormat());
|
|
329
|
-
const { format } = result.current;
|
|
330
|
-
|
|
331
|
-
const dateInput = format({
|
|
332
|
-
date: dayjs(item.initialDate).tz(item.timezone),
|
|
333
|
-
formatString: 'L hh:mm A'
|
|
334
|
-
});
|
|
335
|
-
|
|
336
|
-
cy.get('input').should('have.value', dateInput);
|
|
337
|
-
});
|
|
338
|
-
|
|
339
315
|
it(`displays the correct number of days for the current month when the ${item.button} button is clicked`, () => {
|
|
340
316
|
cy.get('button').click();
|
|
341
317
|
item.data.forEach((element) => {
|
|
@@ -389,7 +365,7 @@ testData.forEach((item) =>
|
|
|
389
365
|
|
|
390
366
|
cy.get('[role="rowgroup"]').children().as('listWeeks');
|
|
391
367
|
|
|
392
|
-
cy.get('@listWeeks').should('have.length',
|
|
368
|
+
cy.get('@listWeeks').should('have.length', numberWeeks);
|
|
393
369
|
cy.get('@listWeeks').eq(0).as('firstWeek');
|
|
394
370
|
cy.get('@firstWeek').children().as('listDaysInFirstWeek');
|
|
395
371
|
|
|
@@ -416,7 +392,7 @@ testData.forEach((item) =>
|
|
|
416
392
|
|
|
417
393
|
cy.get('[role="rowgroup"]').children().as('listWeeks');
|
|
418
394
|
|
|
419
|
-
cy.get('@listWeeks').should('have.length',
|
|
395
|
+
cy.get('@listWeeks').should('have.length', numberWeeks);
|
|
420
396
|
|
|
421
397
|
cy.get('@listWeeks')
|
|
422
398
|
.eq(numberWeeks - 1)
|
|
@@ -88,20 +88,27 @@ const args = {
|
|
|
88
88
|
]
|
|
89
89
|
};
|
|
90
90
|
|
|
91
|
+
const parameters = {
|
|
92
|
+
chromatic: { diffThreshold: 0.1 }
|
|
93
|
+
};
|
|
94
|
+
|
|
91
95
|
export const BasicTimePeriod: Story = {
|
|
92
96
|
...Template,
|
|
93
|
-
argTypes
|
|
97
|
+
argTypes,
|
|
98
|
+
parameters
|
|
94
99
|
};
|
|
95
100
|
|
|
96
101
|
export const WithExtraTimePeriods: Story = {
|
|
97
102
|
...Template,
|
|
98
103
|
argTypes,
|
|
99
|
-
args
|
|
104
|
+
args,
|
|
105
|
+
parameters
|
|
100
106
|
};
|
|
101
107
|
|
|
102
108
|
export const WithExternalComponent: Story = {
|
|
103
109
|
...TemplateWithExternalComponent,
|
|
104
|
-
argTypes
|
|
110
|
+
argTypes,
|
|
111
|
+
parameters
|
|
105
112
|
};
|
|
106
113
|
|
|
107
114
|
export const SimpleTimePeriod: StorySimpleTimePeriod = {
|
|
@@ -109,5 +116,6 @@ export const SimpleTimePeriod: StorySimpleTimePeriod = {
|
|
|
109
116
|
args: {
|
|
110
117
|
endDate: dayjs(Date.now()).toDate(),
|
|
111
118
|
startDate: dayjs(Date.now()).subtract(29, 'day').toDate()
|
|
112
|
-
}
|
|
119
|
+
},
|
|
120
|
+
parameters
|
|
113
121
|
};
|
|
@@ -20,7 +20,7 @@ import { useStyles } from './TimePeriods.styles';
|
|
|
20
20
|
import {
|
|
21
21
|
CustomTimePeriod as CustomTimePeriodModel,
|
|
22
22
|
EndStartInterval,
|
|
23
|
-
TimePeriod
|
|
23
|
+
TimePeriod as TimePeriodModel
|
|
24
24
|
} from './models';
|
|
25
25
|
import useTimePeriod from './useTimePeriod';
|
|
26
26
|
|
|
@@ -34,7 +34,7 @@ interface Parameters extends EndStartInterval {
|
|
|
34
34
|
export interface Props {
|
|
35
35
|
adjustTimePeriodData?: Omit<CustomTimePeriodModel, 'timelineEventsLimit'>;
|
|
36
36
|
disabled?: boolean;
|
|
37
|
-
extraTimePeriods?: Array<Omit<
|
|
37
|
+
extraTimePeriods?: Array<Omit<TimePeriodModel, 'timelineEventsLimit'>>;
|
|
38
38
|
getIsError?: (value: boolean) => void;
|
|
39
39
|
getParameters?: ({ start, end, timelineEventsLimit }: Parameters) => void;
|
|
40
40
|
renderExternalComponent?: ReactNode;
|
|
@@ -27,7 +27,9 @@ export interface UseFetchQueryProps<T> {
|
|
|
27
27
|
getQueryKey: () => QueryKey;
|
|
28
28
|
httpCodesBypassErrorSnackbar?: Array<number>;
|
|
29
29
|
isPaginated?: boolean;
|
|
30
|
-
queryOptions?:
|
|
30
|
+
queryOptions?: {
|
|
31
|
+
suspense?: boolean;
|
|
32
|
+
} & Omit<
|
|
31
33
|
UseQueryOptions<T | ResponseError, Error, T | ResponseError, QueryKey>,
|
|
32
34
|
'queryKey' | 'queryFn'
|
|
33
35
|
>;
|
|
@@ -35,11 +37,12 @@ export interface UseFetchQueryProps<T> {
|
|
|
35
37
|
|
|
36
38
|
export type UseFetchQueryState<T> = {
|
|
37
39
|
data?: T;
|
|
40
|
+
error: Omit<ResponseError, 'isError'> | null;
|
|
38
41
|
fetchQuery: () => Promise<T | ResponseError>;
|
|
39
42
|
prefetchNextPage: ({ page, getPrefetchQueryKey }) => void;
|
|
40
43
|
prefetchPreviousPage: ({ page, getPrefetchQueryKey }) => void;
|
|
41
44
|
prefetchQuery: ({ endpointParams, queryKey }) => void;
|
|
42
|
-
} & Omit<QueryObserverBaseResult, 'data'>;
|
|
45
|
+
} & Omit<QueryObserverBaseResult, 'data' | 'error'>;
|
|
43
46
|
|
|
44
47
|
export interface PrefetchEndpointParams {
|
|
45
48
|
page: number;
|
|
@@ -64,9 +67,8 @@ const useFetchQuery = <T extends object>({
|
|
|
64
67
|
|
|
65
68
|
const { showErrorMessage } = useSnackbar();
|
|
66
69
|
|
|
67
|
-
const queryData = useQuery<T | ResponseError, Error>(
|
|
68
|
-
|
|
69
|
-
({ signal }): Promise<T | ResponseError> =>
|
|
70
|
+
const queryData = useQuery<T | ResponseError, Error>({
|
|
71
|
+
queryFn: ({ signal }): Promise<T | ResponseError> =>
|
|
70
72
|
customFetch<T>({
|
|
71
73
|
baseEndpoint,
|
|
72
74
|
catchError,
|
|
@@ -76,10 +78,9 @@ const useFetchQuery = <T extends object>({
|
|
|
76
78
|
headers: new Headers(fetchHeaders),
|
|
77
79
|
signal
|
|
78
80
|
}),
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
);
|
|
81
|
+
queryKey: getQueryKey(),
|
|
82
|
+
...queryOptions
|
|
83
|
+
});
|
|
83
84
|
|
|
84
85
|
const queryClient = useQueryClient();
|
|
85
86
|
|
|
@@ -99,9 +100,8 @@ const useFetchQuery = <T extends object>({
|
|
|
99
100
|
};
|
|
100
101
|
|
|
101
102
|
const prefetchQuery = ({ endpointParams, queryKey }): void => {
|
|
102
|
-
queryClient.prefetchQuery(
|
|
103
|
-
|
|
104
|
-
({ signal }): Promise<T | ResponseError> =>
|
|
103
|
+
queryClient.prefetchQuery({
|
|
104
|
+
queryFn: ({ signal }): Promise<T | ResponseError> =>
|
|
105
105
|
customFetch<T>({
|
|
106
106
|
baseEndpoint,
|
|
107
107
|
catchError,
|
|
@@ -110,8 +110,9 @@ const useFetchQuery = <T extends object>({
|
|
|
110
110
|
endpoint: getEndpoint(endpointParams),
|
|
111
111
|
headers: new Headers(fetchHeaders),
|
|
112
112
|
signal
|
|
113
|
-
})
|
|
114
|
-
|
|
113
|
+
}),
|
|
114
|
+
queryKey
|
|
115
|
+
});
|
|
115
116
|
};
|
|
116
117
|
|
|
117
118
|
const prefetchNextPage = ({ page, getPrefetchQueryKey }): void => {
|
|
@@ -141,9 +142,8 @@ const useFetchQuery = <T extends object>({
|
|
|
141
142
|
};
|
|
142
143
|
|
|
143
144
|
const fetchQuery = (): Promise<T | ResponseError> => {
|
|
144
|
-
return queryClient.fetchQuery(
|
|
145
|
-
|
|
146
|
-
({ signal }): Promise<T | ResponseError> =>
|
|
145
|
+
return queryClient.fetchQuery({
|
|
146
|
+
queryFn: ({ signal }): Promise<T | ResponseError> =>
|
|
147
147
|
customFetch<T>({
|
|
148
148
|
baseEndpoint,
|
|
149
149
|
catchError,
|
|
@@ -152,8 +152,9 @@ const useFetchQuery = <T extends object>({
|
|
|
152
152
|
endpoint: getEndpoint(),
|
|
153
153
|
headers: new Headers(fetchHeaders),
|
|
154
154
|
signal
|
|
155
|
-
})
|
|
156
|
-
|
|
155
|
+
}),
|
|
156
|
+
queryKey: getQueryKey()
|
|
157
|
+
});
|
|
157
158
|
};
|
|
158
159
|
|
|
159
160
|
const data = useMemo(
|
|
@@ -174,13 +175,16 @@ const useFetchQuery = <T extends object>({
|
|
|
174
175
|
return;
|
|
175
176
|
}
|
|
176
177
|
|
|
177
|
-
queryClient.cancelQueries(getQueryKey());
|
|
178
|
+
queryClient.cancelQueries({ queryKey: getQueryKey() });
|
|
178
179
|
};
|
|
179
180
|
}, []);
|
|
180
181
|
|
|
181
|
-
useEffect(
|
|
182
|
-
|
|
183
|
-
|
|
182
|
+
useEffect(
|
|
183
|
+
() => {
|
|
184
|
+
manageError();
|
|
185
|
+
},
|
|
186
|
+
useDeepCompare([queryData.data])
|
|
187
|
+
);
|
|
184
188
|
|
|
185
189
|
return {
|
|
186
190
|
...omit(['data', 'error'], queryData),
|
|
@@ -8,7 +8,7 @@ import {
|
|
|
8
8
|
} from '@tanstack/react-query';
|
|
9
9
|
import { JsonDecoder } from 'ts.data.json';
|
|
10
10
|
import anylogger from 'anylogger';
|
|
11
|
-
import { includes } from 'ramda';
|
|
11
|
+
import { includes, omit } from 'ramda';
|
|
12
12
|
|
|
13
13
|
import { CatchErrorProps, customFetch, ResponseError } from '../customFetch';
|
|
14
14
|
import useSnackbar from '../../Snackbar/useSnackbar';
|
|
@@ -31,14 +31,33 @@ export type UseMutationQueryProps<T, TMeta> = {
|
|
|
31
31
|
getEndpoint: (_meta: TMeta) => string;
|
|
32
32
|
httpCodesBypassErrorSnackbar?: Array<number>;
|
|
33
33
|
method: Method;
|
|
34
|
-
|
|
34
|
+
onError?: (
|
|
35
|
+
error: ResponseError,
|
|
36
|
+
variables: T & { _meta: TMeta },
|
|
37
|
+
context: unknown
|
|
38
|
+
) => unknown;
|
|
39
|
+
onMutate?: (variables: T & { _meta: TMeta }) => Promise<unknown> | unknown;
|
|
40
|
+
onSuccess?: (
|
|
41
|
+
data: ResponseError | T,
|
|
42
|
+
variables: T & {
|
|
43
|
+
_meta: TMeta;
|
|
44
|
+
},
|
|
45
|
+
context: unknown
|
|
46
|
+
) => unknown;
|
|
47
|
+
} & Omit<
|
|
48
|
+
UseMutationOptions<T & { _meta?: TMeta }>,
|
|
49
|
+
'mutationFn' | 'onError' | 'onMutate' | 'onSuccess'
|
|
50
|
+
>;
|
|
35
51
|
|
|
36
52
|
const log = anylogger('API Request');
|
|
37
53
|
|
|
38
|
-
export type UseMutationQueryState<T> =
|
|
54
|
+
export type UseMutationQueryState<T> = Omit<
|
|
55
|
+
UseMutationResult<T | ResponseError>,
|
|
56
|
+
'isError'
|
|
57
|
+
> & {
|
|
39
58
|
isError: boolean;
|
|
40
59
|
isMutating: boolean;
|
|
41
|
-
}
|
|
60
|
+
};
|
|
42
61
|
|
|
43
62
|
const useMutationQuery = <T extends object, TMeta>({
|
|
44
63
|
getEndpoint,
|
|
@@ -51,6 +70,7 @@ const useMutationQuery = <T extends object, TMeta>({
|
|
|
51
70
|
onMutate,
|
|
52
71
|
onError,
|
|
53
72
|
onSuccess,
|
|
73
|
+
onSettled,
|
|
54
74
|
baseEndpoint
|
|
55
75
|
}: UseMutationQueryProps<T, TMeta>): UseMutationQueryState<T> => {
|
|
56
76
|
const { showErrorMessage } = useSnackbar();
|
|
@@ -59,8 +79,10 @@ const useMutationQuery = <T extends object, TMeta>({
|
|
|
59
79
|
T | ResponseError,
|
|
60
80
|
ResponseError,
|
|
61
81
|
T & { _meta: TMeta }
|
|
62
|
-
>(
|
|
63
|
-
|
|
82
|
+
>({
|
|
83
|
+
mutationFn: (
|
|
84
|
+
_payload: T & { _meta: TMeta }
|
|
85
|
+
): Promise<T | ResponseError> => {
|
|
64
86
|
const { _meta, ...payload } = _payload || {};
|
|
65
87
|
|
|
66
88
|
return customFetch<T>({
|
|
@@ -78,19 +100,18 @@ const useMutationQuery = <T extends object, TMeta>({
|
|
|
78
100
|
payload
|
|
79
101
|
});
|
|
80
102
|
},
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
103
|
+
onError,
|
|
104
|
+
onMutate,
|
|
105
|
+
onSettled,
|
|
106
|
+
onSuccess: (data, variables, context) => {
|
|
107
|
+
if (data?.isError) {
|
|
108
|
+
onError?.(data, variables, context);
|
|
87
109
|
|
|
88
|
-
|
|
89
|
-
}
|
|
90
|
-
onSuccess?.(data, variables, context);
|
|
110
|
+
return;
|
|
91
111
|
}
|
|
112
|
+
onSuccess?.(data, variables, context);
|
|
92
113
|
}
|
|
93
|
-
);
|
|
114
|
+
});
|
|
94
115
|
|
|
95
116
|
const manageError = (): void => {
|
|
96
117
|
const data = queryData.data as ResponseError | undefined;
|
|
@@ -107,14 +128,17 @@ const useMutationQuery = <T extends object, TMeta>({
|
|
|
107
128
|
}
|
|
108
129
|
};
|
|
109
130
|
|
|
110
|
-
useEffect(
|
|
111
|
-
|
|
112
|
-
|
|
131
|
+
useEffect(
|
|
132
|
+
() => {
|
|
133
|
+
manageError();
|
|
134
|
+
},
|
|
135
|
+
useDeepCompare([queryData.data])
|
|
136
|
+
);
|
|
113
137
|
|
|
114
138
|
return {
|
|
115
|
-
...queryData,
|
|
139
|
+
...omit(['isError'], queryData),
|
|
116
140
|
isError: (queryData.data as ResponseError | undefined)?.isError || false,
|
|
117
|
-
isMutating: queryData.
|
|
141
|
+
isMutating: queryData.isPending
|
|
118
142
|
};
|
|
119
143
|
};
|
|
120
144
|
|
|
@@ -1,6 +1,9 @@
|
|
|
1
1
|
import React, { ReactElement, ReactNode } from 'react';
|
|
2
2
|
|
|
3
|
-
import {
|
|
3
|
+
import {
|
|
4
|
+
IconButton as MuiIconButton,
|
|
5
|
+
IconButtonProps as MuiIconButtonProps
|
|
6
|
+
} from '@mui/material';
|
|
4
7
|
|
|
5
8
|
import { AriaLabelingAttributes } from '../../../@types/aria-attributes';
|
|
6
9
|
import { DataTestAttributes } from '../../../@types/data-attributes';
|
|
@@ -23,7 +26,8 @@ type IconButtonProps = {
|
|
|
23
26
|
size?: 'small' | 'medium' | 'large';
|
|
24
27
|
variant?: 'primary' | 'secondary' | 'ghost';
|
|
25
28
|
} & AriaLabelingAttributes &
|
|
26
|
-
DataTestAttributes
|
|
29
|
+
DataTestAttributes &
|
|
30
|
+
MuiIconButtonProps;
|
|
27
31
|
|
|
28
32
|
/**
|
|
29
33
|
* @todo re-factor as `iconVariant: 'icon-only'` Button variant, and remove IconButton component (reason: code duplication)
|
|
@@ -0,0 +1,193 @@
|
|
|
1
|
+
import { Box } from '@mui/material';
|
|
2
|
+
|
|
3
|
+
import { ColumnType } from '../../Listing/models';
|
|
4
|
+
|
|
5
|
+
import { DataTable } from '.';
|
|
6
|
+
|
|
7
|
+
const data = Array(5)
|
|
8
|
+
.fill(0)
|
|
9
|
+
.map((_, idx) => ({
|
|
10
|
+
description: `Description ${idx}`,
|
|
11
|
+
id: idx,
|
|
12
|
+
title: `Entity ${idx}`
|
|
13
|
+
}));
|
|
14
|
+
|
|
15
|
+
const initializeDataTableGrid = ({
|
|
16
|
+
hasActions,
|
|
17
|
+
hasCardAction,
|
|
18
|
+
canDelete
|
|
19
|
+
}): void => {
|
|
20
|
+
cy.viewport(1200, 590);
|
|
21
|
+
cy.mount({
|
|
22
|
+
Component: (
|
|
23
|
+
<DataTable variant="grid">
|
|
24
|
+
{data.map(({ title, description }) => (
|
|
25
|
+
<DataTable.Item
|
|
26
|
+
description={description}
|
|
27
|
+
hasActions={hasActions}
|
|
28
|
+
hasCardAction={hasCardAction}
|
|
29
|
+
key={title}
|
|
30
|
+
labelsDelete={{
|
|
31
|
+
cancel: 'Cancel',
|
|
32
|
+
confirm: {
|
|
33
|
+
label: 'Delete'
|
|
34
|
+
}
|
|
35
|
+
}}
|
|
36
|
+
title={title}
|
|
37
|
+
onDelete={canDelete ? cy.stub() : undefined}
|
|
38
|
+
/>
|
|
39
|
+
))}
|
|
40
|
+
</DataTable>
|
|
41
|
+
)
|
|
42
|
+
});
|
|
43
|
+
};
|
|
44
|
+
|
|
45
|
+
const initializeDataTableEmpty = (canCreate = false): void => {
|
|
46
|
+
cy.viewport(1200, 590);
|
|
47
|
+
cy.mount({
|
|
48
|
+
Component: (
|
|
49
|
+
<DataTable isEmpty variant="grid">
|
|
50
|
+
<DataTable.EmptyState
|
|
51
|
+
canCreate={canCreate}
|
|
52
|
+
labels={{
|
|
53
|
+
actions: {
|
|
54
|
+
create: 'Create'
|
|
55
|
+
},
|
|
56
|
+
title: 'Welcome'
|
|
57
|
+
}}
|
|
58
|
+
onCreate={cy.stub()}
|
|
59
|
+
/>
|
|
60
|
+
</DataTable>
|
|
61
|
+
)
|
|
62
|
+
});
|
|
63
|
+
};
|
|
64
|
+
|
|
65
|
+
const initializeDataTableListing = (): void => {
|
|
66
|
+
cy.viewport(1200, 590);
|
|
67
|
+
cy.mount({
|
|
68
|
+
Component: (
|
|
69
|
+
<Box sx={{ height: '100vh' }}>
|
|
70
|
+
<DataTable variant="listing">
|
|
71
|
+
<DataTable.Listing
|
|
72
|
+
columns={[
|
|
73
|
+
{
|
|
74
|
+
getFormattedString: (row) => row.title,
|
|
75
|
+
id: 'title',
|
|
76
|
+
label: 'Title',
|
|
77
|
+
type: ColumnType.string
|
|
78
|
+
},
|
|
79
|
+
{
|
|
80
|
+
getFormattedString: (row) => row.description,
|
|
81
|
+
id: 'description',
|
|
82
|
+
label: 'Description',
|
|
83
|
+
type: ColumnType.string
|
|
84
|
+
}
|
|
85
|
+
]}
|
|
86
|
+
rows={data}
|
|
87
|
+
/>
|
|
88
|
+
</DataTable>
|
|
89
|
+
</Box>
|
|
90
|
+
)
|
|
91
|
+
});
|
|
92
|
+
};
|
|
93
|
+
|
|
94
|
+
describe('DataTable: Grid', () => {
|
|
95
|
+
it('displays items with title and description only', () => {
|
|
96
|
+
initializeDataTableGrid({
|
|
97
|
+
canDelete: false,
|
|
98
|
+
hasActions: false,
|
|
99
|
+
hasCardAction: false
|
|
100
|
+
});
|
|
101
|
+
|
|
102
|
+
data.forEach(({ title, description }) => {
|
|
103
|
+
cy.contains(title).should('be.visible');
|
|
104
|
+
cy.contains(description).should('be.visible');
|
|
105
|
+
});
|
|
106
|
+
|
|
107
|
+
cy.makeSnapshot();
|
|
108
|
+
});
|
|
109
|
+
|
|
110
|
+
it('displays items with actions', () => {
|
|
111
|
+
initializeDataTableGrid({
|
|
112
|
+
canDelete: false,
|
|
113
|
+
hasActions: true,
|
|
114
|
+
hasCardAction: false
|
|
115
|
+
});
|
|
116
|
+
|
|
117
|
+
cy.findAllByLabelText('edit access rights').should('have.length', 5);
|
|
118
|
+
cy.findAllByLabelText('edit').should('have.length', 5);
|
|
119
|
+
|
|
120
|
+
cy.makeSnapshot();
|
|
121
|
+
});
|
|
122
|
+
|
|
123
|
+
it('displays items with delete action', () => {
|
|
124
|
+
initializeDataTableGrid({
|
|
125
|
+
canDelete: true,
|
|
126
|
+
hasActions: true,
|
|
127
|
+
hasCardAction: false
|
|
128
|
+
});
|
|
129
|
+
|
|
130
|
+
cy.findAllByLabelText('delete').should('have.length', 5);
|
|
131
|
+
|
|
132
|
+
cy.makeSnapshot();
|
|
133
|
+
});
|
|
134
|
+
|
|
135
|
+
it('displays items with card action only', () => {
|
|
136
|
+
initializeDataTableGrid({
|
|
137
|
+
canDelete: false,
|
|
138
|
+
hasActions: false,
|
|
139
|
+
hasCardAction: true
|
|
140
|
+
});
|
|
141
|
+
|
|
142
|
+
cy.findAllByLabelText('view').should('have.length', 5);
|
|
143
|
+
|
|
144
|
+
cy.makeSnapshot();
|
|
145
|
+
});
|
|
146
|
+
|
|
147
|
+
it('displays items with card action and bottom actions', () => {
|
|
148
|
+
initializeDataTableGrid({
|
|
149
|
+
canDelete: true,
|
|
150
|
+
hasActions: true,
|
|
151
|
+
hasCardAction: true
|
|
152
|
+
});
|
|
153
|
+
|
|
154
|
+
cy.findAllByLabelText('view').should('have.length', 5);
|
|
155
|
+
cy.findAllByLabelText('delete').should('have.length', 5);
|
|
156
|
+
cy.findAllByLabelText('edit access rights').should('have.length', 5);
|
|
157
|
+
cy.findAllByLabelText('edit').should('have.length', 5);
|
|
158
|
+
|
|
159
|
+
cy.makeSnapshot();
|
|
160
|
+
});
|
|
161
|
+
});
|
|
162
|
+
|
|
163
|
+
describe('DataTable: Empty', () => {
|
|
164
|
+
it('displays the title', () => {
|
|
165
|
+
initializeDataTableEmpty();
|
|
166
|
+
|
|
167
|
+
cy.contains('Welcome').should('be.visible');
|
|
168
|
+
|
|
169
|
+
cy.makeSnapshot();
|
|
170
|
+
});
|
|
171
|
+
|
|
172
|
+
it('displays the title and the action button', () => {
|
|
173
|
+
initializeDataTableEmpty(true);
|
|
174
|
+
|
|
175
|
+
cy.contains('Welcome').should('be.visible');
|
|
176
|
+
cy.contains('Create').should('be.visible');
|
|
177
|
+
|
|
178
|
+
cy.makeSnapshot();
|
|
179
|
+
});
|
|
180
|
+
});
|
|
181
|
+
|
|
182
|
+
describe('DataTable: Listing', () => {
|
|
183
|
+
it('displays the listing', () => {
|
|
184
|
+
initializeDataTableListing();
|
|
185
|
+
|
|
186
|
+
data.forEach(({ title, description }) => {
|
|
187
|
+
cy.contains(title).should('be.visible');
|
|
188
|
+
cy.contains(description).should('be.visible');
|
|
189
|
+
});
|
|
190
|
+
|
|
191
|
+
cy.makeSnapshot();
|
|
192
|
+
});
|
|
193
|
+
});
|
|
@@ -1,5 +1,9 @@
|
|
|
1
1
|
import { Meta, StoryObj } from '@storybook/react';
|
|
2
2
|
|
|
3
|
+
import { Box } from '@mui/material';
|
|
4
|
+
|
|
5
|
+
import { ColumnType } from '../../Listing/models';
|
|
6
|
+
|
|
3
7
|
import { DataTable } from './index';
|
|
4
8
|
|
|
5
9
|
const meta: Meta<typeof DataTable> = {
|
|
@@ -49,3 +53,39 @@ export const withFixedHeightContainer: Story = {
|
|
|
49
53
|
</div>
|
|
50
54
|
)
|
|
51
55
|
};
|
|
56
|
+
|
|
57
|
+
const ListingTemplate = (args): JSX.Element => (
|
|
58
|
+
<Box sx={{ height: '80vh' }}>
|
|
59
|
+
<DataTable {...args} />
|
|
60
|
+
</Box>
|
|
61
|
+
);
|
|
62
|
+
|
|
63
|
+
export const listing: Story = {
|
|
64
|
+
args: {
|
|
65
|
+
children: (
|
|
66
|
+
<DataTable.Listing
|
|
67
|
+
columns={[
|
|
68
|
+
{
|
|
69
|
+
getFormattedString: (row) => row.title,
|
|
70
|
+
id: 'title',
|
|
71
|
+
label: 'Title',
|
|
72
|
+
type: ColumnType.string
|
|
73
|
+
},
|
|
74
|
+
{
|
|
75
|
+
getFormattedString: (row) => row.description,
|
|
76
|
+
id: 'description',
|
|
77
|
+
label: 'Description',
|
|
78
|
+
type: ColumnType.string
|
|
79
|
+
}
|
|
80
|
+
]}
|
|
81
|
+
rows={[...Array(5)].map((_, i) => ({
|
|
82
|
+
description: `Item description ${i}`,
|
|
83
|
+
id: i,
|
|
84
|
+
title: `Item ${i}`
|
|
85
|
+
}))}
|
|
86
|
+
/>
|
|
87
|
+
),
|
|
88
|
+
variant: 'listing'
|
|
89
|
+
},
|
|
90
|
+
render: ListingTemplate
|
|
91
|
+
};
|
|
@@ -10,6 +10,9 @@ const useStyles = makeStyles()((theme) => ({
|
|
|
10
10
|
gridGap: theme.spacing(2.5),
|
|
11
11
|
gridTemplateColumns: `repeat(auto-fill, ${theme.spacing(45)})`
|
|
12
12
|
},
|
|
13
|
+
'&[data-variant="listing"]': {
|
|
14
|
+
height: '100%'
|
|
15
|
+
},
|
|
13
16
|
'&[data-variant][data-is-empty="true"]': {
|
|
14
17
|
display: 'flex',
|
|
15
18
|
justifyContent: 'center',
|