@centreon/ui 24.4.1-sync-release-34022.1 → 24.4.1-test-code-coverage.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 (220) hide show
  1. package/package.json +48 -40
  2. package/src/ActionsList/ActionsList.styles.ts +40 -71
  3. package/src/Button/Icon/index.stories.tsx +1 -1
  4. package/src/Button/Icon/index.tsx +1 -1
  5. package/src/Dashboard/Dashboard.styles.ts +6 -5
  6. package/src/Dialog/Confirm/index.tsx +10 -2
  7. package/src/Dialog/UnsavedChanges/index.tsx +21 -20
  8. package/src/Dialog/UnsavedChanges/translatedLabels.ts +4 -6
  9. package/src/Dialog/index.tsx +8 -1
  10. package/src/Form/Inputs/CheckboxGroup.tsx +4 -1
  11. package/src/Form/Inputs/Text.tsx +3 -1
  12. package/src/Form/Inputs/models.ts +1 -0
  13. package/src/Graph/BarStack/BarStack.cypress.spec.tsx +139 -0
  14. package/src/Graph/BarStack/BarStack.stories.tsx +123 -0
  15. package/src/Graph/BarStack/BarStack.styles.ts +37 -0
  16. package/src/Graph/BarStack/BarStack.tsx +14 -0
  17. package/src/Graph/BarStack/ResponsiveBarStack.tsx +209 -0
  18. package/src/Graph/BarStack/index.ts +1 -0
  19. package/src/Graph/BarStack/models.ts +19 -0
  20. package/src/Graph/BarStack/useResponsiveBarStack.ts +131 -0
  21. package/src/Graph/Gauge/Gauge.cypress.spec.tsx +102 -0
  22. package/src/Graph/Gauge/Gauge.tsx +1 -1
  23. package/src/Graph/HeatMap/HeatMap.cypress.spec.tsx +145 -0
  24. package/src/Graph/HeatMap/HeatMap.stories.tsx +0 -25
  25. package/src/Graph/HeatMap/ResponsiveHeatMap.tsx +8 -2
  26. package/src/Graph/Legend/Legend.tsx +21 -0
  27. package/src/Graph/Legend/index.ts +1 -0
  28. package/src/Graph/Legend/models.ts +11 -0
  29. package/src/Graph/LineChart/BasicComponents/Lines/Threshold/index.tsx +4 -5
  30. package/src/Graph/LineChart/BasicComponents/ThresholdLine.tsx +3 -1
  31. package/src/Graph/LineChart/Header/index.tsx +3 -31
  32. package/src/Graph/LineChart/InteractiveComponents/AnchorPoint/useTickGraph.ts +9 -11
  33. package/src/Graph/LineChart/InteractiveComponents/Annotations/Annotation/index.tsx +3 -2
  34. package/src/Graph/LineChart/InteractiveComponents/GraphValueTooltip/GraphValueTooltip.tsx +68 -0
  35. package/src/Graph/LineChart/InteractiveComponents/GraphValueTooltip/useGraphValueTooltip.ts +27 -0
  36. package/src/Graph/LineChart/InteractiveComponents/GraphValueTooltip/useGraphValueTooltipStyles.ts +31 -0
  37. package/src/Graph/LineChart/InteractiveComponents/index.tsx +132 -17
  38. package/src/Graph/LineChart/InteractiveComponents/interactionWithGraphAtoms.ts +7 -27
  39. package/src/Graph/LineChart/Legend/Legend.styles.ts +5 -9
  40. package/src/Graph/LineChart/Legend/LegendHeader.tsx +10 -22
  41. package/src/Graph/LineChart/Legend/index.tsx +17 -55
  42. package/src/Graph/LineChart/LineChart.cypress.spec.tsx +91 -0
  43. package/src/Graph/LineChart/LineChart.styles.ts +8 -0
  44. package/src/Graph/LineChart/LineChart.tsx +106 -116
  45. package/src/Graph/LineChart/LoadingSkeleton.tsx +2 -2
  46. package/src/Graph/LineChart/index.tsx +6 -7
  47. package/src/Graph/LineChart/mockedData/lastDayWithIncompleteValues.json +1320 -0
  48. package/src/Graph/LineChart/mockedData/lastDayWithNullValues.json +1314 -0
  49. package/src/Graph/LineChart/models.ts +12 -1
  50. package/src/Graph/PieChart/PieChart.cypress.spec.tsx +154 -0
  51. package/src/Graph/PieChart/PieChart.stories.tsx +194 -0
  52. package/src/Graph/PieChart/PieChart.styles.ts +39 -0
  53. package/src/Graph/PieChart/PieChart.tsx +14 -0
  54. package/src/Graph/PieChart/ResponsivePie.tsx +243 -0
  55. package/src/Graph/PieChart/index.ts +1 -0
  56. package/src/Graph/PieChart/models.ts +19 -0
  57. package/src/Graph/PieChart/useResponsivePie.ts +81 -0
  58. package/src/Graph/SingleBar/SingleBar.cypress.spec.tsx +121 -0
  59. package/src/Graph/Text/Text.cypress.spec.tsx +101 -0
  60. package/src/Graph/Text/Text.styles.ts +12 -1
  61. package/src/Graph/Text/Text.tsx +17 -12
  62. package/src/Graph/Tree/DescendantNodes.tsx +89 -0
  63. package/src/Graph/Tree/Links.tsx +77 -0
  64. package/src/Graph/Tree/StandaloneTree.tsx +32 -0
  65. package/src/Graph/Tree/Tree.cypress.spec.tsx +195 -0
  66. package/src/Graph/Tree/Tree.stories.tsx +160 -0
  67. package/src/Graph/Tree/Tree.tsx +116 -0
  68. package/src/Graph/Tree/constants.ts +2 -0
  69. package/src/Graph/Tree/index.ts +4 -0
  70. package/src/Graph/Tree/models.ts +55 -0
  71. package/src/Graph/Tree/stories/contents.tsx +164 -0
  72. package/src/Graph/Tree/stories/datas.ts +305 -0
  73. package/src/Graph/Tree/utils.ts +49 -0
  74. package/src/Graph/common/testUtils.ts +71 -0
  75. package/src/Graph/common/timeSeries/index.ts +50 -12
  76. package/src/Graph/common/utils.ts +19 -0
  77. package/src/Graph/index.ts +4 -0
  78. package/src/InputField/Number/Number.cypress.spec.tsx +85 -0
  79. package/src/InputField/Number/Number.stories.tsx +66 -0
  80. package/src/InputField/Number/Number.tsx +74 -0
  81. package/src/InputField/Search/index.tsx +2 -2
  82. package/src/InputField/Select/Autocomplete/Multi/index.tsx +4 -2
  83. package/src/InputField/Select/Autocomplete/index.tsx +10 -3
  84. package/src/InputField/Select/IconPopover/index.tsx +1 -1
  85. package/src/InputField/Select/index.tsx +14 -1
  86. package/src/InputField/Text/index.tsx +38 -38
  87. package/src/Listing/ActionBar/index.tsx +10 -10
  88. package/src/Listing/Cell/DataCell.styles.ts +3 -0
  89. package/src/Listing/Cell/DataCell.tsx +8 -4
  90. package/src/Listing/Listing.cypress.spec.tsx +217 -33
  91. package/src/Listing/Listing.styles.ts +3 -5
  92. package/src/Listing/Row/Row.tsx +7 -3
  93. package/src/Listing/index.stories.tsx +25 -2
  94. package/src/Listing/index.test.tsx +1 -1
  95. package/src/Listing/index.tsx +202 -143
  96. package/src/Listing/models.ts +1 -0
  97. package/src/Listing/useStyleTable.ts +1 -0
  98. package/src/Panel/index.tsx +1 -1
  99. package/src/PopoverMenu/index.tsx +6 -5
  100. package/src/ThemeProvider/index.tsx +3 -0
  101. package/src/TimePeriods/CustomTimePeriod/CompactCustomTimePeriod.styles.ts +6 -7
  102. package/src/TimePeriods/ResolutionTimePeriod.cypress.spec.tsx +12 -9
  103. package/src/Typography/FluidTypography/FluidTypography.cypress.spec.tsx +27 -0
  104. package/src/Typography/FluidTypography/index.stories.tsx +2 -2
  105. package/src/Typography/FluidTypography/index.tsx +21 -28
  106. package/src/api/index.ts +3 -3
  107. package/src/api/useGraphQuery/index.ts +26 -5
  108. package/src/api/useGraphQuery/models.ts +5 -0
  109. package/src/api/useMutationQuery/index.test.ts +4 -4
  110. package/src/api/useMutationQuery/index.ts +24 -13
  111. package/src/components/CollapsibleItem/CollapsibleItem.cypress.spec.tsx +76 -0
  112. package/src/components/CollapsibleItem/CollapsibleItem.stories.tsx +26 -0
  113. package/src/components/CollapsibleItem/CollapsibleItem.tsx +43 -14
  114. package/src/components/CollapsibleItem/useCollapsibleItemStyles.ts +24 -1
  115. package/src/components/DataTable/DataTable.cypress.spec.tsx +14 -33
  116. package/src/components/DataTable/Item/DataTableItem.tsx +4 -60
  117. package/src/components/Form/{AccessRightsV2 → AccessRights}/AccessRights.cypress.spec.tsx +36 -13
  118. package/src/components/Form/{AccessRightsV2 → AccessRights}/ShareInput/ContactSwitch.tsx +11 -3
  119. package/src/components/Form/{AccessRightsV2 → AccessRights}/ShareInput/ShareInput.styles.ts +8 -0
  120. package/src/components/Form/{AccessRightsV2 → AccessRights}/ShareInput/ShareInput.tsx +1 -0
  121. package/src/components/Form/{AccessRightsV2 → AccessRights}/ShareInput/useShareInput.tsx +4 -0
  122. package/src/components/Form/{AccessRightsV2 → AccessRights}/models.ts +1 -0
  123. package/src/components/Form/{AccessRightsV2 → AccessRights}/storiesData.ts +23 -22
  124. package/src/components/Form/Dashboard/DashboardDuplicationForm.tsx +85 -0
  125. package/src/components/Form/Dashboard/index.ts +1 -0
  126. package/src/components/Form/FormActions.tsx +7 -2
  127. package/src/components/Form/index.ts +2 -2
  128. package/src/components/ItemComposition/Item.tsx +1 -1
  129. package/src/components/ItemComposition/ItemComposition.cypress.spec.tsx +113 -0
  130. package/src/components/ItemComposition/ItemComposition.stories.tsx +14 -0
  131. package/src/components/ItemComposition/ItemComposition.styles.ts +36 -3
  132. package/src/components/ItemComposition/ItemComposition.tsx +41 -17
  133. package/src/components/List/Item/ListItem.tsx +3 -3
  134. package/src/components/Modal/ConfirmationModal/ConfirmationModal.cypress.spec.tsx +168 -0
  135. package/src/components/Modal/ConfirmationModal/ConfirmationModal.stories.tsx +62 -0
  136. package/src/components/Modal/ConfirmationModal/ConfirmationModal.tsx +87 -0
  137. package/src/components/Modal/Modal.styles.ts +8 -3
  138. package/src/components/Modal/index.ts +2 -0
  139. package/src/components/Tooltip/ConfirmationTooltip/ConfirmationTooltip.stories.tsx +3 -3
  140. package/src/components/Tooltip/ConfirmationTooltip/ConfirmationTooltip.tsx +1 -1
  141. package/src/components/Tooltip/ConfirmationTooltip/models.ts +1 -1
  142. package/src/components/Zoom/Minimap.tsx +129 -0
  143. package/src/components/Zoom/Zoom.cypress.spec.tsx +246 -0
  144. package/src/components/Zoom/Zoom.stories.tsx +115 -0
  145. package/src/components/Zoom/Zoom.styles.tsx +68 -0
  146. package/src/components/Zoom/Zoom.tsx +64 -0
  147. package/src/components/Zoom/ZoomContent.tsx +170 -0
  148. package/src/components/Zoom/constants.ts +2 -0
  149. package/src/components/Zoom/localPoint.ts +51 -0
  150. package/src/components/Zoom/models.ts +25 -0
  151. package/src/components/Zoom/useMinimap.ts +156 -0
  152. package/src/components/Zoom/useZoom.ts +70 -0
  153. package/src/components/Zoom/utils.ts +55 -0
  154. package/src/components/index.ts +1 -0
  155. package/src/index.ts +1 -0
  156. package/src/utils/index.ts +3 -0
  157. package/src/utils/resourcesStatusURL.ts +166 -0
  158. package/src/utils/useFullscreen/Fullscreen.cypress.spec.tsx +130 -0
  159. package/src/utils/useFullscreen/atoms.ts +3 -0
  160. package/src/utils/useFullscreen/index.ts +2 -0
  161. package/src/utils/useFullscreen/translatedLabels.ts +1 -0
  162. package/src/utils/useFullscreen/useFullscreen.ts +73 -0
  163. package/src/utils/useFullscreen/useFullscreenListener.ts +62 -0
  164. package/src/utils/useInfiniteScrollListing.ts +4 -1
  165. package/src/Graph/LineChart/BasicComponents/LoadingProgress.tsx +0 -46
  166. package/src/Graph/LineChart/InteractiveComponents/AnchorPoint/TooltipAnchorPoint.tsx +0 -96
  167. package/src/Graph/LineChart/InteractiveComponents/AnchorPoint/useTooltipAnchorPoint.ts +0 -107
  168. package/src/Graph/LineChart/Legend/InteractiveValue.tsx +0 -22
  169. package/src/Graph/LineChart/Legend/useInteractiveValues.ts +0 -99
  170. package/src/Typography/FluidTypography/useFluidResizeObserver.ts +0 -56
  171. package/src/components/Form/AccessRights/AccessRights.resource.ts +0 -45
  172. package/src/components/Form/AccessRights/AccessRightsForm.stories.tsx +0 -59
  173. package/src/components/Form/AccessRights/AccessRightsForm.styles.ts +0 -21
  174. package/src/components/Form/AccessRights/AccessRightsForm.tsx +0 -67
  175. package/src/components/Form/AccessRights/AccessRightsFormActions.tsx +0 -80
  176. package/src/components/Form/AccessRights/Input/AddAction.tsx +0 -31
  177. package/src/components/Form/AccessRights/Input/ContactAccessRightInput.stories.tsx +0 -54
  178. package/src/components/Form/AccessRights/Input/ContactAccessRightInput.tsx +0 -72
  179. package/src/components/Form/AccessRights/Input/ContactAccessRightsInput.styles.ts +0 -22
  180. package/src/components/Form/AccessRights/Input/ContactInputField.tsx +0 -105
  181. package/src/components/Form/AccessRights/Input/RoleInputField.tsx +0 -29
  182. package/src/components/Form/AccessRights/List/ContactAccessRightsList.stories.tsx +0 -97
  183. package/src/components/Form/AccessRights/List/ContactAccessRightsList.styles.ts +0 -71
  184. package/src/components/Form/AccessRights/List/ContactAccessRightsList.tsx +0 -51
  185. package/src/components/Form/AccessRights/List/ContactAccessRightsListItem.stories.tsx +0 -116
  186. package/src/components/Form/AccessRights/List/ContactAccessRightsListItem.tsx +0 -118
  187. package/src/components/Form/AccessRights/List/ContactAccessRightsListItemSkeleton.tsx +0 -26
  188. package/src/components/Form/AccessRights/List/ContactAccessRightsListSkeleton.tsx +0 -28
  189. package/src/components/Form/AccessRights/Stats/AccessRightsStats.styles.ts +0 -18
  190. package/src/components/Form/AccessRights/Stats/AccessRightsStats.tsx +0 -41
  191. package/src/components/Form/AccessRights/__fixtures__/contactAccessRight.mock.ts +0 -54
  192. package/src/components/Form/AccessRights/common/GroupLabel.styles.ts +0 -18
  193. package/src/components/Form/AccessRights/common/GroupLabel.tsx +0 -15
  194. package/src/components/Form/AccessRights/common/Input.styles.ts +0 -48
  195. package/src/components/Form/AccessRights/common/RoleInputSelect.styles.ts +0 -11
  196. package/src/components/Form/AccessRights/common/RoleInputSelect.tsx +0 -57
  197. package/src/components/Form/AccessRights/index.ts +0 -3
  198. package/src/components/Form/AccessRights/useAccessRightsForm.test.tsx +0 -531
  199. package/src/components/Form/AccessRights/useAccessRightsForm.tsx +0 -282
  200. package/src/components/Form/AccessRights/useAccessRightsForm.utils.ts +0 -41
  201. /package/src/components/Form/{AccessRightsV2 → AccessRights}/AccessRights.stories.tsx +0 -0
  202. /package/src/components/Form/{AccessRightsV2 → AccessRights}/AccessRights.styles.ts +0 -0
  203. /package/src/components/Form/{AccessRightsV2 → AccessRights}/AccessRights.tsx +0 -0
  204. /package/src/components/Form/{AccessRightsV2 → AccessRights}/Actions/Actions.styles.ts +0 -0
  205. /package/src/components/Form/{AccessRightsV2 → AccessRights}/Actions/Actions.tsx +0 -0
  206. /package/src/components/Form/{AccessRightsV2 → AccessRights}/Actions/useActions.ts +0 -0
  207. /package/src/components/Form/{AccessRightsV2 → AccessRights}/List/Item.tsx +0 -0
  208. /package/src/components/Form/{AccessRightsV2 → AccessRights}/List/List.styles.tsx +0 -0
  209. /package/src/components/Form/{AccessRightsV2 → AccessRights}/List/List.tsx +0 -0
  210. /package/src/components/Form/{AccessRightsV2 → AccessRights}/List/ListItemSkeleton.tsx +0 -0
  211. /package/src/components/Form/{AccessRightsV2 → AccessRights}/List/ListSkeleton.tsx +0 -0
  212. /package/src/components/Form/{AccessRightsV2 → AccessRights}/List/RemoveAccessRight.tsx +0 -0
  213. /package/src/components/Form/{AccessRightsV2 → AccessRights}/List/StateChip.tsx +0 -0
  214. /package/src/components/Form/{AccessRightsV2 → AccessRights}/List/useItem.ts +0 -0
  215. /package/src/components/Form/{AccessRightsV2 → AccessRights}/Provider.tsx +0 -0
  216. /package/src/components/Form/{AccessRightsV2 → AccessRights}/Stats/Stats.tsx +0 -0
  217. /package/src/components/Form/{AccessRightsV2 → AccessRights}/atoms.ts +0 -0
  218. /package/src/components/Form/{AccessRightsV2 → AccessRights}/common/RoleSelectField.styles.tsx +0 -0
  219. /package/src/components/Form/{AccessRightsV2 → AccessRights}/common/RoleSelectField.tsx +0 -0
  220. /package/src/components/Form/{AccessRightsV2 → AccessRights}/useAccessRightsInitValues.ts +0 -0
