@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,195 @@
1
+ import { equals } from 'ramda';
2
+
3
+ import {
4
+ ComplexData,
5
+ complexData,
6
+ SimpleData,
7
+ simpleData
8
+ } from './stories/datas';
9
+ import { ComplexContent, SimpleContent } from './stories/contents';
10
+
11
+ import { Node, StandaloneTree, TreeProps } from '.';
12
+
13
+ const validateTree = (tree): void => {
14
+ if (!tree.children) {
15
+ cy.contains(tree.data.name).should('be.visible');
16
+
17
+ return;
18
+ }
19
+
20
+ cy.contains(tree.data.name).should('be.visible');
21
+ tree.children.forEach((child) => {
22
+ validateTree(child);
23
+ });
24
+ };
25
+
26
+ interface InitializeProps
27
+ extends Pick<TreeProps<SimpleData | ComplexData>, 'treeLink' | 'children'> {
28
+ data?: Node<SimpleData | ComplexData>;
29
+ isDefaultExpanded?: (data: SimpleData | ComplexData) => boolean;
30
+ }
31
+
32
+ const initializeStandaloneTree = ({
33
+ data = simpleData,
34
+ isDefaultExpanded = undefined,
35
+ treeLink,
36
+ children = SimpleContent
37
+ }: InitializeProps): void => {
38
+ cy.mount({
39
+ Component: (
40
+ <div style={{ height: '99vh' }}>
41
+ <StandaloneTree
42
+ node={{ height: 70, isDefaultExpanded, width: 70 }}
43
+ tree={data}
44
+ treeLink={treeLink}
45
+ >
46
+ {children}
47
+ </StandaloneTree>
48
+ </div>
49
+ )
50
+ });
51
+ };
52
+
53
+ describe('Simple data tree', () => {
54
+ it('displays the whole tree', () => {
55
+ initializeStandaloneTree({});
56
+
57
+ validateTree(simpleData);
58
+
59
+ cy.makeSnapshot();
60
+ });
61
+
62
+ it("collapses a node's childrens when a node is clicked", () => {
63
+ initializeStandaloneTree({});
64
+
65
+ cy.contains(/^E$/).should('be.visible');
66
+ cy.contains(/^E1$/).should('be.visible');
67
+
68
+ cy.contains(/^C$/).click();
69
+
70
+ cy.contains(/^E$/).should('not.exist');
71
+ cy.contains(/^E1$/).should('not.exist');
72
+
73
+ cy.makeSnapshot();
74
+ });
75
+
76
+ it("expands a node's childrens when a node is clicked", () => {
77
+ initializeStandaloneTree({});
78
+
79
+ cy.contains(/^A$/).click();
80
+
81
+ cy.contains(/^A1$/).should('not.exist');
82
+ cy.contains(/^A2$/).should('not.exist');
83
+ cy.contains(/^A3$/).should('not.exist');
84
+ cy.contains(/^C$/).should('not.exist');
85
+
86
+ cy.contains(/^A$/).click();
87
+
88
+ cy.contains(/^A1$/).should('be.visible');
89
+ cy.contains(/^A2$/).should('be.visible');
90
+ cy.contains(/^A3$/).should('be.visible');
91
+ cy.contains(/^C$/).should('be.visible');
92
+
93
+ cy.makeSnapshot();
94
+ });
95
+
96
+ it('cannot collapses a node when a leaf is clicked', () => {
97
+ initializeStandaloneTree({});
98
+
99
+ cy.contains(/^Z$/).click();
100
+
101
+ cy.contains(/^Z$/).should('be.visible');
102
+
103
+ cy.makeSnapshot();
104
+ });
105
+
106
+ it('expands nodes by default when a prop is set', () => {
107
+ initializeStandaloneTree({
108
+ isDefaultExpanded: (data: SimpleData) => equals('critical', data.status)
109
+ });
110
+
111
+ cy.contains(/^T$/).should('be.visible');
112
+ cy.contains(/^A$/).should('be.visible');
113
+ cy.contains(/^A3$/).should('be.visible');
114
+ cy.contains(/^C$/).should('be.visible');
115
+ cy.contains(/^E$/).should('be.visible');
116
+ cy.contains(/^E1$/).should('be.visible');
117
+
118
+ cy.contains(/^B1$/).should('not.exist');
119
+ cy.contains(/^D1$/).should('not.exist');
120
+
121
+ cy.makeSnapshot();
122
+ });
123
+
124
+ it('displays customized links when a prop is set', () => {
125
+ initializeStandaloneTree({
126
+ treeLink: {
127
+ getStroke: ({ target }) => (target.status === 'ok' ? 'grey' : 'black'),
128
+ getStrokeDasharray: ({ target }) =>
129
+ target.status === 'ok' ? '5,5' : '0',
130
+ getStrokeOpacity: ({ target }) => (target.status === 'ok' ? 0.8 : 1),
131
+ getStrokeWidth: ({ target }) => (target.status === 'ok' ? 1 : 2)
132
+ }
133
+ });
134
+
135
+ cy.contains(/^Z$/).should('be.visible');
136
+
137
+ cy.makeSnapshot();
138
+ });
139
+ });
140
+
141
+ describe('Complex data tree', () => {
142
+ it('cannot collapse a node when a node is not clickable', () => {
143
+ initializeStandaloneTree({
144
+ children: ComplexContent,
145
+ data: complexData
146
+ });
147
+
148
+ cy.contains('BA 3').should('be.visible');
149
+
150
+ cy.contains('BA 2').click();
151
+
152
+ cy.contains('BA 3').should('be.visible');
153
+
154
+ cy.makeSnapshot();
155
+ });
156
+
157
+ it('collapses a node when a node is clickable', () => {
158
+ initializeStandaloneTree({
159
+ children: ComplexContent,
160
+ data: complexData
161
+ });
162
+
163
+ cy.contains('BA 3').should('be.visible');
164
+
165
+ cy.contains('2').click();
166
+
167
+ cy.contains('BA 3').should('not.exist');
168
+
169
+ cy.makeSnapshot();
170
+ });
171
+
172
+ it('displays the tree with step links when a prop is set', () => {
173
+ initializeStandaloneTree({
174
+ treeLink: {
175
+ type: 'step'
176
+ }
177
+ });
178
+
179
+ cy.contains('T').should('be.visible');
180
+
181
+ cy.makeSnapshot();
182
+ });
183
+
184
+ it('displays the tree with line links when a prop is set', () => {
185
+ initializeStandaloneTree({
186
+ treeLink: {
187
+ type: 'line'
188
+ }
189
+ });
190
+
191
+ cy.contains('T').should('be.visible');
192
+
193
+ cy.makeSnapshot();
194
+ });
195
+ });
@@ -0,0 +1,160 @@
1
+ import { useState } from 'react';
2
+
3
+ import { Meta, StoryObj } from '@storybook/react';
4
+ import { equals, has } from 'ramda';
5
+
6
+ import { Zoom } from '../../components';
7
+
8
+ import {
9
+ ComplexData,
10
+ complexData,
11
+ moreComplexData,
12
+ SimpleData,
13
+ simpleData
14
+ } from './stories/datas';
15
+ import { ComplexContent, SimpleContent } from './stories/contents';
16
+
17
+ import { StandaloneTree, Tree, TreeProps } from '.';
18
+
19
+ const meta: Meta<typeof StandaloneTree> = {
20
+ component: StandaloneTree
21
+ };
22
+
23
+ export default meta;
24
+ type Story = StoryObj<typeof StandaloneTree>;
25
+
26
+ const StandaloneTreeTemplate = (args): JSX.Element => (
27
+ <div style={{ height: '90vh', width: '100%' }}>
28
+ <StandaloneTree<SimpleData> {...args} />
29
+ </div>
30
+ );
31
+
32
+ const TreeWithZoom = <TData,>({
33
+ tree,
34
+ ...args
35
+ }: TreeProps<TData>): JSX.Element => {
36
+ const [currentTree, setTree] = useState(tree);
37
+
38
+ return (
39
+ <div style={{ height: '90vh', width: '100%' }}>
40
+ <Zoom
41
+ showMinimap
42
+ labels={{
43
+ clear: 'Clear'
44
+ }}
45
+ >
46
+ {({ width, height }) => (
47
+ <Tree<ComplexData>
48
+ {...args}
49
+ changeTree={setTree}
50
+ containerHeight={height}
51
+ containerWidth={width}
52
+ tree={currentTree}
53
+ />
54
+ )}
55
+ </Zoom>
56
+ </div>
57
+ );
58
+ };
59
+
60
+ export const DefaultStandaloneTree: Story = {
61
+ args: {
62
+ children: SimpleContent,
63
+ node: {
64
+ height: 90,
65
+ width: 90
66
+ },
67
+ tree: simpleData
68
+ },
69
+ render: StandaloneTreeTemplate
70
+ };
71
+
72
+ export const WithDefaultExpandFilter: Story = {
73
+ args: {
74
+ children: SimpleContent,
75
+ node: {
76
+ height: 90,
77
+ isDefaultExpanded: (data: SimpleData) => equals('critical', data.status),
78
+ width: 90
79
+ },
80
+ tree: simpleData
81
+ },
82
+ render: StandaloneTreeTemplate
83
+ };
84
+
85
+ export const WithStepLink: Story = {
86
+ args: {
87
+ children: SimpleContent,
88
+ node: {
89
+ height: 90,
90
+ width: 90
91
+ },
92
+ tree: simpleData,
93
+ treeLink: {
94
+ type: 'step'
95
+ }
96
+ },
97
+ render: StandaloneTreeTemplate
98
+ };
99
+
100
+ export const WithCustomLinks: Story = {
101
+ args: {
102
+ children: SimpleContent,
103
+ node: {
104
+ height: 90,
105
+ width: 90
106
+ },
107
+ tree: simpleData,
108
+ treeLink: {
109
+ getStroke: ({ target }) => (target.status === 'ok' ? 'grey' : 'black'),
110
+ getStrokeDasharray: ({ target }) =>
111
+ target.status === 'ok' ? '5,5' : '0',
112
+ getStrokeOpacity: ({ target }) => (target.status === 'ok' ? 0.8 : 1),
113
+ getStrokeWidth: ({ target }) => (target.status === 'ok' ? 1 : 2),
114
+ type: 'line'
115
+ }
116
+ },
117
+ render: StandaloneTreeTemplate
118
+ };
119
+
120
+ export const WithComplexData: Story = {
121
+ args: {
122
+ children: ComplexContent,
123
+ node: {
124
+ height: 90,
125
+ isDefaultExpanded: (data: SimpleData) =>
126
+ equals('critical', data.status) || !has('count', data),
127
+ width: 90
128
+ },
129
+ tree: complexData,
130
+ treeLink: {
131
+ getStroke: ({ target }) => (target.status === 'ok' ? 'grey' : 'black'),
132
+ getStrokeDasharray: ({ target }) =>
133
+ target.status === 'ok' ? '5,5' : '0',
134
+ getStrokeOpacity: ({ target }) => (target.status === 'ok' ? 0.8 : 1),
135
+ getStrokeWidth: ({ target }) => (target.status === 'ok' ? 1 : 2)
136
+ }
137
+ },
138
+ render: StandaloneTreeTemplate
139
+ };
140
+
141
+ export const treeWithZoom: Story = {
142
+ args: {
143
+ children: ComplexContent,
144
+ node: {
145
+ height: 90,
146
+ isDefaultExpanded: (data: SimpleData) =>
147
+ equals('critical', data.status) || !has('count', data),
148
+ width: 90
149
+ },
150
+ tree: moreComplexData,
151
+ treeLink: {
152
+ getStroke: ({ target }) => (target.status === 'ok' ? 'grey' : 'black'),
153
+ getStrokeDasharray: ({ target }) =>
154
+ target.status === 'ok' ? '5,5' : '0',
155
+ getStrokeOpacity: ({ target }) => (target.status === 'ok' ? 0.8 : 1),
156
+ getStrokeWidth: ({ target }) => (target.status === 'ok' ? 1 : 2)
157
+ }
158
+ },
159
+ render: TreeWithZoom
160
+ };
@@ -0,0 +1,116 @@
1
+ import { useCallback, useMemo } from 'react';
2
+
3
+ import { Group } from '@visx/group';
4
+ import { hierarchy, Tree as VisxTree } from '@visx/hierarchy';
5
+ import { isNil } from 'ramda';
6
+
7
+ import { useDeepCompare } from '../../utils';
8
+
9
+ import { nodeMargins } from './constants';
10
+ import { BaseProp, Node, TreeProps } from './models';
11
+ import { updateNodeFromTree } from './utils';
12
+ import Links from './Links';
13
+ import DescendantNodes from './DescendantNodes';
14
+
15
+ export const Tree = <TData extends BaseProp>({
16
+ containerHeight,
17
+ containerWidth,
18
+ tree,
19
+ node,
20
+ treeLink = {},
21
+ changeTree,
22
+ children
23
+ }: TreeProps<TData>): JSX.Element => {
24
+ const formattedTree: Node<TData> = useMemo(
25
+ () => ({
26
+ ...tree,
27
+ isExpanded: true
28
+ }),
29
+ useDeepCompare([tree])
30
+ );
31
+
32
+ const toggleTreeNodesExpanded = useCallback(
33
+ ({ currentTree, targetNode }): Node<TData> => {
34
+ return updateNodeFromTree({
35
+ callback: (subTree) => {
36
+ if (isNil(subTree.isExpanded) && isNil(node.isDefaultExpanded)) {
37
+ return {
38
+ isExpanded: false
39
+ };
40
+ }
41
+
42
+ return {
43
+ isExpanded: isNil(subTree.isExpanded)
44
+ ? !node.isDefaultExpanded?.(subTree.data)
45
+ : !subTree.isExpanded || false
46
+ };
47
+ },
48
+ targetNode,
49
+ tree: currentTree
50
+ });
51
+ },
52
+ [node.isDefaultExpanded]
53
+ );
54
+
55
+ const expandCollapseNode = useCallback(
56
+ (targetNode: Node<TData>): void => {
57
+ changeTree?.(
58
+ toggleTreeNodesExpanded({ currentTree: formattedTree, targetNode })
59
+ );
60
+ },
61
+ [formattedTree]
62
+ );
63
+
64
+ const getExpanded = useCallback(
65
+ (d: Node<TData>): Array<Node<TData>> | undefined | null => {
66
+ if (isNil(d.isExpanded) && isNil(node.isDefaultExpanded)) {
67
+ return d.children;
68
+ }
69
+ if (isNil(d.isExpanded)) {
70
+ return node.isDefaultExpanded?.(d.data) ? d.children : null;
71
+ }
72
+
73
+ return d.isExpanded ? d.children : null;
74
+ },
75
+ [node.isDefaultExpanded]
76
+ );
77
+
78
+ const origin = useMemo(
79
+ () => ({
80
+ x: 0,
81
+ y: containerHeight / 2
82
+ }),
83
+ [containerHeight]
84
+ );
85
+
86
+ return (
87
+ <Group left={node.width}>
88
+ <VisxTree
89
+ left={0}
90
+ nodeSize={[node.width + nodeMargins.y, node.height + nodeMargins.x]}
91
+ root={hierarchy(formattedTree, getExpanded)}
92
+ separation={() => 1}
93
+ size={[containerWidth, containerHeight]}
94
+ top={0}
95
+ >
96
+ {(subTree) => (
97
+ <Group left={origin.x} top={origin.y}>
98
+ <Links links={subTree.links()} treeLink={treeLink} />
99
+
100
+ <DescendantNodes
101
+ descendants={subTree.descendants()}
102
+ expandCollapseNode={expandCollapseNode}
103
+ getExpanded={getExpanded}
104
+ nodeSize={{
105
+ height: node.height,
106
+ width: node.width
107
+ }}
108
+ >
109
+ {children}
110
+ </DescendantNodes>
111
+ </Group>
112
+ )}
113
+ </VisxTree>
114
+ </Group>
115
+ );
116
+ };
@@ -0,0 +1,2 @@
1
+ export const margins = { bottom: 30, left: 30, right: 30, top: 30 };
2
+ export const nodeMargins = { x: 90, y: 16 };
@@ -0,0 +1,4 @@
1
+ export { Tree } from './Tree';
2
+ export { StandaloneTree } from './StandaloneTree';
3
+ export * from './utils';
4
+ export type { Node, TreeProps, ChildrenProps } from './models';
@@ -0,0 +1,55 @@
1
+ export interface Node<T> {
2
+ children?: Array<Node<T>>;
3
+ data: T;
4
+ isExpanded?: boolean;
5
+ }
6
+
7
+ export interface LinkProps<T> {
8
+ source: T;
9
+ target: T;
10
+ }
11
+
12
+ export interface BaseProp {
13
+ id: number;
14
+ name: string;
15
+ }
16
+
17
+ export type Link = 'curve' | 'line' | 'step';
18
+
19
+ export interface ChildrenProps<TData> {
20
+ ancestors: Array<Node<TData>>;
21
+ depth: number;
22
+ expandCollapseNode: (targetNode: Node<TData>) => void;
23
+ isExpanded: boolean;
24
+ node: Node<TData>;
25
+ nodeSize: {
26
+ height: number;
27
+ width: number;
28
+ };
29
+ onMouseDown: (e) => void;
30
+ onMouseUp: (callback) => (e) => void;
31
+ }
32
+
33
+ export interface TreeProps<TData> {
34
+ changeTree?: (newTree: Node<TData>) => void;
35
+ children: (props: ChildrenProps<TData>) => JSX.Element;
36
+ containerHeight: number;
37
+ containerWidth: number;
38
+ contentHeight?: number;
39
+ contentWidth?: number;
40
+ node: {
41
+ height: number;
42
+ isDefaultExpanded?: (data: TData) => boolean;
43
+ width: number;
44
+ };
45
+ tree: Node<TData>;
46
+ treeLink?: {
47
+ getStroke?: (props: LinkProps<TData>) => string | undefined;
48
+ getStrokeDasharray?: (
49
+ props: LinkProps<TData>
50
+ ) => string | number | undefined;
51
+ getStrokeOpacity?: (props: LinkProps<TData>) => string | number | undefined;
52
+ getStrokeWidth?: (props: LinkProps<TData>) => string | number | undefined;
53
+ type?: Link;
54
+ };
55
+ }
@@ -0,0 +1,164 @@
1
+ import { always, cond, equals, has, T } from 'ramda';
2
+
3
+ import { Avatar, Paper, Typography, useTheme } from '@mui/material';
4
+ import SpaIcon from '@mui/icons-material/Spa';
5
+
6
+ import { ChildrenProps } from '..';
7
+
8
+ import { ComplexData, SimpleData } from './datas';
9
+
10
+ export const SimpleContent = ({
11
+ node,
12
+ depth,
13
+ isExpanded,
14
+ nodeSize,
15
+ expandCollapseNode
16
+ }: ChildrenProps<SimpleData>): JSX.Element => {
17
+ const theme = useTheme();
18
+ const fillColor = cond([
19
+ [equals('critical'), always(theme.palette.error.main)],
20
+ [equals('warning'), always(theme.palette.warning.main)],
21
+ [T, always(theme.palette.success.main)]
22
+ ])(node.data.status);
23
+
24
+ if (equals(depth, 0)) {
25
+ return (
26
+ <div
27
+ style={{
28
+ alignItems: 'center',
29
+ display: 'flex',
30
+ height: '100%',
31
+ justifyContent: 'center',
32
+ width: '100%'
33
+ }}
34
+ >
35
+ <Avatar
36
+ sx={{
37
+ backgroundColor: fillColor,
38
+ color: theme.palette.text.primary,
39
+ cursor: 'pointer'
40
+ }}
41
+ >
42
+ {node.data.name}
43
+ </Avatar>
44
+ </div>
45
+ );
46
+ }
47
+
48
+ return (
49
+ <Paper
50
+ sx={{
51
+ alignItems: 'center',
52
+ backgroundColor: fillColor,
53
+ cursor: node.children ? 'pointer' : 'default',
54
+ display: 'flex',
55
+ flexDirection: 'column',
56
+ height: nodeSize.height,
57
+ justifyContent: 'center',
58
+ p: 1,
59
+ position: 'relative',
60
+ width: nodeSize.width
61
+ }}
62
+ onClick={() => {
63
+ expandCollapseNode(node);
64
+ }}
65
+ >
66
+ {!node.children && (
67
+ <SpaIcon
68
+ fontSize="small"
69
+ sx={{ position: 'absolute', right: 8, top: 8 }}
70
+ />
71
+ )}
72
+ <Typography>{node.data.name}</Typography>
73
+ {node.children && (
74
+ <Typography>{isExpanded ? 'Expanded' : 'Collapsed'}</Typography>
75
+ )}
76
+ </Paper>
77
+ );
78
+ };
79
+
80
+ export const ComplexContent = ({
81
+ node,
82
+ depth,
83
+ nodeSize,
84
+ expandCollapseNode,
85
+ onMouseDown,
86
+ onMouseUp
87
+ }: ChildrenProps<ComplexData>): JSX.Element => {
88
+ const theme = useTheme();
89
+ const fillColor = cond([
90
+ [equals('critical'), always(theme.palette.error.main)],
91
+ [equals('warning'), always(theme.palette.warning.main)],
92
+ [T, always(theme.palette.success.main)]
93
+ ])(node.data.status);
94
+
95
+ if (equals(depth, 0)) {
96
+ return (
97
+ <Paper
98
+ sx={{
99
+ alignItems: 'center',
100
+ backgroundColor: fillColor,
101
+ display: 'flex',
102
+ flexDirection: 'column',
103
+ height: nodeSize.height,
104
+ justifyContent: 'center',
105
+ p: 1,
106
+ position: 'relative',
107
+ width: nodeSize.width
108
+ }}
109
+ >
110
+ <Typography>{node.data.name}</Typography>
111
+ </Paper>
112
+ );
113
+ }
114
+
115
+ if (has('count', node.data)) {
116
+ return (
117
+ <div
118
+ style={{
119
+ alignItems: 'center',
120
+ display: 'flex',
121
+ height: '100%',
122
+ justifyContent: 'center',
123
+ width: '100%'
124
+ }}
125
+ >
126
+ <Avatar
127
+ sx={{
128
+ backgroundColor: fillColor,
129
+ color: theme.palette.text.primary,
130
+ cursor: 'pointer'
131
+ }}
132
+ onMouseDown={onMouseDown}
133
+ onMouseUp={onMouseUp(() => expandCollapseNode(node))}
134
+ >
135
+ {node.data.count}
136
+ </Avatar>
137
+ </div>
138
+ );
139
+ }
140
+
141
+ return (
142
+ <Paper
143
+ sx={{
144
+ alignItems: 'center',
145
+ backgroundColor: fillColor,
146
+ display: 'flex',
147
+ flexDirection: 'column',
148
+ height: nodeSize.height,
149
+ justifyContent: 'center',
150
+ p: 1,
151
+ position: 'relative',
152
+ width: nodeSize.width
153
+ }}
154
+ >
155
+ {!node.children && (
156
+ <SpaIcon
157
+ fontSize="small"
158
+ sx={{ position: 'absolute', right: 8, top: 8 }}
159
+ />
160
+ )}
161
+ <Typography>{node.data.name}</Typography>
162
+ </Paper>
163
+ );
164
+ };