@backstage-community/plugin-apiiro 0.1.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.
- package/CHANGELOG.md +12 -0
- package/README.md +232 -0
- package/config.d.ts +30 -0
- package/dist/App.esm.js +12 -0
- package/dist/App.esm.js.map +1 -0
- package/dist/api/index.esm.js +71 -0
- package/dist/api/index.esm.js.map +1 -0
- package/dist/assets/BulleyeIcon.esm.js +454 -0
- package/dist/assets/BulleyeIcon.esm.js.map +1 -0
- package/dist/assets/NoResultIcon.esm.js +146 -0
- package/dist/assets/NoResultIcon.esm.js.map +1 -0
- package/dist/assets/RiskIcon.esm.js +27 -0
- package/dist/assets/RiskIcon.esm.js.map +1 -0
- package/dist/assets/SettingIcon.esm.js +49 -0
- package/dist/assets/SettingIcon.esm.js.map +1 -0
- package/dist/assets/apiiroLogo/apiiroLogo.esm.js +21 -0
- package/dist/assets/apiiroLogo/apiiroLogo.esm.js.map +1 -0
- package/dist/assets/apiiroLogo/apiiroSidebar.esm.js +23 -0
- package/dist/assets/apiiroLogo/apiiroSidebar.esm.js.map +1 -0
- package/dist/assets/apiiroLogo/apiiroSmall.esm.js +19 -0
- package/dist/assets/apiiroLogo/apiiroSmall.esm.js.map +1 -0
- package/dist/assets/languageIcons/C.esm.js +7 -0
- package/dist/assets/languageIcons/C.esm.js.map +1 -0
- package/dist/assets/languageIcons/Cicd.esm.js +7 -0
- package/dist/assets/languageIcons/Cicd.esm.js.map +1 -0
- package/dist/assets/languageIcons/Clojure.esm.js +7 -0
- package/dist/assets/languageIcons/Clojure.esm.js.map +1 -0
- package/dist/assets/languageIcons/Cpp.esm.js +7 -0
- package/dist/assets/languageIcons/Cpp.esm.js.map +1 -0
- package/dist/assets/languageIcons/Csharp.esm.js +7 -0
- package/dist/assets/languageIcons/Csharp.esm.js.map +1 -0
- package/dist/assets/languageIcons/Dart.esm.js +7 -0
- package/dist/assets/languageIcons/Dart.esm.js.map +1 -0
- package/dist/assets/languageIcons/Go.esm.js +7 -0
- package/dist/assets/languageIcons/Go.esm.js.map +1 -0
- package/dist/assets/languageIcons/Groovy.esm.js +7 -0
- package/dist/assets/languageIcons/Groovy.esm.js.map +1 -0
- package/dist/assets/languageIcons/HTML.esm.js +7 -0
- package/dist/assets/languageIcons/HTML.esm.js.map +1 -0
- package/dist/assets/languageIcons/HclLanguage.esm.js +7 -0
- package/dist/assets/languageIcons/HclLanguage.esm.js.map +1 -0
- package/dist/assets/languageIcons/Java.esm.js +7 -0
- package/dist/assets/languageIcons/Java.esm.js.map +1 -0
- package/dist/assets/languageIcons/Javascript.esm.js +7 -0
- package/dist/assets/languageIcons/Javascript.esm.js.map +1 -0
- package/dist/assets/languageIcons/Kotlin.esm.js +7 -0
- package/dist/assets/languageIcons/Kotlin.esm.js.map +1 -0
- package/dist/assets/languageIcons/ObjectiveC.esm.js +7 -0
- package/dist/assets/languageIcons/ObjectiveC.esm.js.map +1 -0
- package/dist/assets/languageIcons/PHP.esm.js +7 -0
- package/dist/assets/languageIcons/PHP.esm.js.map +1 -0
- package/dist/assets/languageIcons/Perl.esm.js +13 -0
- package/dist/assets/languageIcons/Perl.esm.js.map +1 -0
- package/dist/assets/languageIcons/Python.esm.js +7 -0
- package/dist/assets/languageIcons/Python.esm.js.map +1 -0
- package/dist/assets/languageIcons/Ruby.esm.js +7 -0
- package/dist/assets/languageIcons/Ruby.esm.js.map +1 -0
- package/dist/assets/languageIcons/Rust.esm.js +7 -0
- package/dist/assets/languageIcons/Rust.esm.js.map +1 -0
- package/dist/assets/languageIcons/Scala.esm.js +7 -0
- package/dist/assets/languageIcons/Scala.esm.js.map +1 -0
- package/dist/assets/languageIcons/Swift.esm.js +7 -0
- package/dist/assets/languageIcons/Swift.esm.js.map +1 -0
- package/dist/assets/languageIcons/Terraform.esm.js +7 -0
- package/dist/assets/languageIcons/Terraform.esm.js.map +1 -0
- package/dist/assets/languageIcons/Typescript.esm.js +7 -0
- package/dist/assets/languageIcons/Typescript.esm.js.map +1 -0
- package/dist/assets/languageIcons/Unknown.esm.js +7 -0
- package/dist/assets/languageIcons/Unknown.esm.js.map +1 -0
- package/dist/assets/languageIcons/VB.esm.js +10 -0
- package/dist/assets/languageIcons/VB.esm.js.map +1 -0
- package/dist/assets/languageIcons/YAML.esm.js +7 -0
- package/dist/assets/languageIcons/YAML.esm.js.map +1 -0
- package/dist/assets/providerIcons/Azure.esm.js +7 -0
- package/dist/assets/providerIcons/Azure.esm.js.map +1 -0
- package/dist/assets/providerIcons/Bitbucket.esm.js +7 -0
- package/dist/assets/providerIcons/Bitbucket.esm.js.map +1 -0
- package/dist/assets/providerIcons/Gitlab.esm.js +7 -0
- package/dist/assets/providerIcons/Gitlab.esm.js.map +1 -0
- package/dist/components/ApiiroSidebar.esm.js +10 -0
- package/dist/components/ApiiroSidebar.esm.js.map +1 -0
- package/dist/components/ApplicationsList/ApplicationsList.esm.js +196 -0
- package/dist/components/ApplicationsList/ApplicationsList.esm.js.map +1 -0
- package/dist/components/CalendarDatePicker.esm.js +154 -0
- package/dist/components/CalendarDatePicker.esm.js.map +1 -0
- package/dist/components/CalendarDatePicker.styles.esm.js +198 -0
- package/dist/components/CalendarDatePicker.styles.esm.js.map +1 -0
- package/dist/components/Chip.esm.js +60 -0
- package/dist/components/Chip.esm.js.map +1 -0
- package/dist/components/ChipsList.esm.js +207 -0
- package/dist/components/ChipsList.esm.js.map +1 -0
- package/dist/components/ComponentDisplay.esm.js +42 -0
- package/dist/components/ComponentDisplay.esm.js.map +1 -0
- package/dist/components/DataGrid/CustomColumnMenu.esm.js +29 -0
- package/dist/components/DataGrid/CustomColumnMenu.esm.js.map +1 -0
- package/dist/components/DataGrid/CustomPagination.esm.js +113 -0
- package/dist/components/DataGrid/CustomPagination.esm.js.map +1 -0
- package/dist/components/DataGrid/CustomSearchToolbar.esm.js +117 -0
- package/dist/components/DataGrid/CustomSearchToolbar.esm.js.map +1 -0
- package/dist/components/DataGrid/DataGrid.esm.js +336 -0
- package/dist/components/DataGrid/DataGrid.esm.js.map +1 -0
- package/dist/components/DataGrid/PinColumnMenuItem.esm.js +24 -0
- package/dist/components/DataGrid/PinColumnMenuItem.esm.js.map +1 -0
- package/dist/components/DueDate.esm.js +34 -0
- package/dist/components/DueDate.esm.js.map +1 -0
- package/dist/components/Header.esm.js +27 -0
- package/dist/components/Header.esm.js.map +1 -0
- package/dist/components/MainContributors/MainContributors.esm.js +62 -0
- package/dist/components/MainContributors/MainContributors.esm.js.map +1 -0
- package/dist/components/MetricsGroup/TabMetricsGroup.esm.js +37 -0
- package/dist/components/MetricsGroup/TabMetricsGroup.esm.js.map +1 -0
- package/dist/components/MetricsGroup/WidgetMetricsGroup.esm.js +36 -0
- package/dist/components/MetricsGroup/WidgetMetricsGroup.esm.js.map +1 -0
- package/dist/components/RepositoryDisplay/RepositoryDisplay.esm.js +121 -0
- package/dist/components/RepositoryDisplay/RepositoryDisplay.esm.js.map +1 -0
- package/dist/components/RiskLevel.esm.js +88 -0
- package/dist/components/RiskLevel.esm.js.map +1 -0
- package/dist/components/RiskStatus.esm.js +58 -0
- package/dist/components/RiskStatus.esm.js.map +1 -0
- package/dist/components/SimpleTooltip.esm.js +24 -0
- package/dist/components/SimpleTooltip.esm.js.map +1 -0
- package/dist/components/SourcesDisplay.esm.js +47 -0
- package/dist/components/SourcesDisplay.esm.js.map +1 -0
- package/dist/components/TagsList/TagsList.esm.js +38 -0
- package/dist/components/TagsList/TagsList.esm.js.map +1 -0
- package/dist/components/TeamsDisplay.esm.js +47 -0
- package/dist/components/TeamsDisplay.esm.js.map +1 -0
- package/dist/components/charts/ColumnChart.esm.js +402 -0
- package/dist/components/charts/ColumnChart.esm.js.map +1 -0
- package/dist/components/charts/GaugeChart.esm.js +249 -0
- package/dist/components/charts/GaugeChart.esm.js.map +1 -0
- package/dist/components/charts/LineChart.esm.js +328 -0
- package/dist/components/charts/LineChart.esm.js.map +1 -0
- package/dist/components/charts/PieChart.esm.js +233 -0
- package/dist/components/charts/PieChart.esm.js.map +1 -0
- package/dist/components/common/ChartBox.esm.js +88 -0
- package/dist/components/common/ChartBox.esm.js.map +1 -0
- package/dist/components/common/CustomTooltip.esm.js +255 -0
- package/dist/components/common/CustomTooltip.esm.js.map +1 -0
- package/dist/components/common/ErrorSnackbar.esm.js +39 -0
- package/dist/components/common/ErrorSnackbar.esm.js.map +1 -0
- package/dist/components/common/NotFound.esm.js +30 -0
- package/dist/components/common/NotFound.esm.js.map +1 -0
- package/dist/components/common/SomethingWentWrong.esm.js +35 -0
- package/dist/components/common/SomethingWentWrong.esm.js.map +1 -0
- package/dist/components/common/languageIcons.esm.js +61 -0
- package/dist/components/common/languageIcons.esm.js.map +1 -0
- package/dist/components/common/logoSpinner.esm.js +28 -0
- package/dist/components/common/logoSpinner.esm.js.map +1 -0
- package/dist/components/common/scmProviders.esm.js +41 -0
- package/dist/components/common/scmProviders.esm.js.map +1 -0
- package/dist/components/filters/DiscoveredOnFilter.esm.js +284 -0
- package/dist/components/filters/DiscoveredOnFilter.esm.js.map +1 -0
- package/dist/components/filters/FilterDropdown.esm.js +325 -0
- package/dist/components/filters/FilterDropdown.esm.js.map +1 -0
- package/dist/components/filters/FilterDropdownClear.esm.js +45 -0
- package/dist/components/filters/FilterDropdownClear.esm.js.map +1 -0
- package/dist/components/filters/FilterDropdownList.esm.js +102 -0
- package/dist/components/filters/FilterDropdownList.esm.js.map +1 -0
- package/dist/components/filters/FilterDropdownSearch.esm.js +65 -0
- package/dist/components/filters/FilterDropdownSearch.esm.js.map +1 -0
- package/dist/components/filters/RiskInsightFilter.esm.js +579 -0
- package/dist/components/filters/RiskInsightFilter.esm.js.map +1 -0
- package/dist/components/tiles/MttrVsSLATile.esm.js +170 -0
- package/dist/components/tiles/MttrVsSLATile.esm.js.map +1 -0
- package/dist/components/tiles/RiskOverTimeTile.esm.js +311 -0
- package/dist/components/tiles/RiskOverTimeTile.esm.js.map +1 -0
- package/dist/components/tiles/SLAAdherenceTile.esm.js +115 -0
- package/dist/components/tiles/SLAAdherenceTile.esm.js.map +1 -0
- package/dist/components/tiles/StatusTile.esm.js +235 -0
- package/dist/components/tiles/StatusTile.esm.js.map +1 -0
- package/dist/components/tiles/TopLanguagesTile.esm.js +234 -0
- package/dist/components/tiles/TopLanguagesTile.esm.js.map +1 -0
- package/dist/components/tiles/TopRiskTile.esm.js +208 -0
- package/dist/components/tiles/TopRiskTile.esm.js.map +1 -0
- package/dist/hooks/useUrlFilters.esm.js +102 -0
- package/dist/hooks/useUrlFilters.esm.js.map +1 -0
- package/dist/index.d.ts +28 -0
- package/dist/index.esm.js +42 -0
- package/dist/index.esm.js.map +1 -0
- package/dist/pages/Repositories/Repositories.esm.js +102 -0
- package/dist/pages/Repositories/Repositories.esm.js.map +1 -0
- package/dist/pages/Repositories/tableConfig.esm.js +294 -0
- package/dist/pages/Repositories/tableConfig.esm.js.map +1 -0
- package/dist/pages/Risks/Risks.esm.js +258 -0
- package/dist/pages/Risks/Risks.esm.js.map +1 -0
- package/dist/pages/Risks/tableConfig.esm.js +305 -0
- package/dist/pages/Risks/tableConfig.esm.js.map +1 -0
- package/dist/pages/tab/Tab.esm.js +147 -0
- package/dist/pages/tab/Tab.esm.js.map +1 -0
- package/dist/pages/tab/TabProvider.esm.js +11 -0
- package/dist/pages/tab/TabProvider.esm.js.map +1 -0
- package/dist/pages/widget/Widget.esm.js +161 -0
- package/dist/pages/widget/Widget.esm.js.map +1 -0
- package/dist/pages/widget/WidgetProvider.esm.js +12 -0
- package/dist/pages/widget/WidgetProvider.esm.js.map +1 -0
- package/dist/plugin.esm.js +30 -0
- package/dist/plugin.esm.js.map +1 -0
- package/dist/queries/filterOptions.queries.esm.js +46 -0
- package/dist/queries/filterOptions.queries.esm.js.map +1 -0
- package/dist/queries/mttr-statistics.queries.esm.js +61 -0
- package/dist/queries/mttr-statistics.queries.esm.js.map +1 -0
- package/dist/queries/repository.queries.esm.js +60 -0
- package/dist/queries/repository.queries.esm.js.map +1 -0
- package/dist/queries/risk-score-over-time.queries.esm.js +61 -0
- package/dist/queries/risk-score-over-time.queries.esm.js.map +1 -0
- package/dist/queries/risks.queries.esm.js +65 -0
- package/dist/queries/risks.queries.esm.js.map +1 -0
- package/dist/queries/sla-breach.queries.esm.js +57 -0
- package/dist/queries/sla-breach.queries.esm.js.map +1 -0
- package/dist/queries/top-risks.queries.esm.js +47 -0
- package/dist/queries/top-risks.queries.esm.js.map +1 -0
- package/dist/routes.esm.js +8 -0
- package/dist/routes.esm.js.map +1 -0
- package/dist/theme/themeUtils.esm.js +290 -0
- package/dist/theme/themeUtils.esm.js.map +1 -0
- package/dist/utils/dateFormatter.esm.js +67 -0
- package/dist/utils/dateFormatter.esm.js.map +1 -0
- package/dist/utils/numberFormatter.esm.js +21 -0
- package/dist/utils/numberFormatter.esm.js.map +1 -0
- package/dist/utils/utils.esm.js +27 -0
- package/dist/utils/utils.esm.js.map +1 -0
- package/package.json +90 -0
|
@@ -0,0 +1,249 @@
|
|
|
1
|
+
import { jsx, jsxs } from 'react/jsx-runtime';
|
|
2
|
+
import { Gauge, gaugeClasses, useGaugeState } from '@mui/x-charts/Gauge';
|
|
3
|
+
import { useState, useRef, useEffect } from 'react';
|
|
4
|
+
import { useTheme } from '@mui/material/styles';
|
|
5
|
+
import { getGaugeColors } from '../../theme/themeUtils.esm.js';
|
|
6
|
+
import { CustomTooltip } from '../common/CustomTooltip.esm.js';
|
|
7
|
+
import Typography from '@mui/material/Typography';
|
|
8
|
+
|
|
9
|
+
const getArcColor = (value, tickValue, gaugeColors) => {
|
|
10
|
+
if (tickValue === 0) {
|
|
11
|
+
return gaugeColors.background;
|
|
12
|
+
}
|
|
13
|
+
return value > tickValue ? gaugeColors.warning : gaugeColors.success;
|
|
14
|
+
};
|
|
15
|
+
function GaugeCenterLabel({
|
|
16
|
+
value,
|
|
17
|
+
width,
|
|
18
|
+
height,
|
|
19
|
+
unit = "Hours"
|
|
20
|
+
}) {
|
|
21
|
+
const theme = useTheme();
|
|
22
|
+
return /* @__PURE__ */ jsxs("g", { children: [
|
|
23
|
+
/* @__PURE__ */ jsx(
|
|
24
|
+
"text",
|
|
25
|
+
{
|
|
26
|
+
x: width / 2,
|
|
27
|
+
y: height / 2 / 2 + 20,
|
|
28
|
+
fontSize: "24px",
|
|
29
|
+
fontWeight: "bold",
|
|
30
|
+
fill: theme.palette.text.primary,
|
|
31
|
+
textAnchor: "middle",
|
|
32
|
+
dominantBaseline: "middle",
|
|
33
|
+
children: value ? Math.round(value) : "-"
|
|
34
|
+
}
|
|
35
|
+
),
|
|
36
|
+
/* @__PURE__ */ jsx(
|
|
37
|
+
"text",
|
|
38
|
+
{
|
|
39
|
+
x: width / 2,
|
|
40
|
+
y: height / 2 / 2 + 40,
|
|
41
|
+
fontSize: "12px",
|
|
42
|
+
fontWeight: "100",
|
|
43
|
+
fill: theme.palette.text.secondary,
|
|
44
|
+
textAnchor: "middle",
|
|
45
|
+
dominantBaseline: "middle",
|
|
46
|
+
children: value ? unit : null
|
|
47
|
+
}
|
|
48
|
+
)
|
|
49
|
+
] });
|
|
50
|
+
}
|
|
51
|
+
function GaugeBottomLabels({
|
|
52
|
+
minValue,
|
|
53
|
+
maxValue,
|
|
54
|
+
categoryLabel,
|
|
55
|
+
width = "135px"
|
|
56
|
+
}) {
|
|
57
|
+
const theme = useTheme();
|
|
58
|
+
return /* @__PURE__ */ jsxs(
|
|
59
|
+
"div",
|
|
60
|
+
{
|
|
61
|
+
style: {
|
|
62
|
+
display: "flex",
|
|
63
|
+
justifyContent: "space-between",
|
|
64
|
+
alignItems: "center",
|
|
65
|
+
width,
|
|
66
|
+
marginTop: 7
|
|
67
|
+
},
|
|
68
|
+
children: [
|
|
69
|
+
/* @__PURE__ */ jsx(
|
|
70
|
+
Typography,
|
|
71
|
+
{
|
|
72
|
+
style: { fontSize: "14px", color: theme.palette.text.secondary },
|
|
73
|
+
children: minValue
|
|
74
|
+
}
|
|
75
|
+
),
|
|
76
|
+
/* @__PURE__ */ jsx(
|
|
77
|
+
Typography,
|
|
78
|
+
{
|
|
79
|
+
style: {
|
|
80
|
+
fontSize: "14px",
|
|
81
|
+
fontWeight: 400,
|
|
82
|
+
color: theme.palette.text.primary
|
|
83
|
+
},
|
|
84
|
+
children: categoryLabel
|
|
85
|
+
}
|
|
86
|
+
),
|
|
87
|
+
/* @__PURE__ */ jsx(
|
|
88
|
+
Typography,
|
|
89
|
+
{
|
|
90
|
+
style: { fontSize: "14px", color: theme.palette.text.secondary },
|
|
91
|
+
children: maxValue
|
|
92
|
+
}
|
|
93
|
+
)
|
|
94
|
+
]
|
|
95
|
+
}
|
|
96
|
+
);
|
|
97
|
+
}
|
|
98
|
+
function GaugePointer({
|
|
99
|
+
markerValue,
|
|
100
|
+
min = 0,
|
|
101
|
+
max = 100,
|
|
102
|
+
displayMarkerValue
|
|
103
|
+
}) {
|
|
104
|
+
const theme = useTheme();
|
|
105
|
+
const gaugeColors = getGaugeColors(theme);
|
|
106
|
+
const { startAngle, endAngle, innerRadius, outerRadius, cx, cy } = useGaugeState();
|
|
107
|
+
if (markerValue === null || markerValue === void 0) {
|
|
108
|
+
return null;
|
|
109
|
+
}
|
|
110
|
+
const clampedValue = Math.min(Math.max(markerValue, min), max);
|
|
111
|
+
const valueRatio = (clampedValue - min) / (max - min);
|
|
112
|
+
const valueAngle = startAngle + valueRatio * (endAngle - startAngle);
|
|
113
|
+
const x1 = cx + innerRadius * Math.sin(valueAngle);
|
|
114
|
+
const y1 = cy - innerRadius * Math.cos(valueAngle);
|
|
115
|
+
const x2 = cx + outerRadius * Math.sin(valueAngle);
|
|
116
|
+
const y2 = cy - outerRadius * Math.cos(valueAngle);
|
|
117
|
+
const labelOffset = 10;
|
|
118
|
+
const labelRadius = outerRadius + labelOffset;
|
|
119
|
+
const labelX = cx + labelRadius * Math.sin(valueAngle);
|
|
120
|
+
const labelY = cy - labelRadius * Math.cos(valueAngle);
|
|
121
|
+
return /* @__PURE__ */ jsxs("g", { children: [
|
|
122
|
+
/* @__PURE__ */ jsx(
|
|
123
|
+
"line",
|
|
124
|
+
{
|
|
125
|
+
x1,
|
|
126
|
+
y1,
|
|
127
|
+
x2,
|
|
128
|
+
y2,
|
|
129
|
+
stroke: gaugeColors.pointer,
|
|
130
|
+
strokeWidth: 2,
|
|
131
|
+
strokeLinecap: "round"
|
|
132
|
+
}
|
|
133
|
+
),
|
|
134
|
+
/* @__PURE__ */ jsxs(
|
|
135
|
+
"text",
|
|
136
|
+
{
|
|
137
|
+
x: labelX,
|
|
138
|
+
y: labelY,
|
|
139
|
+
fontSize: "12px",
|
|
140
|
+
textAnchor: "middle",
|
|
141
|
+
dominantBaseline: "middle",
|
|
142
|
+
fill: theme.palette.text.primary,
|
|
143
|
+
children: [
|
|
144
|
+
displayMarkerValue || markerValue,
|
|
145
|
+
"*"
|
|
146
|
+
]
|
|
147
|
+
}
|
|
148
|
+
)
|
|
149
|
+
] });
|
|
150
|
+
}
|
|
151
|
+
function GaugeChart({
|
|
152
|
+
width,
|
|
153
|
+
height,
|
|
154
|
+
value,
|
|
155
|
+
tickValue,
|
|
156
|
+
minValue,
|
|
157
|
+
maxValue,
|
|
158
|
+
tooltip,
|
|
159
|
+
unit = "Hours",
|
|
160
|
+
displayValue,
|
|
161
|
+
displayTickValue
|
|
162
|
+
}) {
|
|
163
|
+
const theme = useTheme();
|
|
164
|
+
const gaugeColors = getGaugeColors(theme);
|
|
165
|
+
const [animatedValue, setAnimatedValue] = useState(0);
|
|
166
|
+
const animationRef = useRef();
|
|
167
|
+
const currentValueRef = useRef(0);
|
|
168
|
+
const animateValue = (start, end, duration = 1e3) => {
|
|
169
|
+
if (animationRef.current) {
|
|
170
|
+
cancelAnimationFrame(animationRef.current);
|
|
171
|
+
}
|
|
172
|
+
const startTime = Date.now();
|
|
173
|
+
const animate = () => {
|
|
174
|
+
const elapsed = Date.now() - startTime;
|
|
175
|
+
const progress = Math.min(elapsed / duration, 1);
|
|
176
|
+
const easeOut = 1 - Math.pow(1 - progress, 3);
|
|
177
|
+
const currentValue = start + (end - start) * easeOut;
|
|
178
|
+
setAnimatedValue(currentValue);
|
|
179
|
+
currentValueRef.current = currentValue;
|
|
180
|
+
if (progress < 1) {
|
|
181
|
+
animationRef.current = requestAnimationFrame(animate);
|
|
182
|
+
}
|
|
183
|
+
};
|
|
184
|
+
animate();
|
|
185
|
+
};
|
|
186
|
+
useEffect(() => {
|
|
187
|
+
animateValue(currentValueRef.current, value);
|
|
188
|
+
return () => {
|
|
189
|
+
if (animationRef.current) {
|
|
190
|
+
cancelAnimationFrame(animationRef.current);
|
|
191
|
+
}
|
|
192
|
+
};
|
|
193
|
+
}, [value]);
|
|
194
|
+
const margin = { top: 20, right: 20, bottom: 20, left: 20 };
|
|
195
|
+
const innerWidth = width - margin.left - margin.right;
|
|
196
|
+
const innerHeight = height - margin.top - margin.bottom;
|
|
197
|
+
const outerRadius = Math.min(innerWidth, innerHeight) / 2.2;
|
|
198
|
+
const innerRadius = outerRadius * 0.72;
|
|
199
|
+
const arcColor = getArcColor(value, tickValue, gaugeColors);
|
|
200
|
+
const gaugeElement = /* @__PURE__ */ jsxs(
|
|
201
|
+
Gauge,
|
|
202
|
+
{
|
|
203
|
+
width,
|
|
204
|
+
height: height / 2,
|
|
205
|
+
startAngle: -90,
|
|
206
|
+
endAngle: 90,
|
|
207
|
+
innerRadius,
|
|
208
|
+
outerRadius,
|
|
209
|
+
value: animatedValue,
|
|
210
|
+
valueMin: minValue,
|
|
211
|
+
valueMax: maxValue,
|
|
212
|
+
text: () => null,
|
|
213
|
+
sx: () => ({
|
|
214
|
+
[`& .${gaugeClasses.valueArc}`]: {
|
|
215
|
+
fill: arcColor,
|
|
216
|
+
transition: "none"
|
|
217
|
+
// Disable default transitions to let react-spring handle it
|
|
218
|
+
},
|
|
219
|
+
[`& .${gaugeClasses.referenceArc}`]: {
|
|
220
|
+
fill: gaugeColors.background
|
|
221
|
+
}
|
|
222
|
+
}),
|
|
223
|
+
children: [
|
|
224
|
+
/* @__PURE__ */ jsx(
|
|
225
|
+
GaugePointer,
|
|
226
|
+
{
|
|
227
|
+
markerValue: tickValue,
|
|
228
|
+
min: minValue,
|
|
229
|
+
max: maxValue,
|
|
230
|
+
displayMarkerValue: displayTickValue
|
|
231
|
+
}
|
|
232
|
+
),
|
|
233
|
+
/* @__PURE__ */ jsx(
|
|
234
|
+
GaugeCenterLabel,
|
|
235
|
+
{
|
|
236
|
+
value: displayValue || animatedValue,
|
|
237
|
+
width,
|
|
238
|
+
height,
|
|
239
|
+
unit
|
|
240
|
+
}
|
|
241
|
+
)
|
|
242
|
+
]
|
|
243
|
+
}
|
|
244
|
+
);
|
|
245
|
+
return tooltip ? /* @__PURE__ */ jsx("div", { style: { display: "inline-block", cursor: "pointer" }, children: /* @__PURE__ */ jsx(CustomTooltip, { title: tooltip, placement: "top", children: /* @__PURE__ */ jsx("div", { style: { width: "100%", height: "100%" }, children: gaugeElement }) }) }) : gaugeElement;
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
export { GaugeBottomLabels, GaugeChart as default };
|
|
249
|
+
//# sourceMappingURL=GaugeChart.esm.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"GaugeChart.esm.js","sources":["../../../src/components/charts/GaugeChart.tsx"],"sourcesContent":["/*\n * Copyright 2025 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\nimport { useGaugeState, Gauge, gaugeClasses } from '@mui/x-charts/Gauge';\nimport { useState, useEffect, useRef } from 'react';\nimport { useTheme } from '@mui/material/styles';\nimport { getGaugeColors } from '../../theme/themeUtils';\nimport CustomTooltip from '../common/CustomTooltip';\nimport Typography from '@mui/material/Typography';\n\nconst getArcColor = (\n value: number,\n tickValue: number,\n gaugeColors: ReturnType<typeof getGaugeColors>,\n) => {\n if (tickValue === 0) {\n return gaugeColors.background;\n }\n return value > tickValue ? gaugeColors.warning : gaugeColors.success;\n};\n\nfunction GaugeCenterLabel({\n value,\n width,\n height,\n unit = 'Hours',\n}: {\n value: number;\n width: number;\n height: number;\n unit?: string;\n}) {\n const theme = useTheme();\n return (\n <g>\n <text\n x={width / 2}\n y={height / 2 / 2 + 20}\n fontSize=\"24px\"\n fontWeight=\"bold\"\n fill={theme.palette.text.primary}\n textAnchor=\"middle\"\n dominantBaseline=\"middle\"\n >\n {value ? Math.round(value) : '-'}\n </text>\n <text\n x={width / 2}\n y={height / 2 / 2 + 40}\n fontSize=\"12px\"\n fontWeight=\"100\"\n fill={theme.palette.text.secondary}\n textAnchor=\"middle\"\n dominantBaseline=\"middle\"\n >\n {value ? unit : null}\n </text>\n </g>\n );\n}\n\nfunction GaugeBottomLabels({\n minValue,\n maxValue,\n categoryLabel,\n width = '135px',\n}: {\n minValue: number;\n maxValue: number;\n categoryLabel: string;\n width?: string;\n}) {\n const theme = useTheme();\n return (\n <div\n style={{\n display: 'flex',\n justifyContent: 'space-between',\n alignItems: 'center',\n width,\n marginTop: 7,\n }}\n >\n <Typography\n style={{ fontSize: '14px', color: theme.palette.text.secondary }}\n >\n {minValue}\n </Typography>\n <Typography\n style={{\n fontSize: '14px',\n fontWeight: 400,\n color: theme.palette.text.primary,\n }}\n >\n {categoryLabel}\n </Typography>\n <Typography\n style={{ fontSize: '14px', color: theme.palette.text.secondary }}\n >\n {maxValue}\n </Typography>\n </div>\n );\n}\n\nfunction GaugePointer({\n markerValue,\n min = 0,\n max = 100,\n displayMarkerValue,\n}: {\n markerValue: number | null | undefined;\n min?: number;\n max?: number;\n displayMarkerValue?: number;\n}) {\n const theme = useTheme();\n const gaugeColors = getGaugeColors(theme);\n const { startAngle, endAngle, innerRadius, outerRadius, cx, cy } =\n useGaugeState();\n\n // Handle null or undefined values - show tick in middle with \"_\" label\n if (markerValue === null || markerValue === undefined) {\n return null;\n }\n\n // Clamp markerValue between min and max\n const clampedValue = Math.min(Math.max(markerValue, min), max);\n\n // Map markerValue to angle (radians)\n const valueRatio = (clampedValue - min) / (max - min);\n const valueAngle = startAngle + valueRatio * (endAngle - startAngle);\n\n // Pointer line endpoints\n const x1 = cx + innerRadius * Math.sin(valueAngle);\n const y1 = cy - innerRadius * Math.cos(valueAngle);\n\n const x2 = cx + outerRadius * Math.sin(valueAngle);\n const y2 = cy - outerRadius * Math.cos(valueAngle);\n\n // Label position outside outer radius\n const labelOffset = 10; // pixels outside outerRadius\n const labelRadius = outerRadius + labelOffset;\n\n const labelX = cx + labelRadius * Math.sin(valueAngle);\n const labelY = cy - labelRadius * Math.cos(valueAngle);\n\n return (\n <g>\n {/* Pointer line */}\n <line\n x1={x1}\n y1={y1}\n x2={x2}\n y2={y2}\n stroke={gaugeColors.pointer}\n strokeWidth={2}\n strokeLinecap=\"round\"\n />\n\n {/* Marker value label positioned outside outerRadius */}\n <text\n x={labelX}\n y={labelY}\n fontSize=\"12px\"\n textAnchor=\"middle\"\n dominantBaseline=\"middle\"\n fill={theme.palette.text.primary}\n >\n {displayMarkerValue || markerValue}*\n </text>\n </g>\n );\n}\n\nexport { GaugeBottomLabels };\n\nexport default function GaugeChart({\n width,\n height,\n value,\n tickValue,\n minValue,\n maxValue,\n tooltip,\n unit = 'Hours',\n displayValue,\n displayTickValue,\n}: {\n width: number;\n height: number;\n value: number;\n tickValue: number;\n minValue: number;\n maxValue: number;\n tooltip?: string;\n unit?: string;\n displayValue?: number;\n displayTickValue?: number;\n}) {\n const theme = useTheme();\n const gaugeColors = getGaugeColors(theme);\n const [animatedValue, setAnimatedValue] = useState(0);\n const animationRef = useRef<number>();\n const currentValueRef = useRef(0);\n\n // Custom animation function\n const animateValue = (\n start: number,\n end: number,\n duration: number = 1000,\n ) => {\n if (animationRef.current) {\n cancelAnimationFrame(animationRef.current);\n }\n\n const startTime = Date.now();\n const animate = () => {\n const elapsed = Date.now() - startTime;\n const progress = Math.min(elapsed / duration, 1);\n\n // Easing function (ease-out)\n const easeOut = 1 - Math.pow(1 - progress, 3);\n const currentValue = start + (end - start) * easeOut;\n\n setAnimatedValue(currentValue);\n currentValueRef.current = currentValue;\n\n if (progress < 1) {\n animationRef.current = requestAnimationFrame(animate);\n }\n };\n\n animate();\n };\n\n // Animate when value changes\n useEffect(() => {\n animateValue(currentValueRef.current, value);\n\n return () => {\n if (animationRef.current) {\n cancelAnimationFrame(animationRef.current);\n }\n };\n }, [value]);\n\n // Define margins (adjust as needed)\n const margin = { top: 20, right: 20, bottom: 20, left: 20 };\n\n const innerWidth = width - margin.left - margin.right; // 220 - 20 - 20 = 180\n const innerHeight = height - margin.top - margin.bottom; // 200 - 20 - 20 = 160 // fixed centerY (you can adjust this)\n\n const outerRadius = Math.min(innerWidth, innerHeight) / 2.2;\n // Math.min(180, 160) / 2.2 ≈ 72.73\n\n const innerRadius = outerRadius * 0.72;\n\n const arcColor = getArcColor(value, tickValue, gaugeColors);\n\n // 72.73 * 0.67 ≈ 48.72\n\n const gaugeElement = (\n <Gauge\n width={width}\n height={height / 2}\n startAngle={-90}\n endAngle={90}\n innerRadius={innerRadius}\n outerRadius={outerRadius}\n value={animatedValue}\n valueMin={minValue}\n valueMax={maxValue}\n text={() => null}\n sx={() => ({\n [`& .${gaugeClasses.valueArc}`]: {\n fill: arcColor,\n transition: 'none', // Disable default transitions to let react-spring handle it\n },\n [`& .${gaugeClasses.referenceArc}`]: {\n fill: gaugeColors.background,\n },\n })}\n >\n <GaugePointer\n markerValue={tickValue}\n min={minValue}\n max={maxValue}\n displayMarkerValue={displayTickValue}\n />\n <GaugeCenterLabel\n value={displayValue || animatedValue}\n width={width}\n height={height}\n unit={unit}\n />\n </Gauge>\n );\n\n // If tooltip is provided, wrap with CustomTooltip, otherwise return gauge directly\n return tooltip ? (\n <div style={{ display: 'inline-block', cursor: 'pointer' }}>\n <CustomTooltip title={tooltip} placement=\"top\">\n <div style={{ width: '100%', height: '100%' }}>{gaugeElement}</div>\n </CustomTooltip>\n </div>\n ) : (\n gaugeElement\n );\n}\n"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,328 @@
|
|
|
1
|
+
import { jsxs, jsx } from 'react/jsx-runtime';
|
|
2
|
+
import { useRef, useState, useEffect, useMemo } from 'react';
|
|
3
|
+
import { LineChart as LineChart$1 } from '@mui/x-charts/LineChart';
|
|
4
|
+
import { axisClasses } from '@mui/x-charts/ChartsAxis';
|
|
5
|
+
import Box from '@mui/material/Box';
|
|
6
|
+
import { styled } from '@mui/material/styles';
|
|
7
|
+
import { formatNumberWithSuffix } from '../../utils/numberFormatter.esm.js';
|
|
8
|
+
|
|
9
|
+
const formatValue = (value) => {
|
|
10
|
+
return formatNumberWithSuffix(value, 1);
|
|
11
|
+
};
|
|
12
|
+
const CustomLegendContainer = styled(Box)(() => ({
|
|
13
|
+
display: "flex",
|
|
14
|
+
justifyContent: "center",
|
|
15
|
+
alignItems: "center",
|
|
16
|
+
gap: "16px",
|
|
17
|
+
flexWrap: "wrap",
|
|
18
|
+
maxWidth: "100%",
|
|
19
|
+
padding: "8px",
|
|
20
|
+
boxSizing: "border-box"
|
|
21
|
+
}));
|
|
22
|
+
const LegendItem = styled(Box)(() => ({
|
|
23
|
+
display: "flex",
|
|
24
|
+
alignItems: "center",
|
|
25
|
+
gap: "8px",
|
|
26
|
+
flexShrink: 0,
|
|
27
|
+
whiteSpace: "nowrap"
|
|
28
|
+
}));
|
|
29
|
+
const LegendColor = styled(Box)(({ color }) => ({
|
|
30
|
+
width: "16px",
|
|
31
|
+
height: "3px",
|
|
32
|
+
backgroundColor: color,
|
|
33
|
+
borderRadius: "1px",
|
|
34
|
+
flexShrink: 0
|
|
35
|
+
}));
|
|
36
|
+
const LegendText = styled("p")(({ theme }) => ({
|
|
37
|
+
fontSize: "11px",
|
|
38
|
+
color: theme.palette.text.secondary,
|
|
39
|
+
fontWeight: 400,
|
|
40
|
+
lineHeight: 1.2,
|
|
41
|
+
margin: 0,
|
|
42
|
+
whiteSpace: "nowrap",
|
|
43
|
+
overflow: "hidden",
|
|
44
|
+
textOverflow: "ellipsis",
|
|
45
|
+
maxWidth: "100px",
|
|
46
|
+
minWidth: "40px"
|
|
47
|
+
}));
|
|
48
|
+
const LineChart = ({
|
|
49
|
+
series,
|
|
50
|
+
width = 400,
|
|
51
|
+
height = 300,
|
|
52
|
+
showLegend = true,
|
|
53
|
+
legendPosition = "bottom",
|
|
54
|
+
showGrid = true,
|
|
55
|
+
formatYValue = formatValue,
|
|
56
|
+
formatXValue,
|
|
57
|
+
showDataPoints = false,
|
|
58
|
+
customXSort,
|
|
59
|
+
formatTooltipXValue,
|
|
60
|
+
yAxisMin,
|
|
61
|
+
yAxisMax,
|
|
62
|
+
yAxisTicks
|
|
63
|
+
}) => {
|
|
64
|
+
const containerRef = useRef(null);
|
|
65
|
+
const [containerWidth, setContainerWidth] = useState(400);
|
|
66
|
+
useEffect(() => {
|
|
67
|
+
const updateWidth = () => {
|
|
68
|
+
if (containerRef.current) {
|
|
69
|
+
const rect = containerRef.current.getBoundingClientRect();
|
|
70
|
+
const newWidth = rect.width;
|
|
71
|
+
setContainerWidth(newWidth > 0 ? Math.floor(newWidth) - 10 : 400);
|
|
72
|
+
}
|
|
73
|
+
};
|
|
74
|
+
updateWidth();
|
|
75
|
+
const resizeObserver = new ResizeObserver(() => {
|
|
76
|
+
updateWidth();
|
|
77
|
+
});
|
|
78
|
+
if (containerRef.current) {
|
|
79
|
+
resizeObserver.observe(containerRef.current);
|
|
80
|
+
}
|
|
81
|
+
window.addEventListener("resize", updateWidth);
|
|
82
|
+
return () => {
|
|
83
|
+
resizeObserver.disconnect();
|
|
84
|
+
window.removeEventListener("resize", updateWidth);
|
|
85
|
+
};
|
|
86
|
+
}, []);
|
|
87
|
+
const chartWidth = typeof width === "string" ? containerWidth : width || 400;
|
|
88
|
+
const { chartSeries, xAxisData } = useMemo(() => {
|
|
89
|
+
const allXValues = /* @__PURE__ */ new Set();
|
|
90
|
+
series.forEach((s) => {
|
|
91
|
+
s.data.forEach((point) => {
|
|
92
|
+
allXValues.add(point.x);
|
|
93
|
+
});
|
|
94
|
+
});
|
|
95
|
+
const sortedXValues = customXSort ? customXSort(Array.from(allXValues)) : Array.from(allXValues).sort((a, b) => {
|
|
96
|
+
if (typeof a === "number" && typeof b === "number") {
|
|
97
|
+
return a - b;
|
|
98
|
+
}
|
|
99
|
+
return String(a).localeCompare(String(b));
|
|
100
|
+
});
|
|
101
|
+
const processedSeries = series.map((s) => ({
|
|
102
|
+
id: s.id,
|
|
103
|
+
label: s.label,
|
|
104
|
+
data: sortedXValues.map((xVal) => {
|
|
105
|
+
const point = s.data.find((p) => p.x === xVal);
|
|
106
|
+
return point ? point.y : null;
|
|
107
|
+
}),
|
|
108
|
+
color: s.color,
|
|
109
|
+
connectNulls: false,
|
|
110
|
+
curve: "linear"
|
|
111
|
+
}));
|
|
112
|
+
return {
|
|
113
|
+
chartSeries: processedSeries,
|
|
114
|
+
xAxisData: sortedXValues
|
|
115
|
+
};
|
|
116
|
+
}, [series, customXSort]);
|
|
117
|
+
const CustomLegend = () => /* @__PURE__ */ jsx(CustomLegendContainer, { children: series.map((s) => /* @__PURE__ */ jsxs(LegendItem, { children: [
|
|
118
|
+
/* @__PURE__ */ jsx(LegendColor, { color: s.color }),
|
|
119
|
+
/* @__PURE__ */ jsx(LegendText, { children: s.label })
|
|
120
|
+
] }, s.id)) });
|
|
121
|
+
return /* @__PURE__ */ jsxs(
|
|
122
|
+
Box,
|
|
123
|
+
{
|
|
124
|
+
ref: containerRef,
|
|
125
|
+
sx: {
|
|
126
|
+
width: "100%",
|
|
127
|
+
display: "flex",
|
|
128
|
+
flexDirection: "column",
|
|
129
|
+
alignItems: "center"
|
|
130
|
+
},
|
|
131
|
+
children: [
|
|
132
|
+
showLegend && legendPosition === "top" && /* @__PURE__ */ jsx(CustomLegend, {}),
|
|
133
|
+
/* @__PURE__ */ jsx(Box, { sx: { width: "100%", overflowX: "auto", minWidth: 0 }, children: /* @__PURE__ */ jsx(
|
|
134
|
+
LineChart$1,
|
|
135
|
+
{
|
|
136
|
+
width: chartWidth,
|
|
137
|
+
height: showLegend && legendPosition === "bottom" ? height - 35 : height,
|
|
138
|
+
series: chartSeries,
|
|
139
|
+
legend: { hidden: true },
|
|
140
|
+
margin: { left: 40, right: 20, top: 20, bottom: 40 },
|
|
141
|
+
xAxis: [
|
|
142
|
+
{
|
|
143
|
+
scaleType: "point",
|
|
144
|
+
data: xAxisData,
|
|
145
|
+
valueFormatter: formatXValue,
|
|
146
|
+
tickInterval: (_, index) => {
|
|
147
|
+
return [0, 7, 14, 21, 28].includes(index);
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
],
|
|
151
|
+
yAxis: [
|
|
152
|
+
{
|
|
153
|
+
valueFormatter: formatYValue,
|
|
154
|
+
tickNumber: yAxisTicks && yAxisTicks.length > 0 ? void 0 : 6,
|
|
155
|
+
tickInterval: yAxisTicks && yAxisTicks.length > 0 ? yAxisTicks : void 0,
|
|
156
|
+
min: yAxisMin,
|
|
157
|
+
max: yAxisMax
|
|
158
|
+
}
|
|
159
|
+
],
|
|
160
|
+
skipAnimation: false,
|
|
161
|
+
grid: showGrid ? { horizontal: true, vertical: false } : void 0,
|
|
162
|
+
tooltip: formatTooltipXValue ? {
|
|
163
|
+
axisContent: (props) => {
|
|
164
|
+
const dataIndex = props?.dataIndex ?? 0;
|
|
165
|
+
const rawX = xAxisData?.[dataIndex];
|
|
166
|
+
const header = rawX !== void 0 ? formatTooltipXValue(rawX) : props?.label;
|
|
167
|
+
return /* @__PURE__ */ jsx("div", { className: "MuiChartsTooltip-root", children: /* @__PURE__ */ jsx("table", { className: "MuiChartsTooltip-table", children: /* @__PURE__ */ jsxs("tbody", { children: [
|
|
168
|
+
/* @__PURE__ */ jsx("tr", { className: "MuiChartsTooltip-row", children: /* @__PURE__ */ jsx("td", { className: "MuiChartsTooltip-cell", colSpan: 2, children: header }) }),
|
|
169
|
+
chartSeries.map((s, idx) => {
|
|
170
|
+
const value = s.data?.[dataIndex];
|
|
171
|
+
let formatted;
|
|
172
|
+
if (typeof value === "number") {
|
|
173
|
+
formatted = formatYValue(value);
|
|
174
|
+
} else if (value === null || value === void 0) {
|
|
175
|
+
formatted = "-";
|
|
176
|
+
} else {
|
|
177
|
+
formatted = String(value);
|
|
178
|
+
}
|
|
179
|
+
return /* @__PURE__ */ jsxs("tr", { className: "MuiChartsTooltip-row", children: [
|
|
180
|
+
/* @__PURE__ */ jsx("td", { className: "MuiChartsTooltip-labelCell", children: s.label }),
|
|
181
|
+
/* @__PURE__ */ jsx("td", { className: "MuiChartsTooltip-valueCell", children: formatted })
|
|
182
|
+
] }, idx);
|
|
183
|
+
})
|
|
184
|
+
] }) }) });
|
|
185
|
+
}
|
|
186
|
+
} : void 0,
|
|
187
|
+
sx: (theme) => ({
|
|
188
|
+
[`.${axisClasses.left} .${axisClasses.line}`]: {
|
|
189
|
+
display: "none"
|
|
190
|
+
},
|
|
191
|
+
[`.${axisClasses.bottom} .${axisClasses.line}`]: {
|
|
192
|
+
display: "none"
|
|
193
|
+
},
|
|
194
|
+
[`.${axisClasses.left} .${axisClasses.tick}`]: {
|
|
195
|
+
display: "none"
|
|
196
|
+
},
|
|
197
|
+
[`.${axisClasses.bottom} .${axisClasses.tick}`]: {
|
|
198
|
+
stroke: theme.palette.divider,
|
|
199
|
+
strokeWidth: 1,
|
|
200
|
+
transform: "translate(0, -5px)"
|
|
201
|
+
},
|
|
202
|
+
[`.${axisClasses.bottom} .${axisClasses.tickLabel}`]: {
|
|
203
|
+
transform: "translate(5px, 0px)"
|
|
204
|
+
},
|
|
205
|
+
"& .MuiChartsGrid-line": {
|
|
206
|
+
strokeDasharray: "3 3",
|
|
207
|
+
stroke: theme.palette.divider,
|
|
208
|
+
strokeWidth: 1
|
|
209
|
+
},
|
|
210
|
+
"& .MuiLineElement-root": {
|
|
211
|
+
strokeWidth: 2
|
|
212
|
+
},
|
|
213
|
+
"& .MuiMarkElement-root": {
|
|
214
|
+
r: showDataPoints ? 3 : 0,
|
|
215
|
+
strokeWidth: 2,
|
|
216
|
+
fill: theme.palette.background.paper,
|
|
217
|
+
opacity: showDataPoints ? 1 : 0,
|
|
218
|
+
transition: "all 0.2s ease-in-out",
|
|
219
|
+
"&:hover": {
|
|
220
|
+
r: 4,
|
|
221
|
+
opacity: 1
|
|
222
|
+
}
|
|
223
|
+
}
|
|
224
|
+
}),
|
|
225
|
+
slotProps: {
|
|
226
|
+
popper: {
|
|
227
|
+
placement: "auto",
|
|
228
|
+
modifiers: [
|
|
229
|
+
{
|
|
230
|
+
name: "flip",
|
|
231
|
+
enabled: true,
|
|
232
|
+
options: {
|
|
233
|
+
fallbackPlacements: ["top", "bottom", "right", "left"],
|
|
234
|
+
allowedAutoPlacements: ["top", "bottom"]
|
|
235
|
+
}
|
|
236
|
+
},
|
|
237
|
+
{
|
|
238
|
+
name: "preventOverflow",
|
|
239
|
+
enabled: true,
|
|
240
|
+
options: {
|
|
241
|
+
boundary: "clippingParents",
|
|
242
|
+
padding: 8
|
|
243
|
+
}
|
|
244
|
+
},
|
|
245
|
+
{
|
|
246
|
+
name: "offset",
|
|
247
|
+
enabled: true,
|
|
248
|
+
options: {
|
|
249
|
+
offset: [90, 10]
|
|
250
|
+
}
|
|
251
|
+
},
|
|
252
|
+
{
|
|
253
|
+
name: "arrow",
|
|
254
|
+
enabled: false
|
|
255
|
+
}
|
|
256
|
+
],
|
|
257
|
+
sx: (theme) => ({
|
|
258
|
+
"& .MuiChartsTooltip-root": {
|
|
259
|
+
backgroundColor: theme.palette.background.paper,
|
|
260
|
+
border: `1px solid ${theme.palette.divider}`,
|
|
261
|
+
borderRadius: "12px",
|
|
262
|
+
filter: theme.palette.mode === "dark" ? "drop-shadow(0 4px 12px rgba(0,0,0,0.3))" : "drop-shadow(0 4px 12px rgba(0,0,0,0.1))",
|
|
263
|
+
padding: "12px 16px",
|
|
264
|
+
minWidth: "140px",
|
|
265
|
+
fontSize: "12px"
|
|
266
|
+
},
|
|
267
|
+
"& .MuiChartsTooltip-table": {
|
|
268
|
+
margin: 0,
|
|
269
|
+
borderSpacing: 0,
|
|
270
|
+
width: "100%"
|
|
271
|
+
},
|
|
272
|
+
"& .MuiChartsTooltip-row": {
|
|
273
|
+
"&:first-of-type": {
|
|
274
|
+
"& .MuiChartsTooltip-cell": {
|
|
275
|
+
fontSize: "16px",
|
|
276
|
+
fontWeight: 600,
|
|
277
|
+
color: theme.palette.text.secondary,
|
|
278
|
+
paddingBottom: "16px",
|
|
279
|
+
borderBottom: "none"
|
|
280
|
+
}
|
|
281
|
+
},
|
|
282
|
+
"&:not(:first-of-type)": {
|
|
283
|
+
"& .MuiChartsTooltip-cell": {
|
|
284
|
+
padding: "4px 0",
|
|
285
|
+
borderBottom: "none"
|
|
286
|
+
},
|
|
287
|
+
"& .MuiChartsTooltip-labelCell": {
|
|
288
|
+
position: "relative",
|
|
289
|
+
paddingLeft: "16px",
|
|
290
|
+
paddingRight: "4px",
|
|
291
|
+
paddingBottom: "8px",
|
|
292
|
+
fontSize: "14px",
|
|
293
|
+
color: theme.palette.text.secondary,
|
|
294
|
+
"&::before": {
|
|
295
|
+
content: '""',
|
|
296
|
+
position: "absolute",
|
|
297
|
+
left: "0",
|
|
298
|
+
top: "10%",
|
|
299
|
+
width: "3px",
|
|
300
|
+
height: "16px",
|
|
301
|
+
backgroundColor: theme.palette.primary.main,
|
|
302
|
+
borderRadius: "2px"
|
|
303
|
+
}
|
|
304
|
+
},
|
|
305
|
+
"& .MuiChartsTooltip-valueCell": {
|
|
306
|
+
fontSize: "14px",
|
|
307
|
+
fontWeight: 600,
|
|
308
|
+
color: theme.palette.text.secondary,
|
|
309
|
+
textAlign: "right",
|
|
310
|
+
paddingLeft: "4px",
|
|
311
|
+
paddingBottom: "8px",
|
|
312
|
+
verticalAlign: "middle"
|
|
313
|
+
}
|
|
314
|
+
}
|
|
315
|
+
}
|
|
316
|
+
})
|
|
317
|
+
}
|
|
318
|
+
}
|
|
319
|
+
}
|
|
320
|
+
) }),
|
|
321
|
+
showLegend && legendPosition === "bottom" && /* @__PURE__ */ jsx(CustomLegend, {})
|
|
322
|
+
]
|
|
323
|
+
}
|
|
324
|
+
);
|
|
325
|
+
};
|
|
326
|
+
|
|
327
|
+
export { LineChart, LineChart as default };
|
|
328
|
+
//# sourceMappingURL=LineChart.esm.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"LineChart.esm.js","sources":["../../../src/components/charts/LineChart.tsx"],"sourcesContent":["/*\n * Copyright 2025 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\nimport { useMemo, useRef, useEffect, useState } from 'react';\nimport { LineChart as MuiLineChart } from '@mui/x-charts/LineChart';\nimport { axisClasses } from '@mui/x-charts/ChartsAxis';\nimport Box from '@mui/material/Box';\nimport { styled } from '@mui/material/styles';\nimport { formatNumberWithSuffix } from '../../utils/numberFormatter';\n\nexport interface LineChartDataPoint {\n x: string | number;\n y: number;\n}\n\nexport type LineChartCurveType =\n | 'linear'\n | 'step'\n | 'natural'\n | 'monotoneX'\n | 'monotoneY'\n | 'catmullRom';\n\nexport interface LineChartSeries {\n id: string;\n label: string;\n data: LineChartDataPoint[];\n color: string;\n connectNulls?: boolean;\n curve?: LineChartCurveType;\n}\n\nexport interface LineChartProps {\n series: LineChartSeries[];\n width?: number | string;\n height?: number;\n showLegend?: boolean;\n legendPosition?: 'top' | 'bottom';\n showGrid?: boolean;\n xAxisLabel?: string;\n yAxisLabel?: string;\n formatYValue?: (value: number) => string;\n formatXValue?: (value: string | number) => string;\n showDataPoints?: boolean;\n customXSort?: (values: (string | number)[]) => (string | number)[];\n // Optional: allows different formatting for tooltip header than x-axis ticks\n formatTooltipXValue?: (value: string | number) => string;\n // Optional: custom y-axis range\n yAxisMin?: number;\n yAxisMax?: number;\n // Optional: explicit y-axis tick values (overrides tickNumber)\n yAxisTicks?: number[];\n}\n\nconst formatValue = (value: number): string => {\n return formatNumberWithSuffix(value, 1);\n};\n\nconst CustomLegendContainer = styled(Box)(() => ({\n display: 'flex',\n justifyContent: 'center',\n alignItems: 'center',\n gap: '16px',\n flexWrap: 'wrap',\n maxWidth: '100%',\n padding: '8px',\n boxSizing: 'border-box',\n}));\n\nconst LegendItem = styled(Box)(() => ({\n display: 'flex',\n alignItems: 'center',\n gap: '8px',\n flexShrink: 0,\n whiteSpace: 'nowrap',\n}));\n\nconst LegendColor = styled(Box)<{ color: string }>(({ color }) => ({\n width: '16px',\n height: '3px',\n backgroundColor: color,\n borderRadius: '1px',\n flexShrink: 0,\n}));\n\nconst LegendText = styled('p')(({ theme }) => ({\n fontSize: '11px',\n color: theme.palette.text.secondary,\n fontWeight: 400,\n lineHeight: 1.2,\n margin: 0,\n whiteSpace: 'nowrap',\n overflow: 'hidden',\n textOverflow: 'ellipsis',\n maxWidth: '100px',\n minWidth: '40px',\n}));\n\nexport const LineChart = ({\n series,\n width = 400,\n height = 300,\n showLegend = true,\n legendPosition = 'bottom',\n showGrid = true,\n formatYValue = formatValue,\n formatXValue,\n showDataPoints = false,\n customXSort,\n formatTooltipXValue,\n yAxisMin,\n yAxisMax,\n yAxisTicks,\n}: LineChartProps) => {\n const containerRef = useRef<HTMLDivElement>(null);\n const [containerWidth, setContainerWidth] = useState<number>(400);\n\n useEffect(() => {\n const updateWidth = () => {\n if (containerRef.current) {\n const rect = containerRef.current.getBoundingClientRect();\n const newWidth = rect.width;\n setContainerWidth(newWidth > 0 ? Math.floor(newWidth) - 10 : 400);\n }\n };\n\n // Initial measurement\n updateWidth();\n\n // Use ResizeObserver for better accuracy\n const resizeObserver = new ResizeObserver(() => {\n updateWidth();\n });\n\n if (containerRef.current) {\n resizeObserver.observe(containerRef.current);\n }\n\n window.addEventListener('resize', updateWidth);\n\n return () => {\n resizeObserver.disconnect();\n window.removeEventListener('resize', updateWidth);\n };\n }, []);\n\n const chartWidth = typeof width === 'string' ? containerWidth : width || 400;\n\n const { chartSeries, xAxisData } = useMemo(() => {\n const allXValues = new Set<string | number>();\n\n // Collect all unique x values\n series.forEach(s => {\n s.data.forEach(point => {\n allXValues.add(point.x);\n });\n });\n\n const sortedXValues = customXSort\n ? customXSort(Array.from(allXValues))\n : Array.from(allXValues).sort((a, b) => {\n if (typeof a === 'number' && typeof b === 'number') {\n return a - b;\n }\n return String(a).localeCompare(String(b));\n });\n\n const processedSeries = series.map(s => ({\n id: s.id,\n label: s.label,\n data: sortedXValues.map(xVal => {\n const point = s.data.find(p => p.x === xVal);\n return point ? point.y : null;\n }),\n color: s.color,\n connectNulls: false,\n curve: 'linear' as const,\n }));\n\n return {\n chartSeries: processedSeries,\n xAxisData: sortedXValues,\n };\n }, [series, customXSort]);\n\n const CustomLegend = () => (\n <CustomLegendContainer>\n {series.map(s => (\n <LegendItem key={s.id}>\n <LegendColor color={s.color} />\n <LegendText>{s.label}</LegendText>\n </LegendItem>\n ))}\n </CustomLegendContainer>\n );\n\n return (\n <Box\n ref={containerRef}\n sx={{\n width: '100%',\n display: 'flex',\n flexDirection: 'column',\n alignItems: 'center',\n }}\n >\n {showLegend && legendPosition === 'top' && <CustomLegend />}\n\n <Box sx={{ width: '100%', overflowX: 'auto', minWidth: 0 }}>\n <MuiLineChart\n width={chartWidth}\n height={\n showLegend && legendPosition === 'bottom' ? height - 35 : height\n }\n series={chartSeries}\n legend={{ hidden: true }}\n margin={{ left: 40, right: 20, top: 20, bottom: 40 }}\n xAxis={[\n {\n scaleType: 'point',\n data: xAxisData,\n valueFormatter: formatXValue,\n tickInterval: (_, index) => {\n // Show ticks at indices: 0, 7, 14, 21, 28 (approximately every 7 days, starting from first date)\n return [0, 7, 14, 21, 28].includes(index);\n },\n },\n ]}\n yAxis={[\n {\n valueFormatter: formatYValue,\n tickNumber: yAxisTicks && yAxisTicks.length > 0 ? undefined : 6,\n tickInterval:\n yAxisTicks && yAxisTicks.length > 0 ? yAxisTicks : undefined,\n min: yAxisMin,\n max: yAxisMax,\n },\n ]}\n skipAnimation={false}\n grid={showGrid ? { horizontal: true, vertical: false } : undefined}\n tooltip={\n formatTooltipXValue\n ? {\n axisContent: (props: any) => {\n const dataIndex = props?.dataIndex ?? 0;\n const rawX = xAxisData?.[dataIndex];\n const header =\n rawX !== undefined\n ? formatTooltipXValue(rawX)\n : props?.label;\n\n return (\n <div className=\"MuiChartsTooltip-root\">\n <table className=\"MuiChartsTooltip-table\">\n <tbody>\n <tr className=\"MuiChartsTooltip-row\">\n <td className=\"MuiChartsTooltip-cell\" colSpan={2}>\n {header}\n </td>\n </tr>\n {chartSeries.map((s, idx) => {\n const value = s.data?.[dataIndex];\n let formatted: string;\n if (typeof value === 'number') {\n formatted = formatYValue(value);\n } else if (\n value === null ||\n value === undefined\n ) {\n formatted = '-';\n } else {\n formatted = String(value);\n }\n return (\n <tr key={idx} className=\"MuiChartsTooltip-row\">\n <td className=\"MuiChartsTooltip-labelCell\">\n {s.label}\n </td>\n <td className=\"MuiChartsTooltip-valueCell\">\n {formatted}\n </td>\n </tr>\n );\n })}\n </tbody>\n </table>\n </div>\n );\n },\n }\n : undefined\n }\n sx={theme => ({\n [`.${axisClasses.left} .${axisClasses.line}`]: {\n display: 'none',\n },\n [`.${axisClasses.bottom} .${axisClasses.line}`]: {\n display: 'none',\n },\n [`.${axisClasses.left} .${axisClasses.tick}`]: {\n display: 'none',\n },\n [`.${axisClasses.bottom} .${axisClasses.tick}`]: {\n stroke: theme.palette.divider,\n strokeWidth: 1,\n transform: 'translate(0, -5px)',\n },\n [`.${axisClasses.bottom} .${axisClasses.tickLabel}`]: {\n transform: 'translate(5px, 0px)',\n },\n '& .MuiChartsGrid-line': {\n strokeDasharray: '3 3',\n stroke: theme.palette.divider,\n strokeWidth: 1,\n },\n '& .MuiLineElement-root': {\n strokeWidth: 2,\n },\n '& .MuiMarkElement-root': {\n r: showDataPoints ? 3 : 0,\n strokeWidth: 2,\n fill: theme.palette.background.paper,\n opacity: showDataPoints ? 1 : 0,\n transition: 'all 0.2s ease-in-out',\n '&:hover': {\n r: 4,\n opacity: 1,\n },\n },\n })}\n slotProps={{\n popper: {\n placement: 'auto',\n modifiers: [\n {\n name: 'flip',\n enabled: true,\n options: {\n fallbackPlacements: ['top', 'bottom', 'right', 'left'],\n allowedAutoPlacements: ['top', 'bottom'],\n },\n },\n {\n name: 'preventOverflow',\n enabled: true,\n options: {\n boundary: 'clippingParents',\n padding: 8,\n },\n },\n {\n name: 'offset',\n enabled: true,\n options: {\n offset: [90, 10],\n },\n },\n {\n name: 'arrow',\n enabled: false,\n },\n ],\n sx: theme => ({\n '& .MuiChartsTooltip-root': {\n backgroundColor: theme.palette.background.paper,\n border: `1px solid ${theme.palette.divider}`,\n borderRadius: '12px',\n filter:\n theme.palette.mode === 'dark'\n ? 'drop-shadow(0 4px 12px rgba(0,0,0,0.3))'\n : 'drop-shadow(0 4px 12px rgba(0,0,0,0.1))',\n padding: '12px 16px',\n minWidth: '140px',\n fontSize: '12px',\n },\n '& .MuiChartsTooltip-table': {\n margin: 0,\n borderSpacing: 0,\n width: '100%',\n },\n '& .MuiChartsTooltip-row': {\n '&:first-of-type': {\n '& .MuiChartsTooltip-cell': {\n fontSize: '16px',\n fontWeight: 600,\n color: theme.palette.text.secondary,\n paddingBottom: '16px',\n borderBottom: 'none',\n },\n },\n '&:not(:first-of-type)': {\n '& .MuiChartsTooltip-cell': {\n padding: '4px 0',\n borderBottom: 'none',\n },\n '& .MuiChartsTooltip-labelCell': {\n position: 'relative',\n paddingLeft: '16px',\n paddingRight: '4px',\n paddingBottom: '8px',\n fontSize: '14px',\n color: theme.palette.text.secondary,\n '&::before': {\n content: '\"\"',\n position: 'absolute',\n left: '0',\n top: '10%',\n width: '3px',\n height: '16px',\n backgroundColor: theme.palette.primary.main,\n borderRadius: '2px',\n },\n },\n '& .MuiChartsTooltip-valueCell': {\n fontSize: '14px',\n fontWeight: 600,\n color: theme.palette.text.secondary,\n textAlign: 'right',\n paddingLeft: '4px',\n paddingBottom: '8px',\n verticalAlign: 'middle',\n },\n },\n },\n }),\n },\n }}\n />\n </Box>\n\n {showLegend && legendPosition === 'bottom' && <CustomLegend />}\n </Box>\n );\n};\n\nexport default LineChart;\n"],"names":["MuiLineChart"],"mappings":"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"}
|