@centreon/ui 24.4.45-develop.0 → 24.4.45

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 (50) hide show
  1. package/package.json +5 -1
  2. package/src/Graph/BarStack/BarStack.cypress.spec.tsx +154 -0
  3. package/src/Graph/BarStack/BarStack.stories.tsx +123 -0
  4. package/src/Graph/BarStack/BarStack.styles.ts +36 -0
  5. package/src/Graph/BarStack/BarStack.tsx +14 -0
  6. package/src/Graph/BarStack/ResponsiveBarStack.tsx +208 -0
  7. package/src/Graph/BarStack/index.ts +1 -0
  8. package/src/Graph/BarStack/models.ts +19 -0
  9. package/src/Graph/BarStack/useResponsiveBarStack.ts +139 -0
  10. package/src/Graph/Gauge/Gauge.cypress.spec.tsx +102 -0
  11. package/src/Graph/Gauge/Gauge.tsx +1 -1
  12. package/src/Graph/HeatMap/HeatMap.cypress.spec.tsx +145 -0
  13. package/src/Graph/HeatMap/HeatMap.stories.tsx +0 -25
  14. package/src/Graph/HeatMap/ResponsiveHeatMap.tsx +8 -2
  15. package/src/Graph/Legend/Legend.tsx +21 -0
  16. package/src/Graph/Legend/index.ts +1 -0
  17. package/src/Graph/Legend/models.ts +11 -0
  18. package/src/Graph/LineChart/Legend/Legend.styles.ts +1 -1
  19. package/src/Graph/LineChart/Legend/LegendHeader.tsx +1 -1
  20. package/src/Graph/LineChart/Legend/useInteractiveValues.ts +2 -2
  21. package/src/Graph/LineChart/index.tsx +1 -1
  22. package/src/Graph/PieChart/PieChart.cypress.spec.tsx +169 -0
  23. package/src/Graph/PieChart/PieChart.stories.tsx +194 -0
  24. package/src/Graph/PieChart/PieChart.styles.ts +39 -0
  25. package/src/Graph/PieChart/PieChart.tsx +14 -0
  26. package/src/Graph/PieChart/ResponsivePie.tsx +251 -0
  27. package/src/Graph/PieChart/index.ts +1 -0
  28. package/src/Graph/PieChart/models.ts +19 -0
  29. package/src/Graph/PieChart/useResponsivePie.ts +86 -0
  30. package/src/Graph/SingleBar/SingleBar.cypress.spec.tsx +121 -0
  31. package/src/Graph/Text/Text.cypress.spec.tsx +101 -0
  32. package/src/Graph/Text/Text.tsx +1 -1
  33. package/src/Graph/common/testUtils.ts +71 -0
  34. package/src/Graph/common/timeSeries/index.ts +19 -11
  35. package/src/Graph/common/utils.ts +19 -0
  36. package/src/Graph/index.ts +3 -0
  37. package/src/Graph/translatedLabels.ts +1 -0
  38. package/src/Listing/ActionBar/index.tsx +9 -8
  39. package/src/Listing/Cell/DataCell.styles.ts +3 -0
  40. package/src/Listing/Cell/DataCell.tsx +8 -4
  41. package/src/Listing/Listing.cypress.spec.tsx +80 -4
  42. package/src/Listing/Listing.styles.ts +3 -5
  43. package/src/Listing/index.stories.tsx +25 -2
  44. package/src/Listing/index.test.tsx +1 -1
  45. package/src/Listing/index.tsx +3 -1
  46. package/src/Listing/models.ts +1 -0
  47. package/src/api/useMutationQuery/index.test.ts +4 -4
  48. package/src/api/useMutationQuery/index.ts +24 -13
  49. package/src/components/Form/AccessRights/ShareInput/ContactSwitch.tsx +3 -3
  50. package/src/components/Form/AccessRights/ShareInput/ShareInput.tsx +1 -0