@@ -0,0 +1,166 @@
1
+ import {
2
+ always,
3
+ cond,
4
+ equals,
5
+ flatten,
6
+ groupBy,
7
+ identity,
8
+ includes,
9
+ map,
10
+ pipe,
11
+ T
12
+ } from 'ramda';
13
+
14
+ import { SelectEntry } from '..';
15
+
16
+ import { centreonBaseURL } from './centreonBaseURL';
17
+
18
+ export interface Resource {
19
+ resourceType: string;
20
+ resources: Array<SelectEntry>;
21
+ }
22
+
23
+ interface GetResourcesUrlProps {
24
+ allResources: Array<Resource>;
25
+ isForOneResource: boolean;
26
+ resource?;
27
+ states: Array<string>;
28
+ statuses: Array<string>;
29
+ type: string;
30
+ }
31
+
32
+ export const getDetailsPanelQueriers = ({ resource, type }): object => {
33
+ const { id, parentId, uuid } = resource;
34
+
35
+ const resourcesDetailsEndpoint = cond([
36
+ [
37
+ equals('host'),
38
+ always(`${centreonBaseURL}/api/latest/monitoring/resources/hosts/${id}`)
39
+ ],
40
+ [
41
+ equals('service'),
42
+ always(
43
+ `${centreonBaseURL}/api/latest/monitoring/resources/hosts/${parentId}/services/${id}`
44
+ )
45
+ ],
46
+ [
47
+ equals('metaservice'),
48
+ always(
49
+ `${centreonBaseURL}/api/latest/monitoring/resources/metaservices/${id}`
50
+ )
51
+ ],
52
+ [
53
+ equals('anomaly-detection'),
54
+ always(
55
+ `${centreonBaseURL}/api/latest/monitoring/resources/anomaly-detection/${id}`
56
+ )
57
+ ]
58
+ ])(type);
59
+
60
+ const queryParameters = {
61
+ id,
62
+ resourcesDetailsEndpoint,
63
+ selectedTimePeriodId: 'last_24_h',
64
+ tab: 'details',
65
+ tabParameters: {},
66
+ uuid
67
+ };
68
+
69
+ return queryParameters;
70
+ };
71
+
72
+ export const getResourcesUrl = ({
73
+ type,
74
+ statuses,
75
+ states,
76
+ allResources,
77
+ isForOneResource,
78
+ resource
79
+ }: GetResourcesUrlProps): string => {
80
+ const resourcesCriterias = equals(type, 'all')
81
+ ? {
82
+ name: 'resource_types',
83
+ value: [
84
+ { id: 'service', name: 'Service' },
85
+ { id: 'host', name: 'Host' }
86
+ ]
87
+ }
88
+ : {
89
+ name: 'resource_types',
90
+ value: [
91
+ { id: type, name: `${type.charAt(0).toUpperCase()}${type.slice(1)}` }
92
+ ]
93
+ };
94
+
95
+ const formattedStatuses = pipe(
96
+ flatten,
97
+ map((status: string) => {
98
+ return {
99
+ id: status.toLocaleUpperCase(),
100
+ name: `${status.charAt(0).toUpperCase()}${status.slice(1)}`
101
+ };
102
+ })
103
+ )(statuses);
104
+
105
+ const formattedStates = states.map((state) => {
106
+ return {
107
+ id: state,
108
+ name: `${state.charAt(0).toUpperCase()}${state.slice(1)}`
109
+ };
110
+ });
111
+
112
+ const groupedResources = groupBy(
113
+ ({ resourceType }) => resourceType,
114
+ allResources
115
+ );
116
+
117
+ const resourcesFilters = Object.entries(groupedResources).map(
118
+ ([resourceType, res]) => {
119
+ const name = cond<Array<string>, string>([
120
+ [equals('host'), always('parent_name')],
121
+ [equals('service'), always('name')],
122
+ [T, identity]
123
+ ])(resourceType);
124
+
125
+ return {
126
+ name: name.replace('-', '_'),
127
+ value: flatten(
128
+ (res || []).map(({ resources: subResources }) => {
129
+ return subResources.map(({ name: resourceName }) => ({
130
+ id: includes(name, ['name', 'parent_name'])
131
+ ? `\\b${resourceName}\\b`
132
+ : resourceName,
133
+ name: resourceName
134
+ }));
135
+ })
136
+ )
137
+ };
138
+ }
139
+ );
140
+
141
+ const filterQueryParameter = {
142
+ criterias: [
143
+ resourcesCriterias,
144
+ { name: 'statuses', value: formattedStatuses },
145
+ { name: 'states', value: formattedStates },
146
+ ...resourcesFilters,
147
+ { name: 'search', value: '' }
148
+ ]
149
+ };
150
+
151
+ const encodedFilterParams = encodeURIComponent(
152
+ JSON.stringify(filterQueryParameter)
153
+ );
154
+
155
+ if (!isForOneResource) {
156
+ return `/monitoring/resources?filter=${encodedFilterParams}&fromTopCounter=true`;
157
+ }
158
+
159
+ const detailsPanelQueriers = getDetailsPanelQueriers({ resource, type });
160
+
161
+ const encodedDetailsParams = encodeURIComponent(
162
+ JSON.stringify(detailsPanelQueriers)
163
+ );
164
+
165
+ return `/monitoring/resources?details=${encodedDetailsParams}&filter=${encodedFilterParams}&fromTopCounter=true`;
166
+ };
@@ -0,0 +1,130 @@
1
+ import { Provider, createStore } from 'jotai';
2
+ import { BrowserRouter } from 'react-router-dom';
3
+
4
+ import { Button } from '@mui/material';
5
+
6
+ import { useFullscreen } from './useFullscreen';
7
+ import { router, useFullscreenListener } from './useFullscreenListener';
8
+
9
+ const labelEnterFullscreen = 'Enter fullscreen';
10
+ const labelExitFullscreen = 'Exit fullscreen';
11
+
12
+ const ChildComponent = (): JSX.Element => {
13
+ const { toggleFullscreen, fullscreenEnabled, isFullscreenActivated } =
14
+ useFullscreen();
15
+
16
+ return (
17
+ <div
18
+ data-fullscreenActivated={isFullscreenActivated}
19
+ data-fullscreenEnabled={fullscreenEnabled}
20
+ id="test"
21
+ >
22
+ <Button onClick={() => toggleFullscreen(document.body)}>
23
+ {isFullscreenActivated ? labelExitFullscreen : labelEnterFullscreen}
24
+ </Button>
25
+ <input id="input" />
26
+ <textarea id="textarea" />
27
+ </div>
28
+ );
29
+ };
30
+
31
+ const TestComponent = (): JSX.Element => {
32
+ useFullscreenListener();
33
+
34
+ return <ChildComponent />;
35
+ };
36
+
37
+ const initialize = (): void => {
38
+ const store = createStore();
39
+
40
+ const queryParameters = new Map();
41
+
42
+ cy.stub(router, 'useSearchParams', () => queryParameters);
43
+
44
+ cy.mount({
45
+ Component: (
46
+ <BrowserRouter>
47
+ <Provider store={store}>
48
+ <TestComponent />
49
+ </Provider>
50
+ </BrowserRouter>
51
+ )
52
+ });
53
+ };
54
+
55
+ describe('Fullscreen', () => {
56
+ it('enters fullscreen mode when the button is clicked', () => {
57
+ initialize();
58
+
59
+ cy.get('#test')
60
+ .should('have.attr', 'data-fullscreenActivated', 'false')
61
+ .should('have.attr', 'data-fullscreenEnabled', 'true');
62
+
63
+ cy.contains(labelEnterFullscreen).realClick();
64
+
65
+ cy.get('#test')
66
+ .should('have.attr', 'data-fullscreenActivated', 'true')
67
+ .should('have.attr', 'data-fullscreenEnabled', 'true');
68
+
69
+ cy.contains(labelExitFullscreen).realClick();
70
+ });
71
+
72
+ it('exits fullscreen mode when the button is clicked', () => {
73
+ initialize();
74
+
75
+ cy.get('#test')
76
+ .should('have.attr', 'data-fullscreenActivated', 'false')
77
+ .should('have.attr', 'data-fullscreenEnabled', 'true');
78
+
79
+ cy.contains(labelEnterFullscreen).realClick();
80
+
81
+ cy.get('#test')
82
+ .should('have.attr', 'data-fullscreenActivated', 'true')
83
+ .should('have.attr', 'data-fullscreenEnabled', 'true');
84
+
85
+ cy.contains(labelExitFullscreen).realClick();
86
+
87
+ cy.get('#test')
88
+ .should('have.attr', 'data-fullscreenActivated', 'false')
89
+ .should('have.attr', 'data-fullscreenEnabled', 'true');
90
+ });
91
+
92
+ it('toggles fullscreen mode when the corresponding shortcut is clicked', () => {
93
+ initialize();
94
+
95
+ cy.get('#test')
96
+ .should('have.attr', 'data-fullscreenActivated', 'false')
97
+ .should('have.attr', 'data-fullscreenEnabled', 'true');
98
+
99
+ cy.get('#test').realPress(['F']);
100
+
101
+ cy.get('#test')
102
+ .should('have.attr', 'data-fullscreenActivated', 'true')
103
+ .should('have.attr', 'data-fullscreenEnabled', 'true');
104
+
105
+ cy.get('#test').realPress(['F']);
106
+
107
+ cy.get('#test')
108
+ .should('have.attr', 'data-fullscreenActivated', 'false')
109
+ .should('have.attr', 'data-fullscreenEnabled', 'true');
110
+ });
111
+
112
+ ['input', 'textarea'].forEach((tag) => {
113
+ it(`cannot toggle fullscreen feature using the shortcut when ${tag === 'input' ? 'an' : 'a'} ${tag} is focused`, () => {
114
+ initialize();
115
+
116
+ cy.get('#test')
117
+ .should('have.attr', 'data-fullscreenActivated', 'false')
118
+ .should('have.attr', 'data-fullscreenEnabled', 'true');
119
+
120
+ cy.get(`#${tag}`).focus();
121
+
122
+ cy.get('#test').realPress(['F']);
123
+
124
+ cy.get('#test')
125
+ .should('have.attr', 'data-fullscreenActivated', 'false')
126
+ .should('have.attr', 'data-fullscreenEnabled', 'true');
127
+ cy.get(`#${tag}`).should('have.value', 'F');
128
+ });
129
+ });
130
+ });
@@ -0,0 +1,3 @@
1
+ import { atom } from 'jotai';
2
+
3
+ export const isFullscreenActivatedAtom = atom(false);
@@ -0,0 +1,2 @@
1
+ export * from './useFullscreen';
2
+ export * from './useFullscreenListener';
@@ -0,0 +1 @@
1
+ export const labelCannotEnterInFullscreen = 'Cannot enter fullscreen mode';
@@ -0,0 +1,73 @@
1
+ import { useTranslation } from 'react-i18next';
2
+ import { useAtom } from 'jotai';
3
+
4
+ import { useSnackbar } from '../..';
5
+
6
+ import { labelCannotEnterInFullscreen } from './translatedLabels';
7
+ import { isFullscreenActivatedAtom } from './atoms';
8
+
9
+ interface UseFullscreenState {
10
+ enterInFullscreen: (element: HTMLElement | null) => void;
11
+ exitFullscreen: () => void;
12
+ fullscreenEnabled: boolean;
13
+ isFullscreenActivated: boolean;
14
+ resetVariables: () => void;
15
+ toggleFullscreen: (element: HTMLElement | null) => void;
16
+ }
17
+
18
+ export const useFullscreen = (): UseFullscreenState => {
19
+ const { t } = useTranslation();
20
+
21
+ const { showErrorMessage } = useSnackbar();
22
+
23
+ const [isFullscreenActivated, setIsFullscreenActivated] = useAtom(
24
+ isFullscreenActivatedAtom
25
+ );
26
+
27
+ const resetVariables = (): void => {
28
+ setIsFullscreenActivated(false);
29
+ };
30
+
31
+ const enterInFullscreen = (element: HTMLElement | null): void => {
32
+ if (!document.fullscreenEnabled) {
33
+ return;
34
+ }
35
+
36
+ if (!element) {
37
+ showErrorMessage(t(labelCannotEnterInFullscreen));
38
+ }
39
+
40
+ element
41
+ ?.requestFullscreen({ navigationUI: 'show' })
42
+ .then(() => {
43
+ setIsFullscreenActivated(true);
44
+ })
45
+ .catch(() => {
46
+ showErrorMessage(t(labelCannotEnterInFullscreen));
47
+ setIsFullscreenActivated(false);
48
+ });
49
+ };
50
+
51
+ const exitFullscreen = (): void => {
52
+ document.exitFullscreen().then(resetVariables);
53
+ };
54
+
55
+ const toggleFullscreen = (element: HTMLElement | null): void => {
56
+ if (isFullscreenActivated || document.fullscreenElement) {
57
+ exitFullscreen();
58
+
59
+ return;
60
+ }
61
+
62
+ enterInFullscreen(element);
63
+ };
64
+
65
+ return {
66
+ enterInFullscreen,
67
+ exitFullscreen,
68
+ fullscreenEnabled: document.fullscreenEnabled,
69
+ isFullscreenActivated,
70
+ resetVariables,
71
+ toggleFullscreen
72
+ };
73
+ };
@@ -0,0 +1,62 @@
1
+ import { useEffect } from 'react';
2
+
3
+ import { equals, includes } from 'ramda';
4
+ import { useSearchParams } from 'react-router-dom';
5
+
6
+ import { useDeepCompare } from '../useMemoComponent';
7
+
8
+ import { useFullscreen } from './useFullscreen';
9
+
10
+ export const router = {
11
+ useSearchParams
12
+ };
13
+
14
+ export const useFullscreenListener = (): boolean => {
15
+ const { toggleFullscreen, resetVariables, isFullscreenActivated } =
16
+ useFullscreen();
17
+
18
+ const toggle = (event: KeyboardEvent): void => {
19
+ if (
20
+ includes(document.activeElement?.tagName, ['INPUT', 'TEXTAREA']) ||
21
+ equals(
22
+ document.activeElement?.getAttribute('data-lexical-editor'),
23
+ 'true'
24
+ ) ||
25
+ equals(document.activeElement?.getAttribute('contenteditable'), 'true') ||
26
+ !equals(event.code, 'KeyF')
27
+ ) {
28
+ return;
29
+ }
30
+
31
+ toggleFullscreen(document.querySelector('body'));
32
+ };
33
+
34
+ const changeFullscreen = (): void => {
35
+ if (document.fullscreenElement) {
36
+ return;
37
+ }
38
+
39
+ resetVariables();
40
+ };
41
+
42
+ useEffect(
43
+ () => {
44
+ document.addEventListener('fullscreenchange', changeFullscreen);
45
+
46
+ return () => {
47
+ document.removeEventListener('fullscreenchange', changeFullscreen);
48
+ };
49
+ },
50
+ useDeepCompare([document.fullscreenElement])
51
+ );
52
+
53
+ useEffect(() => {
54
+ window.addEventListener('keypress', toggle);
55
+
56
+ return () => {
57
+ window.removeEventListener('keypress', toggle);
58
+ };
59
+ }, [isFullscreenActivated]);
60
+
61
+ return isFullscreenActivated;
62
+ };
@@ -24,6 +24,7 @@ interface UseInfiniteScrollListing<T> {
24
24
  interface UseInfiniteScrollListingProps<T> {
25
25
  customQueryParameters?: Array<QueryParameter>;
26
26
  decoder?: JsonDecoder.Decoder<Listing<T>>;
27
+ enabled?: boolean;
27
28
  endpoint: string;
28
29
  limit?: number;
29
30
  pageAtom: PrimitiveAtom<number>;
@@ -40,7 +41,8 @@ export const useInfiniteScrollListing = <T>({
40
41
  suspense = true,
41
42
  parameters,
42
43
  customQueryParameters,
43
- limit = 100
44
+ limit = 100,
45
+ enabled = true
44
46
  }: UseInfiniteScrollListingProps<T>): UseInfiniteScrollListing<T> => {
45
47
  const [maxPage, setMaxPage] = useState(1);
46
48
 
@@ -61,6 +63,7 @@ export const useInfiniteScrollListing = <T>({
61
63
  getQueryKey: () => [queryKeyName, page],
62
64
  isPaginated: true,
63
65
  queryOptions: {
66
+ enabled,
64
67
  refetchOnMount: false,
65
68
  refetchOnWindowFocus: false,
66
69
  suspense: suspense && equals(page, 1)
@@ -1,46 +0,0 @@
1
- import { makeStyles } from 'tss-react/mui';
2
-
3
- import { CircularProgress, alpha } from '@mui/material';
4
-
5
- interface StyleProps {
6
- height: number;
7
- width: number;
8
- }
9
-
10
- const useStyles = makeStyles<StyleProps>()((theme, { height, width }) => ({
11
- graphLoader: {
12
- alignItems: 'center',
13
- backgroundColor: alpha(theme.palette.common.white, 0.5),
14
- display: 'flex',
15
- height,
16
- justifyContent: 'center',
17
- position: 'absolute',
18
- width
19
- }
20
- }));
21
-
22
- interface LoadingProgress {
23
- display: boolean;
24
- height: number;
25
- width: number;
26
- }
27
-
28
- const LoadingProgress = ({
29
- display,
30
- height,
31
- width
32
- }: LoadingProgress): JSX.Element | null => {
33
- const { classes } = useStyles({ height, width });
34
-
35
- if (!display) {
36
- return null;
37
- }
38
-
39
- return (
40
- <div className={classes.graphLoader}>
41
- <CircularProgress />
42
- </div>
43
- );
44
- };
45
-
46
- export default LoadingProgress;
@@ -1,96 +0,0 @@
1
- import { Tooltip } from '@visx/visx';
2
-
3
- import { Typography, useTheme } from '@mui/material';
4
-
5
- import useTooltipAnchorPoint from './useTooltipAnchorPoint';
6
- import { TooltipAnchorModel } from './models';
7
-
8
- const baseStyles = {
9
- ...Tooltip.defaultStyles,
10
- textAlign: 'center'
11
- };
12
-
13
- const TooltipAnchorPoint = ({
14
- timeSeries,
15
- xScale,
16
- graphHeight,
17
- leftScale,
18
- rightScale,
19
- graphWidth,
20
- lines,
21
- baseAxis
22
- }: TooltipAnchorModel): JSX.Element => {
23
- const theme = useTheme();
24
-
25
- const {
26
- tooltipDataAxisX,
27
- tooltipDataAxisYLeft,
28
- tooltipLeftAxisX,
29
- tooltipLeftAxisYLeft,
30
- tooltipTopAxisYLeft,
31
- tooltipDataAxisYRight,
32
- tooltipTopAxisYRight,
33
- tooltipLeftAxisYRight
34
- } = useTooltipAnchorPoint({
35
- baseAxis,
36
- graphHeight,
37
- graphWidth,
38
- leftScale,
39
- lines,
40
- rightScale,
41
- timeSeries,
42
- xScale
43
- });
44
-
45
- const cardStyles = {
46
- backgroundColor: theme.palette.background.paper,
47
- color: theme.palette.text.primary,
48
- padding: theme.spacing(0.25, 0.5)
49
- };
50
-
51
- return (
52
- <>
53
- {tooltipDataAxisX && (
54
- <Tooltip.Tooltip
55
- left={tooltipLeftAxisX}
56
- style={{
57
- ...baseStyles,
58
- ...cardStyles,
59
- transform: 'translateX(-70%)'
60
- }}
61
- top={0}
62
- >
63
- <Typography variant="caption">{tooltipDataAxisX}</Typography>
64
- </Tooltip.Tooltip>
65
- )}
66
- {tooltipDataAxisYLeft && (
67
- <Tooltip.Tooltip
68
- left={tooltipLeftAxisYLeft}
69
- style={{
70
- ...baseStyles,
71
- ...cardStyles,
72
- transform: 'translateX(-70%) translateY(-100%)'
73
- }}
74
- top={tooltipTopAxisYLeft}
75
- >
76
- <Typography variant="caption">{tooltipDataAxisYLeft}</Typography>
77
- </Tooltip.Tooltip>
78
- )}
79
- {tooltipDataAxisYRight && (
80
- <Tooltip.Tooltip
81
- left={tooltipLeftAxisYRight}
82
- style={{
83
- ...baseStyles,
84
- ...cardStyles,
85
- transform: 'translateX(-70%) translateY(-80%)'
86
- }}
87
- top={tooltipTopAxisYRight}
88
- >
89
- <Typography variant="caption">{tooltipDataAxisYRight}</Typography>
90
- </Tooltip.Tooltip>
91
- )}
92
- </>
93
- );
94
- };
95
-
96
- export default TooltipAnchorPoint;