@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,129 @@
1
+ import { useMemo } from 'react';
2
+
3
+ import { scaleLinear } from '@visx/scale';
4
+
5
+ import { minimapScale, radius } from './constants';
6
+ import { UseMinimapProps, useMinimap } from './useMinimap';
7
+ import { useZoomStyles } from './Zoom.styles';
8
+
9
+ interface Props extends Omit<UseMinimapProps, 'minimapScale' | 'scale'> {
10
+ children: JSX.Element;
11
+ contentClientRect: {
12
+ height: number;
13
+ width: number;
14
+ };
15
+ diffBetweenContentAndSvg: {
16
+ left: number;
17
+ top: number;
18
+ };
19
+ id?: number | string;
20
+ isDraggingFromContainer: boolean;
21
+ }
22
+
23
+ const Minimap = ({
24
+ zoom,
25
+ children,
26
+ height,
27
+ width,
28
+ contentClientRect,
29
+ isDraggingFromContainer,
30
+ diffBetweenContentAndSvg,
31
+ id
32
+ }: Props): JSX.Element => {
33
+ const { classes } = useZoomStyles();
34
+
35
+ const yMinimapScale = useMemo(
36
+ () => contentClientRect.height / zoom.transformMatrix.scaleY / height,
37
+ [contentClientRect.height, height]
38
+ );
39
+ const xMinimapScale = useMemo(
40
+ () => contentClientRect.width / zoom.transformMatrix.scaleX / width,
41
+ [contentClientRect.width, width]
42
+ );
43
+
44
+ const scale = Math.max(yMinimapScale, xMinimapScale);
45
+ const invertedScale = 1 / scale;
46
+ const scaleToUse = (invertedScale > 1 ? 1 : invertedScale) || 1;
47
+
48
+ const { move, zoomInOut, dragStart, dragEnd } = useMinimap({
49
+ height,
50
+ isDraggingFromContainer,
51
+ minimapScale,
52
+ scale: (invertedScale > 1 ? 1 : scale) || 1,
53
+ width,
54
+ zoom
55
+ });
56
+
57
+ const finalHeight = height;
58
+ const finalWidth = width;
59
+
60
+ const additionalScaleScale = scaleLinear({
61
+ clamp: true,
62
+ domain: [contentClientRect.height, 0],
63
+ range: [0, 0.05]
64
+ });
65
+
66
+ const additionalScale =
67
+ additionalScaleScale(contentClientRect.height - height) /
68
+ 2 /
69
+ zoom.transformMatrix.scaleY;
70
+
71
+ const translateX = useMemo(
72
+ () =>
73
+ -diffBetweenContentAndSvg.left /
74
+ zoom.transformMatrix.scaleX /
75
+ minimapScale,
76
+ [diffBetweenContentAndSvg.left]
77
+ );
78
+ const translateY = useMemo(
79
+ () =>
80
+ -diffBetweenContentAndSvg.top /
81
+ zoom.transformMatrix.scaleX /
82
+ minimapScale,
83
+ [diffBetweenContentAndSvg.top]
84
+ );
85
+
86
+ return (
87
+ <g className={classes.minimap} clipPath={`url(#zoom-clip-${id})`}>
88
+ <rect
89
+ className={classes.minimapBackground}
90
+ height={finalHeight}
91
+ rx={radius}
92
+ width={finalWidth}
93
+ />
94
+ <g
95
+ className={classes.movingZone}
96
+ style={{
97
+ transform: `scale(${scaleToUse - additionalScale}) translate(${translateX}px, ${translateY}px)`
98
+ }}
99
+ >
100
+ {children}
101
+ <g>
102
+ <rect
103
+ className={classes.minimapZoom}
104
+ fillOpacity={0.2}
105
+ height={height}
106
+ rx={radius}
107
+ transform={zoom.toStringInvert()}
108
+ width={width}
109
+ />
110
+ </g>
111
+ </g>
112
+ <rect
113
+ data-testid="minimap-interaction"
114
+ fill="transparent"
115
+ height={finalHeight}
116
+ rx={radius}
117
+ width={finalWidth}
118
+ onMouseDown={dragStart}
119
+ onMouseEnter={dragStart}
120
+ onMouseLeave={dragEnd}
121
+ onMouseMove={move}
122
+ onMouseUp={dragEnd}
123
+ onWheel={zoomInOut}
124
+ />
125
+ </g>
126
+ );
127
+ };
128
+
129
+ export default Minimap;
@@ -0,0 +1,246 @@
1
+ import Zoom from './Zoom';
2
+
3
+ const Content = (): JSX.Element => (
4
+ <g style={{ transform: 'translate(300px, 150px)' }}>
5
+ <circle fill="blue" r={50} stroke="black" />
6
+ </g>
7
+ );
8
+
9
+ const ContentWithMultipleShapes = (): JSX.Element => {
10
+ return (
11
+ <g>
12
+ <g style={{ transform: 'translate(300px, 150px)' }}>
13
+ <circle fill="blue" r={50} stroke="black" />
14
+ </g>
15
+ <g style={{ transform: 'translate(600px, 500px)' }}>
16
+ <circle fill="green" r={70} />
17
+ </g>
18
+ <g style={{ transform: 'translate(150px, 600px)' }}>
19
+ <circle fill="red" r={70} />
20
+ </g>
21
+ </g>
22
+ );
23
+ };
24
+ const ContentWithMultipleShapesWithNegativeTranslations = (): JSX.Element => {
25
+ return (
26
+ <g>
27
+ <g style={{ transform: 'translate(-300px, -150px)' }}>
28
+ <circle fill="blue" r={50} stroke="black" />
29
+ </g>
30
+ <g style={{ transform: 'translate(600px, 500px)' }}>
31
+ <circle fill="green" r={70} />
32
+ </g>
33
+ <g style={{ transform: 'translate(150px, 600px)' }}>
34
+ <circle fill="red" r={70} />
35
+ </g>
36
+ </g>
37
+ );
38
+ };
39
+
40
+ interface Props {
41
+ minimapPosition?;
42
+ showMinimap: boolean;
43
+ tenplate?: () => JSX.Element;
44
+ }
45
+
46
+ const initialize = ({
47
+ showMinimap,
48
+ minimapPosition,
49
+ template = Content
50
+ }: Props): void => {
51
+ cy.mount({
52
+ Component: (
53
+ <div style={{ height: '400px', width: '100%' }}>
54
+ <Zoom minimapPosition={minimapPosition} showMinimap={showMinimap}>
55
+ {template}
56
+ </Zoom>
57
+ </div>
58
+ )
59
+ });
60
+ };
61
+
62
+ describe('Zoom', () => {
63
+ it('displays the minimap when the prop is set', () => {
64
+ initialize({ showMinimap: true });
65
+
66
+ cy.get('g[clip-path="url(#zoom-clip-0)"]').should('be.visible');
67
+
68
+ cy.makeSnapshot();
69
+ });
70
+
71
+ it('zooms in when the corresponding buttom is clicked', () => {
72
+ initialize({ showMinimap: true });
73
+
74
+ cy.get('g[transform="matrix(1, 0, 0, 1, 0, 0)"]');
75
+ cy.get('g[clip-path="url(#zoom-clip-0)"]').should('be.visible');
76
+
77
+ cy.findByTestId('zoom in').click();
78
+ cy.findByTestId('zoom-content')
79
+ .should('have.attr', 'transform')
80
+ .and('include', '1.2, 0, 0, 1.2');
81
+
82
+ cy.makeSnapshot();
83
+ });
84
+
85
+ it('zooms out when the corresponding buttom is clicked', () => {
86
+ initialize({ showMinimap: true });
87
+
88
+ cy.get('g[transform="matrix(1, 0, 0, 1, 0, 0)"]');
89
+ cy.get('g[clip-path="url(#zoom-clip-0)"]').should('be.visible');
90
+
91
+ cy.findByTestId('zoom out').click();
92
+
93
+ cy.findByTestId('zoom-content')
94
+ .should('have.attr', 'transform')
95
+ .and('include', '0.8, 0, 0, 0.8');
96
+
97
+ cy.makeSnapshot();
98
+ });
99
+
100
+ it('zooms out when the content is scrolled up', () => {
101
+ initialize({ showMinimap: true });
102
+
103
+ cy.get('g[transform="matrix(1, 0, 0, 1, 0, 0)"]');
104
+ cy.get('g[clip-path="url(#zoom-clip-0)"]').should('be.visible');
105
+
106
+ cy.findByTestId('zoom-content').realMouseWheel({ deltaY: 20 });
107
+
108
+ cy.findByTestId('zoom-content')
109
+ .should('have.attr', 'transform')
110
+ .and('include', '0.9, 0, 0, 0.9');
111
+
112
+ cy.makeSnapshot();
113
+ });
114
+
115
+ it('zooms in when the content is scrolled down', () => {
116
+ initialize({ showMinimap: true });
117
+
118
+ cy.get('g[transform="matrix(1, 0, 0, 1, 0, 0)"]');
119
+ cy.get('g[clip-path="url(#zoom-clip-0)"]').should('be.visible');
120
+
121
+ cy.findByTestId('zoom-content').realMouseWheel({ deltaY: -20 });
122
+
123
+ cy.findByTestId('zoom-content')
124
+ .should('have.attr', 'transform')
125
+ .and('include', '1.1, 0, 0, 1.1');
126
+
127
+ cy.makeSnapshot();
128
+ });
129
+
130
+ it('clears the zoom when the corresponding button is clicked', () => {
131
+ initialize({ showMinimap: true });
132
+
133
+ cy.get('g[transform="matrix(1, 0, 0, 1, 0, 0)"]');
134
+ cy.get('g[clip-path="url(#zoom-clip-0)"]').should('be.visible');
135
+
136
+ cy.findByTestId('zoom-content').realMouseWheel({ deltaY: -20 });
137
+
138
+ cy.findByTestId('zoom-content')
139
+ .should('have.attr', 'transform')
140
+ .and('include', '1.1, 0, 0, 1.1');
141
+
142
+ cy.findByTestId('clear').click();
143
+
144
+ cy.get('g[transform="matrix(1, 0, 0, 1, 0, 0)"]');
145
+
146
+ cy.makeSnapshot();
147
+ });
148
+
149
+ it('does not display the minimap when the prop is set', () => {
150
+ initialize({ showMinimap: false });
151
+
152
+ cy.get('g[transform="matrix(1, 0, 0, 1, 0, 0)"]');
153
+ cy.get('g[clip-path="url(#zoom-clip-0)"]').should('not.exist');
154
+
155
+ cy.makeSnapshot();
156
+ });
157
+
158
+ it('zooms in when the minimap is scrolled up', () => {
159
+ initialize({ showMinimap: true });
160
+
161
+ cy.get('g[clip-path="url(#zoom-clip-0)"]').should('be.visible');
162
+
163
+ cy.findByTestId('minimap-interaction').realMouseWheel({ deltaY: -20 });
164
+
165
+ cy.findByTestId('zoom-content')
166
+ .should('have.attr', 'transform')
167
+ .and('include', '1.1, 0, 0, 1.1');
168
+
169
+ cy.makeSnapshot();
170
+ });
171
+
172
+ it('zooms out when the minimap is scrolled down', () => {
173
+ initialize({ showMinimap: true });
174
+
175
+ cy.get('g[clip-path="url(#zoom-clip-0)"]').should('be.visible');
176
+
177
+ cy.findByTestId('minimap-interaction').realMouseWheel({ deltaY: 20 });
178
+
179
+ cy.findByTestId('zoom-content')
180
+ .should('have.attr', 'transform')
181
+ .and('include', '0.9, 0, 0, 0.9');
182
+
183
+ cy.makeSnapshot();
184
+ });
185
+
186
+ it('moves the view when the mouse is hover the content with the corresponding button pressed down', () => {
187
+ initialize({ showMinimap: true });
188
+
189
+ cy.get('g[clip-path="url(#zoom-clip-0)"]').should('be.visible');
190
+ cy.get('svg').should('have.attr', 'height', '400');
191
+
192
+ cy.findByTestId('zoom-container')
193
+ .trigger('mousedown', 400, 200)
194
+ .trigger('mousemove', 600, 200);
195
+
196
+ cy.findByTestId('zoom-content').should(
197
+ 'have.attr',
198
+ 'transform',
199
+ 'matrix(1, 0, 0, 1, 0, 0)'
200
+ );
201
+
202
+ cy.makeSnapshot();
203
+ });
204
+
205
+ it('displays the minimap in the bottom right when the prop to the corresponding value', () => {
206
+ initialize({ minimapPosition: 'bottom-right', showMinimap: true });
207
+
208
+ cy.get('g[clip-path="url(#zoom-clip-0)"]').should('be.visible');
209
+ cy.get('svg').should('have.attr', 'height', '400');
210
+
211
+ cy.makeSnapshot();
212
+ });
213
+
214
+ it('applies a scale down on the minimap when the content is higher than the original height', () => {
215
+ initialize({ showMinimap: true, template: ContentWithMultipleShapes });
216
+
217
+ cy.get('g[clip-path="url(#zoom-clip-0)"]').should('be.visible');
218
+ cy.get('svg').should('have.attr', 'height', '400');
219
+
220
+ cy.findByTestId('minimap-interaction')
221
+ .parent()
222
+ .find('g')
223
+ .should('have.attr', 'style')
224
+ .and('include', 'transform: scale(0.684211) translate(-85px, -105px);');
225
+
226
+ cy.makeSnapshot();
227
+ });
228
+
229
+ it('applies a scale down on the minimap when the content has negative translation values', () => {
230
+ initialize({
231
+ showMinimap: true,
232
+ template: ContentWithMultipleShapesWithNegativeTranslations
233
+ });
234
+
235
+ cy.get('g[clip-path="url(#zoom-clip-0)"]').should('be.visible');
236
+ cy.get('svg').should('have.attr', 'height', '400');
237
+
238
+ cy.findByTestId('minimap-interaction')
239
+ .parent()
240
+ .find('g')
241
+ .should('have.attr', 'style')
242
+ .and('include', 'transform: scale(0.448276) translate(345px, 195px);');
243
+
244
+ cy.makeSnapshot();
245
+ });
246
+ });
@@ -0,0 +1,115 @@
1
+ import { Meta, StoryObj } from '@storybook/react';
2
+
3
+ import Zoom, { ZoomProps } from './Zoom';
4
+
5
+ const meta: Meta<typeof Zoom> = {
6
+ argTypes: {
7
+ minimapPosition: {
8
+ control: 'select',
9
+ options: ['top-left', 'top-right', 'bottom-left', 'bottom-right']
10
+ },
11
+ scaleMax: {
12
+ control: { max: 16, min: 0.4, step: 0.2, type: 'range' }
13
+ },
14
+ scaleMin: {
15
+ control: { max: 16, min: 0.4, step: 0.2, type: 'range' }
16
+ },
17
+ showMinimap: {
18
+ control: 'boolean'
19
+ }
20
+ },
21
+ component: Zoom
22
+ };
23
+
24
+ export default meta;
25
+ type Story = StoryObj<typeof Zoom>;
26
+
27
+ const Content = (): JSX.Element => {
28
+ return (
29
+ <g style={{ transform: 'translate(-150px, -100px)' }}>
30
+ <g style={{ transform: 'translate(300px, 150px)' }}>
31
+ <circle fill="blue" r={50} stroke="black" />
32
+ </g>
33
+ <g style={{ transform: 'translate(600px, 400px)' }}>
34
+ <circle fill="green" r={70} />
35
+ </g>
36
+ <g style={{ transform: `translate(2400px, 1400px)` }}>
37
+ <circle fill="red" r={70} />
38
+ </g>
39
+ </g>
40
+ );
41
+ };
42
+
43
+ const Template = ({ children, ...args }: ZoomProps): JSX.Element => (
44
+ <div style={{ height: '400px', width: '100%' }}>
45
+ <Zoom {...args}>{children}</Zoom>
46
+ </div>
47
+ );
48
+
49
+ const TemplateResponsive = ({
50
+ redCircleXPosition,
51
+ redCircleYPosition,
52
+ ...args
53
+ }: ZoomProps): JSX.Element => (
54
+ <div style={{ height: '90vh', width: '100%' }}>
55
+ <Zoom {...args}>
56
+ {() => (
57
+ <g>
58
+ <g style={{ transform: 'translate(500px, 100px)' }}>
59
+ <circle fill="blue" r={50} stroke="black" />
60
+ </g>
61
+ <g style={{ transform: 'translate(600px, 400px)' }}>
62
+ <circle fill="green" r={70} />
63
+ </g>
64
+ <g
65
+ style={{
66
+ transform: `translate(${redCircleXPosition || 100}px, ${redCircleYPosition || 100}px)`
67
+ }}
68
+ >
69
+ <circle fill="red" r={70} />
70
+ </g>
71
+ </g>
72
+ )}
73
+ </Zoom>
74
+ </div>
75
+ );
76
+
77
+ export const WithoutMinimap: Story = {
78
+ args: {
79
+ children: Content
80
+ },
81
+ render: Template
82
+ };
83
+
84
+ export const WithMinimap: Story = {
85
+ args: {
86
+ children: Content,
87
+ showMinimap: true
88
+ },
89
+ render: Template
90
+ };
91
+
92
+ export const WithMinimapPosition: Story = {
93
+ args: {
94
+ children: Content,
95
+ minimapPosition: 'bottom-right',
96
+ showMinimap: true
97
+ },
98
+ render: Template
99
+ };
100
+
101
+ export const Playground: Story = {
102
+ argTypes: {
103
+ redCircleXPosition: {
104
+ control: { max: 2600, min: 400, step: 10, type: 'range' }
105
+ },
106
+ redCircleYPosition: {
107
+ control: { max: 2600, min: 400, step: 10, type: 'range' }
108
+ }
109
+ },
110
+ args: {
111
+ children: Content,
112
+ showMinimap: true
113
+ },
114
+ render: TemplateResponsive
115
+ };
@@ -0,0 +1,68 @@
1
+ import { makeStyles } from 'tss-react/mui';
2
+
3
+ import { alpha } from '@mui/system';
4
+
5
+ import { minimapScale } from './constants';
6
+
7
+ export const useZoomStyles = makeStyles()((theme) => ({
8
+ actions: {
9
+ backgroundColor: alpha(theme.palette.background.paper, 0.8),
10
+ display: 'flex',
11
+ flexDirection: 'row'
12
+ },
13
+ actionsAndZoom: {
14
+ '&[data-position="bottom-left"]': {
15
+ alignItems: 'flex-start',
16
+ bottom: 0,
17
+ flexDirection: 'column-reverse',
18
+ left: 0
19
+ },
20
+ '&[data-position="bottom-right"]': {
21
+ alignItems: 'flex-end',
22
+ bottom: 0,
23
+ flexDirection: 'column-reverse',
24
+ right: 0
25
+ },
26
+ '&[data-position="top-left"]': {
27
+ alignItems: 'flex-start',
28
+ flexDirection: 'column',
29
+ left: 0,
30
+ top: 0
31
+ },
32
+ '&[data-position="top-right"]': {
33
+ alignItems: 'flex-end',
34
+ flexDirection: 'column',
35
+ right: 0,
36
+ top: 0
37
+ },
38
+ display: 'flex',
39
+ position: 'absolute',
40
+ width: 'fit-content'
41
+ },
42
+ minimap: {
43
+ transform: `scale(${minimapScale})`
44
+ },
45
+ minimapBackground: {
46
+ fill: theme.palette.background.paper,
47
+ stroke: theme.palette.background.paper
48
+ },
49
+ minimapContainer: {
50
+ border: `1px solid ${theme.palette.divider}`,
51
+ borderRadius: theme.shape.borderRadius
52
+ },
53
+ minimapZoom: {
54
+ fill: theme.palette.primary.main,
55
+ stroke: theme.palette.primary.main,
56
+ strokeWidth: 10
57
+ },
58
+ movingZone: {
59
+ transition: 'transform 0.1s linear'
60
+ },
61
+ svg: {
62
+ '&[data-is-grabbing="true"]': {
63
+ cursor: 'grabbing'
64
+ },
65
+ cursor: 'grab',
66
+ touchAction: 'none'
67
+ }
68
+ }));
@@ -0,0 +1,64 @@
1
+ import { Zoom as VisxZoom } from '@visx/zoom';
2
+
3
+ import { ParentSize } from '../..';
4
+
5
+ import ZoomContent from './ZoomContent';
6
+ import { MinimapPosition } from './models';
7
+
8
+ export interface ZoomProps {
9
+ children: JSX.Element | (({ width, height }) => JSX.Element);
10
+ id?: number | string;
11
+ minimapPosition?: MinimapPosition;
12
+ scaleMax?: number;
13
+ scaleMin?: number;
14
+ showMinimap?: boolean;
15
+ }
16
+
17
+ const initialTransform = {
18
+ scaleX: 1,
19
+ scaleY: 1,
20
+ skewX: 0,
21
+ skewY: 0,
22
+ translateX: 0,
23
+ translateY: 0
24
+ };
25
+
26
+ const Zoom = ({
27
+ children,
28
+ scaleMin = 0.5,
29
+ scaleMax = 4,
30
+ showMinimap = false,
31
+ minimapPosition = 'top-left',
32
+ id = 0
33
+ }: ZoomProps): JSX.Element => {
34
+ return (
35
+ <ParentSize>
36
+ {({ width, height }) => (
37
+ <VisxZoom<SVGSVGElement>
38
+ height={height}
39
+ initialTransformMatrix={initialTransform}
40
+ scaleXMax={scaleMax}
41
+ scaleXMin={scaleMin}
42
+ scaleYMax={scaleMax}
43
+ scaleYMin={scaleMin}
44
+ width={width}
45
+ >
46
+ {(zoom) => (
47
+ <ZoomContent
48
+ height={height}
49
+ id={id}
50
+ minimapPosition={minimapPosition}
51
+ showMinimap={showMinimap}
52
+ width={width}
53
+ zoom={zoom}
54
+ >
55
+ {children}
56
+ </ZoomContent>
57
+ )}
58
+ </VisxZoom>
59
+ )}
60
+ </ParentSize>
61
+ );
62
+ };
63
+
64
+ export default Zoom;