@@ -0,0 +1,139 @@
1
+ import { scaleBand, scaleLinear, scaleOrdinal } from '@visx/scale';
2
+ import { equals, isEmpty, pluck, reject } 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
+ areAllValuesNull: boolean;
26
+ barSize: Size;
27
+ colorScale;
28
+ input;
29
+ isVerticalBar: boolean;
30
+ keys: Array<string>;
31
+ legendScale: LegendScale;
32
+ svgContainerSize: Size;
33
+ svgWrapperWidth: number;
34
+ total: number;
35
+ xScale;
36
+ yScale;
37
+ }
38
+
39
+ const useResponsiveBarStack = ({
40
+ data,
41
+ variant,
42
+ height,
43
+ width,
44
+ unit = 'number',
45
+ titleRef,
46
+ legendRef,
47
+ size
48
+ }: useBarStackProps): useBarStackState => {
49
+ const isVerticalBar = equals(variant, 'vertical');
50
+
51
+ const heightOfTitle = titleRef.current?.offsetHeight || 0;
52
+ const widthOfLegend = legendRef.current?.offsetWidth || 0;
53
+
54
+ const horizontalGap = widthOfLegend > 0 ? 12 : 0;
55
+ const verticalGap = heightOfTitle > 0 ? 8 : 0;
56
+
57
+ const svgWrapperWidth = isVerticalBar
58
+ ? size + 24
59
+ : width - widthOfLegend - horizontalGap;
60
+
61
+ const svgContainerSize = {
62
+ height: isVerticalBar ? height - heightOfTitle - verticalGap : size,
63
+ width: isVerticalBar ? size : width - widthOfLegend - horizontalGap
64
+ };
65
+
66
+ const barSize = {
67
+ height: svgContainerSize.height - 16,
68
+ width: svgContainerSize.width - 16
69
+ };
70
+
71
+ const total = Math.floor(data.reduce((acc, { value }) => acc + value, 0));
72
+
73
+ const yScale = isVerticalBar
74
+ ? scaleLinear({
75
+ domain: [0, total],
76
+ nice: true
77
+ })
78
+ : scaleBand({
79
+ domain: [0, 0],
80
+ padding: 0
81
+ });
82
+
83
+ const xScale = isVerticalBar
84
+ ? scaleBand({
85
+ domain: [0, 0],
86
+ padding: 0
87
+ })
88
+ : scaleLinear({
89
+ domain: [0, total],
90
+ nice: true
91
+ });
92
+
93
+ const keys = pluck('label', data);
94
+
95
+ const colorsRange = pluck('color', data);
96
+
97
+ const colorScale = scaleOrdinal({
98
+ domain: keys,
99
+ range: colorsRange
100
+ });
101
+
102
+ const legendScale = {
103
+ domain: data.map(({ value }) => getValueByUnit({ total, unit, value })),
104
+ range: colorsRange
105
+ };
106
+
107
+ const xMax = barSize.width;
108
+ const yMax = barSize.height;
109
+
110
+ xScale.rangeRound([0, xMax]);
111
+ yScale.range([yMax, 0]);
112
+
113
+ const input = data.reduce((acc, { label, value }) => {
114
+ acc[label] = value;
115
+
116
+ return acc;
117
+ }, {});
118
+
119
+ const values = pluck('value', data);
120
+
121
+ const areAllValuesNull = isEmpty(reject((value) => equals(value, 0), values));
122
+
123
+ return {
124
+ areAllValuesNull,
125
+ barSize,
126
+ colorScale,
127
+ input,
128
+ isVerticalBar,
129
+ keys,
130
+ legendScale,
131
+ svgContainerSize,
132
+ svgWrapperWidth,
133
+ total,
134
+ xScale,
135
+ yScale
136
+ };
137
+ };
138
+
139
+ export default useResponsiveBarStack;
@@ -0,0 +1,102 @@
1
+ import dataLastWeek from '../LineChart/mockedData/lastWeek.json';
2
+ import {
3
+ criticalThresholds,
4
+ rangedThresholds,
5
+ successThresholds,
6
+ warningThresholds
7
+ } from '../common/testUtils';
8
+
9
+ import { Props, Gauge } from './Gauge';
10
+
11
+ const initialize = (
12
+ args: Omit<Props, 'data' | 'labels' | 'baseColor'>
13
+ ): void => {
14
+ cy.mount({
15
+ Component: (
16
+ <div style={{ height: '100vh', width: '100vw' }}>
17
+ <Gauge
18
+ baseColor="#000"
19
+ data={dataLastWeek}
20
+ labels={{
21
+ critical: 'Critical',
22
+ warning: 'Warning'
23
+ }}
24
+ {...args}
25
+ />
26
+ </div>
27
+ )
28
+ });
29
+ };
30
+
31
+ describe('Gauge', () => {
32
+ it('does not display the gauge when there is no data', () => {
33
+ initialize({ data: undefined, thresholds: successThresholds });
34
+
35
+ cy.contains('0.41 s').should('not.exist');
36
+ });
37
+
38
+ it('displays the gauge as success when corresponding thresholds are set', () => {
39
+ initialize({ thresholds: successThresholds });
40
+
41
+ cy.contains('0.41 s').should('have.css', 'fill', 'rgb(136, 185, 34)');
42
+ cy.findByTestId('1.1500000000000001-arc').should('be.visible');
43
+ cy.findByTestId('0.15000000000000013-arc').should('be.visible');
44
+
45
+ cy.makeSnapshot();
46
+ });
47
+
48
+ it('displays the gauge as warning when corresponding thresholds are set', () => {
49
+ initialize({ thresholds: warningThresholds });
50
+
51
+ cy.contains('0.41 s').should('have.css', 'fill', 'rgb(253, 155, 39)');
52
+ cy.findByTestId('1.25-arc').should('be.visible');
53
+ cy.findByTestId('0.15000000000000013-arc').should('be.visible');
54
+
55
+ cy.makeSnapshot();
56
+ });
57
+
58
+ it('displays the gauge as critical when corresponding thresholds are set', () => {
59
+ initialize({ thresholds: criticalThresholds });
60
+
61
+ cy.contains('0.41 s').should('have.css', 'fill', 'rgb(255, 74, 74)');
62
+ cy.findByTestId('0.6399999999999999-arc').should('be.visible');
63
+ cy.findByTestId('0.54-arc').should('be.visible');
64
+
65
+ cy.makeSnapshot();
66
+ });
67
+
68
+ it('displays ranged thresholds', () => {
69
+ initialize({ thresholds: rangedThresholds });
70
+
71
+ cy.findByTestId('0.37-arc').should('be.visible');
72
+ cy.findByTestId('0.09999999999999998-arc').should('be.visible');
73
+
74
+ cy.makeSnapshot();
75
+ });
76
+
77
+ it('displays the threshold tooltip when a threshold is hovered', () => {
78
+ initialize({ thresholds: successThresholds });
79
+
80
+ cy.findByTestId('1.1500000000000001-arc').trigger('mouseover', {
81
+ force: true
82
+ });
83
+
84
+ cy.contains('Warning').should('be.visible');
85
+
86
+ cy.findByTestId('0.15000000000000013-arc').trigger('mouseover', {
87
+ force: true
88
+ });
89
+
90
+ cy.contains('Critical').should('be.visible');
91
+
92
+ cy.makeSnapshot();
93
+ });
94
+
95
+ it('displays the value as raw when the prop is set', () => {
96
+ initialize({ displayAsRaw: true, thresholds: successThresholds });
97
+
98
+ cy.contains('0.40663333333 s').should('be.visible');
99
+
100
+ cy.makeSnapshot();
101
+ });
102
+ });
@@ -5,7 +5,7 @@ import { Metric } from '../common/timeSeries/models';
5
5
 
