@centreon/ui 24.4.1-sync-release-34022.1 → 24.4.1-test-code-coverage.1

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,139 @@
1
+ import numeral from 'numeral';
2
+
3
+ import BarStack from './BarStack';
4
+ import { BarType, BarStackProps } from './models';
5
+
6
+ const defaultData = [
7
+ { color: '#88B922', label: 'Ok', value: 148 },
8
+ { color: '#999999', label: 'Unknown', value: 13 },
9
+ { color: '#F7931A', label: 'Warning', value: 16 },
10
+ { color: '#FF6666', label: 'Down', value: 62 }
11
+ ];
12
+
13
+ const total = Math.floor(
14
+ defaultData.reduce((acc, { value }) => acc + value, 0)
15
+ );
16
+
17
+ const TooltipContent = ({ label, color, value }: BarType): JSX.Element => {
18
+ return (
19
+ <div data-testid={`tooltip-${label}`} style={{ color }}>
20
+ {label} : {value}
21
+ </div>
22
+ );
23
+ };
24
+
25
+ const initialize = ({
26
+ width = '400px',
27
+ height = '400px',
28
+ data = defaultData,
29
+ ...args
30
+ }: Omit<BarStackProps, 'data'> & {
31
+ data?;
32
+ height?: string;
33
+ width?: string;
34
+ }): void => {
35
+ cy.mount({
36
+ Component: (
37
+ <div style={{ height, width }}>
38
+ <BarStack {...args} data={data} />
39
+ </div>
40
+ )
41
+ });
42
+ };
43
+
44
+ describe('Bar stack', () => {
45
+ it('renders Bar stack correctly with provided data', () => {
46
+ initialize({});
47
+
48
+ defaultData.forEach(({ label }) => {
49
+ cy.findByTestId(label).should('be.visible');
50
+ });
51
+
52
+ cy.makeSnapshot();
53
+ });
54
+
55
+ it('adjusts size based on the provided width and height', () => {
56
+ initialize({ displayLegend: false, height: '300px', width: '300px' });
57
+
58
+ cy.findByTestId('barStack').should('have.css', 'height', '300px');
59
+
60
+ cy.makeSnapshot();
61
+ });
62
+
63
+ it('renders as a horizontal bar when variant is set to "horizontal"', () => {
64
+ initialize({ variant: 'horizontal' });
65
+ cy.get('[data-variant="horizontal"]').should('exist');
66
+
67
+ cy.makeSnapshot();
68
+ });
69
+
70
+ it('renders as a vertical bar when variant is set to "vertical"', () => {
71
+ initialize({ variant: 'vertical' });
72
+ cy.get('[data-variant="vertical"]').should('exist');
73
+
74
+ cy.makeSnapshot();
75
+ });
76
+
77
+ it('displays tooltip with correct information on hover', () => {
78
+ initialize({ TooltipContent });
79
+
80
+ defaultData.forEach(({ label, value }) => {
81
+ cy.findByTestId(label).trigger('mouseover', { force: true });
82
+
83
+ cy.findByTestId(`tooltip-${label}`)
84
+ .should('contain', label)
85
+ .and('contain', numeral(value).format('0a').toUpperCase());
86
+ });
87
+
88
+ cy.makeSnapshot();
89
+ });
90
+
91
+ it('conditionally displays values on rects based on displayValues prop', () => {
92
+ initialize({ displayValues: true });
93
+ defaultData.forEach(({ value }, index) => {
94
+ cy.findAllByTestId('value')
95
+ .eq(index)
96
+ .children()
97
+ .eq(0)
98
+ .should('have.text', value);
99
+ });
100
+
101
+ initialize({ displayValues: false });
102
+ cy.findAllByTestId('value').should('not.exist');
103
+
104
+ cy.makeSnapshot();
105
+ });
106
+
107
+ it('displays values on rects in percentage unit when displayValues is set to true and unit to percentage', () => {
108
+ initialize({ displayValues: true, unit: 'percentage' });
109
+ defaultData.forEach(({ value }, index) => {
110
+ cy.findAllByTestId('value')
111
+ .eq(index)
112
+ .children()
113
+ .eq(0)
114
+ .should('have.text', `${((value * 100) / total).toFixed(1)}%`);
115
+ });
116
+
117
+ cy.makeSnapshot();
118
+ });
119
+
120
+ it('displays Legend component based on displayLegend prop', () => {
121
+ initialize({ displayLegend: true });
122
+ cy.findByTestId('Legend').should('be.visible');
123
+
124
+ initialize({ displayLegend: false });
125
+ cy.findByTestId('Legend').should('not.exist');
126
+
127
+ cy.makeSnapshot();
128
+ });
129
+
130
+ it('displays the title when the title is giving', () => {
131
+ initialize({ title: 'host' });
132
+ cy.findByTestId('Title').should('be.visible');
133
+
134
+ initialize({});
135
+ cy.findByTestId('Title').should('not.exist');
136
+
137
+ cy.makeSnapshot();
138
+ });
139
+ });
@@ -0,0 +1,123 @@
1
+ import { Meta, StoryObj } from '@storybook/react';
2
+
3
+ import { BarType } from './models';
4
+
5
+ import { BarStack } from '.';
6
+
7
+ const data = [
8
+ { color: '#88B922', label: 'Ok', value: 148 },
9
+ { color: '#999999', label: 'Unknown', value: 63 },
10
+ { color: '#F7931A', label: 'Warning', value: 16 },
11
+ { color: '#FF6666', label: 'Down', value: 13 }
12
+ ];
13
+
14
+ const dataWithBigNumbers = [
15
+ { color: '#88B922', label: 'Ok', value: 260000 },
16
+ { color: '#999999', label: 'Unknown', value: 1010900 },
17
+ { color: '#F7931A', label: 'Warning', value: 63114 },
18
+ { color: '#FF6666', label: 'Down', value: 122222 }
19
+ ];
20
+
21
+ const dataWithSmallNumber = [
22
+ { color: '#88B922', label: 'Ok', value: 148 },
23
+ { color: '#999999', label: 'Unknown', value: 42 },
24
+ { color: '#F7931A', label: 'Warning', value: 7 },
25
+ { color: '#FF6666', label: 'Down', value: 5 }
26
+ ];
27
+
28
+ const meta: Meta<typeof BarStack> = {
29
+ component: BarStack
30
+ };
31
+
32
+ export default meta;
33
+ type Story = StoryObj<typeof BarStack>;
34
+
35
+ const TooltipContent = ({ label, color, value }: BarType): JSX.Element => {
36
+ return (
37
+ <div style={{ color }}>
38
+ {label} : {value}
39
+ </div>
40
+ );
41
+ };
42
+
43
+ const Template = (args): JSX.Element => {
44
+ return (
45
+ <div style={{ height: '300px', width: '500px' }}>
46
+ <BarStack {...args} />
47
+ </div>
48
+ );
49
+ };
50
+
51
+ export const Vertical: Story = {
52
+ args: { data, title: 'hosts' },
53
+ render: Template
54
+ };
55
+
56
+ export const WithoutTitle: Story = {
57
+ args: { data },
58
+ render: Template
59
+ };
60
+
61
+ export const WithoutLegend: Story = {
62
+ args: { data, displayLegend: false, title: 'hosts' },
63
+
64
+ render: Template
65
+ };
66
+
67
+ export const withDisplayedValues: Story = {
68
+ args: { data, displayValues: true, title: 'hosts' },
69
+ render: Template
70
+ };
71
+
72
+ export const WithPencentage: Story = {
73
+ args: { data, displayValues: true, title: 'hosts', unit: 'percentage' },
74
+ render: Template
75
+ };
76
+
77
+ export const WithTooltip: Story = {
78
+ args: { TooltipContent, data, title: 'hosts' },
79
+ render: Template
80
+ };
81
+
82
+ export const WithBigNumbers: Story = {
83
+ args: {
84
+ TooltipContent,
85
+ data: dataWithBigNumbers,
86
+ displayValues: true,
87
+ title: 'hosts'
88
+ },
89
+ render: Template
90
+ };
91
+
92
+ export const WithSmallNumbers: Story = {
93
+ args: {
94
+ TooltipContent,
95
+ data: dataWithSmallNumber,
96
+ displayValues: true,
97
+ title: 'hosts'
98
+ },
99
+ render: Template
100
+ };
101
+
102
+ export const Horizontal: Story = {
103
+ args: {
104
+ TooltipContent,
105
+ data,
106
+ displayValues: true,
107
+ title: 'hosts',
108
+ variant: 'horizontal'
109
+ },
110
+ render: Template
111
+ };
112
+
113
+ export const HorizontalWithoutLegend: Story = {
114
+ args: {
115
+ TooltipContent,
116
+ data,
117
+ displayLegend: false,
118
+ displayValues: true,
119
+ title: 'hosts',
120
+ variant: 'horizontal'
121
+ },
122
+ render: Template
123
+ };
@@ -0,0 +1,37 @@
1
+ import { makeStyles } from 'tss-react/mui';
2
+
3
+ export const useBarStackStyles = makeStyles()((theme) => ({
4
+ barStackTooltip: {
5
+ backgroundColor: theme.palette.background.paper,
6
+ color: theme.palette.text.primary,
7
+ padding: 0,
8
+ position: 'relative'
9
+ },
10
+ container: {
11
+ alignItems: 'center',
12
+ display: 'flex',
13
+ gap: theme.spacing(1.5),
14
+ justifyContent: 'center'
15
+ },
16
+ svgContainer: {
17
+ alignItems: 'center',
18
+ backgroundColor: theme.palette.background.panelGroups,
19
+ borderRadius: theme.spacing(1.25),
20
+ display: 'flex',
21
+ justifyContent: 'center'
22
+ },
23
+ svgWrapper: {
24
+ alignItems: 'center',
25
+ display: 'flex',
26
+ flexDirection: 'column',
27
+ gap: theme.spacing(1),
28
+ justifyContent: 'center'
29
+ },
30
+ title: {
31
+ fontSize: theme.typography.h6.fontSize,
32
+ fontWeight: theme.typography.fontWeightMedium,
33
+ margin: 0,
34
+ padding: 0,
35
+ textAlign: 'center'
36
+ }
37
+ }));
@@ -0,0 +1,14 @@
1
+ import { ParentSize } from '../..';
2
+
3
+ import ResponsiveBarStack from './ResponsiveBarStack';
4
+ import { BarStackProps } from './models';
5
+
6
+ const Bar = (props: BarStackProps): JSX.Element => (
7
+ <ParentSize>
8
+ {({ width, height }) => (
9
+ <ResponsiveBarStack {...props} height={height} width={width} />
10
+ )}
11
+ </ParentSize>
12
+ );
13
+
14
+ export default Bar;
@@ -0,0 +1,209 @@
1
+ import { useRef } from 'react';
2
+
3
+ import { BarStack as BarStackVertical, BarStackHorizontal } from '@visx/shape';
4
+ import { Group } from '@visx/group';
5
+ import numeral from 'numeral';
6
+ import { Text } from '@visx/text';
7
+ import { useTranslation } from 'react-i18next';
8
+ import { equals } from 'ramda';
9
+
10
+ import { Tooltip } from '../../components';
11
+ import { LegendProps } from '../Legend/models';
12
+ import { Legend as LegendComponent } from '../Legend';
13
+ import { getValueByUnit } from '../common/utils';
14
+
15
+ import { BarStackProps } from './models';
16
+ import { useBarStackStyles } from './BarStack.styles';
17
+ import useResponsiveBarStack from './useResponsiveBarStack';
18
+
19
+ const DefaultLengd = ({ scale, direction }: LegendProps): JSX.Element => (
20
+ <LegendComponent direction={direction} scale={scale} />
21
+ );
22
+
23
+ const BarStack = ({
24
+ title,
25
+ data,
26
+ width,
27
+ height,
28
+ size = 72,
29
+ onSingleBarClick,
30
+ displayLegend = true,
31
+ TooltipContent,
32
+ Legend = DefaultLengd,
33
+ unit = 'number',
34
+ displayValues,
35
+ variant = 'vertical',
36
+ legendDirection = 'column'
37
+ }: BarStackProps & { height: number; width: number }): JSX.Element => {
38
+ const { t } = useTranslation();
39
+ const { classes } = useBarStackStyles();
40
+
41
+ const titleRef = useRef(null);
42
+ const legendRef = useRef(null);
43
+
44
+ const {
45
+ barSize,
46
+ colorScale,
47
+ input,
48
+ keys,
49
+ legendScale,
50
+ total,
51
+ xScale,
52
+ yScale,
53
+ svgWrapperWidth,
54
+ svgContainerSize,
55
+ isVerticalBar
56
+ } = useResponsiveBarStack({
57
+ data,
58
+ height,
59
+ legendRef,
60
+ size,
61
+ titleRef,
62
+ unit,
63
+ variant,
64
+ width
65
+ });
66
+
67
+ const BarStackComponent = isVerticalBar
68
+ ? BarStackVertical
69
+ : BarStackHorizontal;
70
+
71
+ return (
72
+ <div className={classes.container} style={{ height, width }}>
73
+ <div
74
+ className={classes.svgWrapper}
75
+ style={{
76
+ height,
77
+ width: svgWrapperWidth
78
+ }}
79
+ >
80
+ {title && (
81
+ <div className={classes.title} data-testid="Title" ref={titleRef}>
82
+ {`${numeral(total).format('0a').toUpperCase()} `} {t(title)}
83
+ </div>
84
+ )}
85
+ <div
86
+ className={classes.svgContainer}
87
+ data-testid="barStack"
88
+ style={svgContainerSize}
89
+ >
90
+ <svg
91
+ data-variant={variant}
92
+ height={barSize.height}
93
+ width={barSize.width}
94
+ >
95
+ <Group>
96
+ <BarStackComponent
97
+ color={colorScale}
98
+ data={[input]}
99
+ keys={keys}
100
+ {...(isVerticalBar
101
+ ? { x: () => undefined }
102
+ : { y: () => undefined })}
103
+ xScale={xScale}
104
+ yScale={yScale}
105
+ >
106
+ {(barStacks) =>
107
+ barStacks.map((barStack) =>
108
+ barStack.bars.map((bar) => {
109
+ const shouldDisplayValues = (): boolean => {
110
+ if (bar.height < 10) {
111
+ return false;
112
+ }
113
+
114
+ return (
115
+ (equals(unit, 'number') && bar.width > 10) ||
116
+ (equals(unit, 'percentage') && bar.width > 25)
117
+ );
118
+ };
119
+
120
+ const TextX = bar.x + bar.width / 2;
121
+ const TextY = bar.y + bar.height / 2;
122
+
123
+ const onClick = (): void => {
124
+ onSingleBarClick?.(bar);
125
+ };
126
+
127
+ return (
128
+ <Tooltip
129
+ hasCaret
130
+ classes={{
131
+ tooltip: classes.barStackTooltip
132
+ }}
133
+ followCursor={false}
134
+ key={`bar-stack-${barStack.index}-${bar.index}`}
135
+ label={
136
+ TooltipContent && (
137
+ <TooltipContent
138
+ color={bar.color}
139
+ label={bar.key}
140
+ title={title}
141
+ total={total}
142
+ value={barStack.bars[0].bar.data[barStack.key]}
143
+ />
144
+ )
145
+ }
146
+ position={
147
+ isVerticalBar ? 'right-start' : 'bottom-start'
148
+ }
149
+ >
150
+ <g data-testid={bar.key} onClick={onClick}>
151
+ <rect
152
+ cursor="pointer"
153
+ fill={bar.color}
154
+ height={
155
+ isVerticalBar ? bar.height - 1 : bar.height
156
+ }
157
+ key={`bar-stack-${barStack.index}-${bar.index}`}
158
+ ry={10}
159
+ width={isVerticalBar ? bar.width : bar.width - 1}
160
+ x={bar.x}
161
+ y={bar.y}
162
+ />
163
+ {displayValues && shouldDisplayValues() && (
164
+ <Text
165
+ cursor="pointer"
166
+ data-testid="value"
167
+ fill="#000"
168
+ fontSize={12}
169
+ fontWeight={600}
170
+ textAnchor="middle"
171
+ verticalAnchor="middle"
172
+ x={TextX}
173
+ y={TextY}
174
+ >
175
+ {getValueByUnit({
176
+ total,
177
+ unit,
178
+ value: barStack.bars[0].bar.data[barStack.key]
179
+ })}
180
+ </Text>
181
+ )}
182
+ </g>
183
+ </Tooltip>
184
+ );
185
+ })
186
+ )
187
+ }
188
+ </BarStackComponent>
189
+ </Group>
190
+ </svg>
191
+ </div>
192
+ </div>
193
+ {displayLegend && (
194
+ <div data-testid="Legend" ref={legendRef}>
195
+ <Legend
196
+ data={data}
197
+ direction={legendDirection}
198
+ scale={legendScale}
199
+ title={title}
200
+ total={total}
201
+ unit={unit}
202
+ />
203
+ </div>
204
+ )}
205
+ </div>
206
+ );
207
+ };
208
+
209
+ export default BarStack;
@@ -0,0 +1 @@
1
+ export { default as BarStack } from './BarStack';
@@ -0,0 +1,19 @@
1
+ export interface BarType {
2
+ color: string;
3
+ label: string;
4
+ value: number;
5
+ }
6
+
7
+ export type BarStackProps = {
8
+ Legend?: ({ scale, data, title, total, unit, direction }) => JSX.Element;
9
+ TooltipContent?: (barData) => JSX.Element | boolean | null;
10
+ data: Array<BarType>;
11
+ displayLegend?: boolean;
12
+ displayValues?: boolean;
13
+ legendDirection?: 'row' | 'column';
14
+ onSingleBarClick?: (barData) => void;
15
+ size?: number;
16
+ title?: string;
17
+ unit?: 'percentage' | 'number';
18
+ variant?: 'vertical' | 'horizontal';
19
+ };
@@ -0,0 +1,131 @@
1
+ import { scaleBand, scaleLinear, scaleOrdinal } from '@visx/scale';
2
+ import { equals, pluck } from 'ramda';
3
+
4
+ import { getValueByUnit } from '../common/utils';
5
+ import { LegendScale } from '../Legend/models';
6
+
7
+ import { BarType } from './models';
8
+
9
+ interface Size {
10
+ height: number;
11
+ width: number;
12
+ }
13
+
14
+ interface useBarStackProps {
15
+ data: Array<BarType>;
16
+ height: number;
17
+ legendRef;
18
+ size: number;
19
+ titleRef;
20
+ unit?: 'percentage' | 'number';
21
+ variant?: 'vertical' | 'horizontal';
22
+ width: number;
23
+ }
24
+ interface useBarStackState {
25
+ barSize: Size;
26
+ colorScale;
27
+ input;
28
+ isVerticalBar: boolean;
29
+ keys: Array<string>;
30
+ legendScale: LegendScale;
31
+ svgContainerSize: Size;
32
+ svgWrapperWidth: number;
33
+ total: number;
34
+ xScale;
35
+ yScale;
36
+ }
37
+
38
+ const useResponsiveBarStack = ({
39
+ data,
40
+ variant,
41
+ height,
42
+ width,
43
+ unit = 'number',
44
+ titleRef,
45
+ legendRef,
46
+ size
47
+ }: useBarStackProps): useBarStackState => {
48
+ const isVerticalBar = equals(variant, 'vertical');
49
+
50
+ const heightOfTitle = titleRef.current?.offsetHeight || 0;
51
+ const widthOfLegend = legendRef.current?.offsetWidth || 0;
52
+
53
+ const horizontalGap = widthOfLegend > 0 ? 12 : 0;
54
+ const verticalGap = heightOfTitle > 0 ? 8 : 0;
55
+
56
+ const svgWrapperWidth = isVerticalBar
57
+ ? size + 36
58
+ : width - widthOfLegend - horizontalGap;
59
+
60
+ const svgContainerSize = {
61
+ height: isVerticalBar ? height - heightOfTitle - verticalGap : size,
62
+ width: isVerticalBar ? size : width - widthOfLegend - horizontalGap
63
+ };
64
+
65
+ const barSize = {
66
+ height: svgContainerSize.height - 16,
67
+ width: svgContainerSize.width - 16
68
+ };
69
+
70
+ const total = Math.floor(data.reduce((acc, { value }) => acc + value, 0));
71
+
72
+ const yScale = isVerticalBar
73
+ ? scaleLinear({
74
+ domain: [0, total]
75
+ })
76
+ : scaleBand({
77
+ domain: [0, 0],
78
+ padding: 0
79
+ });
80
+
81
+ const xScale = isVerticalBar
82
+ ? scaleBand({
83
+ domain: [0, 0],
84
+ padding: 0
85
+ })
86
+ : scaleLinear({
87
+ domain: [0, total]
88
+ });
89
+
90
+ const keys = pluck('label', data);
91
+
92
+ const colorsRange = pluck('color', data);
93
+
94
+ const colorScale = scaleOrdinal({
95
+ domain: keys,
96
+ range: colorsRange
97
+ });
98
+
99
+ const legendScale = {
100
+ domain: data.map(({ value }) => getValueByUnit({ total, unit, value })),
101
+ range: colorsRange
102
+ };
103
+
104
+ const xMax = barSize.width;
105
+ const yMax = barSize.height;
106
+
107
+ xScale.rangeRound([0, xMax]);
108
+ yScale.range([yMax, 0]);
109
+
110
+ const input = data.reduce((acc, { label, value }) => {
111
+ acc[label] = value;
112
+
113
+ return acc;
114
+ }, {});
115
+
116
+ return {
117
+ barSize,
118
+ colorScale,
119
+ input,
120
+ isVerticalBar,
121
+ keys,
122
+ legendScale,
123
+ svgContainerSize,
124
+ svgWrapperWidth,
125
+ total,
126
+ xScale,
127
+ yScale
128
+ };
129
+ };
130
+
131
+ export default useResponsiveBarStack;