6
6
  import ResponsiveGauge from './ResponsiveGauge';
7
7
 
8
- interface Props {
8
+ export interface Props {
9
9
  baseColor?: string;
10
10
  data?: LineChartData;
11
11
  displayAsRaw?: boolean;
@@ -0,0 +1,145 @@
1
+ import { pluck } from 'ramda';
2
+
3
+ import { Box, Typography } from '@mui/material';
4
+
5
+ import EllipsisTypography from '../../Typography/EllipsisTypography';
6
+
7
+ import HeatMap from './HeatMap';
8
+ import heatMapData from './HeatMapData.json';
9
+ import { HeatMapProps } from './model';
10
+
11
+ const dataIds = pluck('id', heatMapData);
12
+
13
+ interface Data {
14
+ counter: number;
15
+ host: string;
16
+ service: string;
17
+ }
18
+
19
+ const TileContent = ({
20
+ isSmallestSize,
21
+ data
22
+ }: {
23
+ data: Data;
24
+ isSmallestSize: boolean;
25
+ }): JSX.Element | false =>
26
+ !isSmallestSize && (
27
+ <Box
28
+ sx={{
29
+ alignItems: 'center',
30
+ color: 'common.black',
31
+ display: 'flex',
32
+ flexDirection: 'column',
33
+ width: '100%'
34
+ }}
35
+ >
36
+ <EllipsisTypography textAlign="center">{data.host}</EllipsisTypography>
37
+ <EllipsisTypography textAlign="center">{data.service}</EllipsisTypography>
38
+ <EllipsisTypography textAlign="center">{data.counter}</EllipsisTypography>
39
+ </Box>
40
+ );
41
+
42
+ const initialize = ({
43
+ width = '100vw',
44
+ height = '100vh',
45
+ ...args
46
+ }: Omit<HeatMapProps<Data>, 'tiles' | 'children'> & {
47
+ height?: string;
48
+ width?: string;
49
+ }): void => {
50
+ cy.mount({
51
+ Component: (
52
+ <div style={{ height, width }}>
53
+ <HeatMap tiles={heatMapData} {...args}>
54
+ {TileContent}
55
+ </HeatMap>
56
+ </div>
57
+ )
58
+ });
59
+ };
60
+
61
+ describe('HeatMap', () => {
62
+ it('displays tiles', () => {
63
+ initialize({});
64
+ dataIds.forEach((id) => {
65
+ cy.findByTestId(id).should('be.visible');
66
+ });
67
+ });
68
+
69
+ it('does not display the tooltip when the prop is not set and the tile is hovered', () => {
70
+ initialize({});
71
+
72
+ cy.findByTestId(dataIds[0]).trigger('mouseover');
73
+
74
+ cy.findByTestId(`tooltip-${dataIds[0]}`).should('not.exist');
75
+ });
76
+
77
+ it('displays the tooltip when the prop is set and the tile is hovered', () => {
78
+ initialize({
79
+ tooltipContent: ({ data }) => (
80
+ <Typography>
81
+ This is the tooltip for {data.host}-{data.service}-{data.counter}
82
+ </Typography>
83
+ )
84
+ });
85
+
86
+ cy.findByTestId(dataIds[0]).trigger('mouseover');
87
+
88
+ cy.contains(`This is the tooltip for Server-Service Counter-53`).should(
89
+ 'be.visible'
90
+ );
91
+ });
92
+
93
+ it('displays the tooltip conditionally when the prop is set and the tile is hovered', () => {
94
+ initialize({
95
+ displayTooltipCondition: ({ counter }) => counter > 100,
96
+ tooltipContent: ({ data }) => (
97
+ <Typography>
98
+ This is the tooltip for {data.host}-{data.service}-{data.counter}
99
+ </Typography>
100
+ )
101
+ });
102
+
103
+ cy.findByTestId(dataIds[0]).trigger('mouseover');
104
+
105
+ cy.contains(`This is the tooltip for Server-Service Counter-53`).should(
106
+ 'not.exist'
107
+ );
108
+
109
+ cy.findByTestId(dataIds[1]).trigger('mouseover');
110
+
111
+ cy.contains(`This is the tooltip for Server-Service Counter-779`).should(
112
+ 'be.visible'
113
+ );
114
+ });
115
+
116
+ it('displays tiles with fixed size', () => {
117
+ initialize({ height: '200px', tileSizeFixed: true, width: '590px' });
118
+ dataIds.forEach((id) => {
119
+ cy.findByTestId(id).should('be.visible');
120
+ });
121
+ });
122
+
123
+ it('displays tiles as small when the container width is under the breakpoint', () => {
124
+ initialize({ height: '200px', width: '590px' });
125
+ dataIds.forEach((id) => {
126
+ cy.findByTestId(id).should('be.visible');
127
+ });
128
+
129
+ cy.makeSnapshot();
130
+ });
131
+
132
+ it('does not display tiles as small when the contains width is 0', () => {
133
+ initialize({ height: '200px', width: '0px' });
134
+ dataIds.forEach((id) => {
135
+ cy.findByTestId(id).should('not.exist');
136
+ });
137
+
138
+ cy.makeSnapshot();
139
+ });
140
+
141
+ it('displays a single tile', () => {
142
+ initialize({ tiles: [heatMapData[0]] });
143
+ cy.findByTestId(dataIds[0]).should('be.visible');
144
+ });
145
+ });
@@ -3,8 +3,6 @@ import { makeStyles } from 'tss-react/mui';
3
3
 
4
4
  import { Box, Typography } from '@mui/material';
5
5
 
6
- import EllipsisTypography from '../../Typography/EllipsisTypography';
7
-
8
6
  import heatMapData from './HeatMapData.json';
9
7
 
10
8
  import { HeatMap } from '.';
@@ -22,29 +20,6 @@ const meta: Meta<typeof HeatMap<Data>> = {
22
20
  export default meta;
23
21
  type Story = StoryObj<typeof HeatMap<Data>>;
24
22
 
25
- const TileContent = ({
26
- isSmallestSize,
27
- data
28
- }: {
29
- data: Data;
30
- isSmallestSize: boolean;
31
- }): JSX.Element | false =>
32
- !isSmallestSize && (
33
- <Box
34
- sx={{
35
- alignItems: 'center',
36
- color: 'common.black',
37
- display: 'flex',
38
- flexDirection: 'column',
39
- width: '100%'
40
- }}
41
- >
42
- <EllipsisTypography textAlign="center">{data.host}</EllipsisTypography>
43
- <EllipsisTypography textAlign="center">{data.service}</EllipsisTypography>
44
- <EllipsisTypography textAlign="center">{data.counter}</EllipsisTypography>
45
- </Box>
46
- );
47
-
48
23
  const TooltipContent = ({ data }: { data: Data }): JSX.Element => {
49
24
  return (
50
25
  <Box sx={{ backgroundColor: 'common.white', color: 'common.black' }}>
@@ -40,7 +40,7 @@ const ResponsiveHeatMap = <TData,>({
40
40
  const theoricalTotalTilesWidth =
41
41
  tilesLength * tileWidth + (tilesLength - 1) * gap;
42
42
 
43
- if (lt(width, 680) && gt(maxTotalTilesWidth, width)) {
43
+ if (lt(width, 680) && gt(maxTotalTilesWidth, width) && !tileSizeFixed) {
44
44
  return smallestTileSize;
45
45
  }
46
46
 
@@ -66,13 +66,19 @@ const ResponsiveHeatMap = <TData,>({
66
66
  }}
67
67
  >
68
68
  {tiles.map(({ backgroundColor, id, data }) => (
69
- <Box className={classes.heatMapTile} key={id} sx={{ backgroundColor }}>
69
+ <Box
70
+ className={classes.heatMapTile}
71
+ data-testid={id}
72
+ key={id}
73
+ sx={{ backgroundColor }}
74
+ >
70
75
  <Tooltip
71
76
  hasCaret
72
77
  classes={{
73
78
  arrow: cx(classes.heatMapTooltipArrow, arrowClassName),
74
79
  tooltip: classes.heatMapTooltip
75
80
  }}
81
+ data-testid={`tooltip-${data?.id}`}
76
82
  followCursor={false}
77
83
  label={
78
84
  displayTooltipCondition?.(data) &&
@@ -0,0 +1,21 @@
1
+ import { LegendOrdinal } from '@visx/legend';
2
+ import { scaleOrdinal } from '@visx/scale';
3
+
4
+ import { LegendProps } from './models';
5
+
6
+ const Legend = ({ scale, direction = 'column' }: LegendProps): JSX.Element => {
7
+ const legendScale = scaleOrdinal({
8
+ domain: scale.domain,
9
+ range: scale.range
10
+ });
11
+
12
+ return (
13
+ <LegendOrdinal
14
+ direction={direction}
15
+ labelMargin="0 16px 0 0"
16
+ scale={legendScale}
17
+ />
18
+ );
19
+ };
20
+
21
+ export default Legend;
@@ -0,0 +1 @@
1
+ export { default as Legend } from './Legend';
@@ -0,0 +1,11 @@
1
+ export type LegendDirection = 'row' | 'column';
2
+
3
+ export interface LegendScale {
4
+ domain: Array<number | string>;
5
+ range: Array<string>;
6
+ }
7
+
8
+ export interface LegendProps {
9
+ direction?: LegendDirection;
10
+ scale: LegendScale;
11
+ }
@@ -8,7 +8,7 @@ interface MakeStylesProps {
8
8
 
9
9
  export const legendWidth = 21;
10
10
  const legendItemHeight = 5.25;
11
- const legendItemHeightCompact = 1.75;
11
+ const legendItemHeightCompact = 2;
12
12
 
13
13
  export const useStyles = makeStyles<MakeStylesProps>()(
14
14
  (theme, { limitLegendRows }) => ({
@@ -38,7 +38,7 @@ const LegendHeader = ({
38
38
 
39
39
  const getEndText = (): string => {
40
40
  if (value) {
41
- return `${value}${hasUnit ? ` ${unit}` : ''}`;
41
+ return value;
42
42
  }
43
43
 
44
44
  return hasUnit ? ` ${unitName}` : '';
@@ -5,7 +5,7 @@ import { equals, find, isNil } from 'ramda';
5
5
 
6
6
  import { mousePositionAtom } from '../InteractiveComponents/interactionWithGraphAtoms';
7
7
  import {
8
- formatMetricValue,
8
+ formatMetricValueWithUnit,
9
9
  getLineForMetric,
10
10
  getMetrics,
11
11
  getTimeValue
@@ -73,7 +73,7 @@ const useInteractiveValues = ({
73
73
  metric_id
74
74
  }) as Line;
75
75
 
76
- const formattedValue = formatMetricValue({
76
+ const formattedValue = formatMetricValueWithUnit({
77
77
  base,
78
78
  unit,
79
79
  value
@@ -78,7 +78,7 @@ const WrapperLineChart = ({
78
78
  return (
79
79
  <div
80
80
  ref={lineChartRef as MutableRefObject<HTMLDivElement>}
81
- style={{ height: '100%', width: '100%' }}
81
+ style={{ height: '100%', overflowY: 'hidden', width: '100%' }}
82
82
  >
83
83
  <ParentSize>
84
84
  {({