1
0
mirror of https://github.com/lensapp/lens.git synced 2025-05-20 05:10:56 +00:00

Fix PieChart tooltips (#5223)

* Add tooltipLabels field to ChartData

Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com>

* Use tooltipLabels in ClusterPieCharts for pods

Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com>

* Check for tooltipLabels field to assign tooltip text

Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com>

* Use tooltipLabels inside overview charts

Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com>

* Expand workload overview charts to fit tooltips

Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com>

* Move tooltipLabels into chart datasets

Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com>

* Move tooltipLabels prop to PieCharts

Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com>

* Little clean up

Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com>

* Getting back id field to PieChartData interface

Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com>

* Id fix

Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com>

* More clean up

Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com>
Signed-off-by: Jim Ehrismann <jehrismann@mirantis.com>
This commit is contained in:
Alex Andreev 2022-04-11 15:21:18 +03:00 committed by Jim Ehrismann
parent f217ba0c33
commit 5bb631f267
4 changed files with 41 additions and 19 deletions

View File

@ -11,7 +11,8 @@ import { ClusterOverviewStore, MetricNodeRole } from "./cluster-overview-store/c
import { Spinner } from "../spinner"; import { Spinner } from "../spinner";
import { Icon } from "../icon"; import { Icon } from "../icon";
import { nodesStore } from "../+nodes/nodes.store"; import { nodesStore } from "../+nodes/nodes.store";
import { ChartData, PieChart } from "../chart"; import type { PieChartData } from "../chart";
import { PieChart } from "../chart";
import { ClusterNoMetrics } from "./cluster-no-metrics"; import { ClusterNoMetrics } from "./cluster-no-metrics";
import { bytesToUnits, cssNames } from "../../utils"; import { bytesToUnits, cssNames } from "../../utils";
import { ThemeStore } from "../../theme.store"; import { ThemeStore } from "../../theme.store";
@ -47,7 +48,7 @@ const NonInjectedClusterPieCharts = observer(({ clusterOverviewStore }: Dependen
const defaultColor = ThemeStore.getInstance().activeTheme.colors.pieChartDefaultColor; const defaultColor = ThemeStore.getInstance().activeTheme.colors.pieChartDefaultColor;
if (!memoryCapacity || !cpuCapacity || !podCapacity || !memoryAllocatableCapacity || !cpuAllocatableCapacity || !podAllocatableCapacity) return null; if (!memoryCapacity || !cpuCapacity || !podCapacity || !memoryAllocatableCapacity || !cpuAllocatableCapacity || !podAllocatableCapacity) return null;
const cpuData: ChartData = { const cpuData: PieChartData = {
datasets: [ datasets: [
{ {
data: [ data: [
@ -94,7 +95,7 @@ const NonInjectedClusterPieCharts = observer(({ clusterOverviewStore }: Dependen
["Capacity", cpuCapacity], ["Capacity", cpuCapacity],
]), ]),
}; };
const memoryData: ChartData = { const memoryData: PieChartData = {
datasets: [ datasets: [
{ {
data: [ data: [
@ -141,7 +142,7 @@ const NonInjectedClusterPieCharts = observer(({ clusterOverviewStore }: Dependen
`Capacity: ${bytesToUnits(memoryCapacity)}`, `Capacity: ${bytesToUnits(memoryCapacity)}`,
], ],
}; };
const podsData: ChartData = { const podsData: PieChartData = {
datasets: [ datasets: [
{ {
data: [ data: [
@ -154,6 +155,10 @@ const NonInjectedClusterPieCharts = observer(({ clusterOverviewStore }: Dependen
], ],
id: "podUsage", id: "podUsage",
label: "Usage", label: "Usage",
tooltipLabels: [
(percent) => `Usage: ${percent}`,
(percent) => `Available: ${percent}`,
],
}, },
], ],
labels: [ labels: [

View File

@ -13,10 +13,4 @@
--workload-status-failed: #{$pod-status-failed-color}; --workload-status-failed: #{$pod-status-failed-color};
--workload-status-terminated: #{$pod-status-terminated-color}; --workload-status-terminated: #{$pod-status-terminated-color};
--workload-status-unknown: #{$pod-status-unknown-color}; --workload-status-unknown: #{$pod-status-unknown-color};
.PieChart {
.chart-container {
width: 110px
}
}
} }

View File

@ -8,9 +8,9 @@ import "./overview-workload-status.scss";
import React from "react"; import React from "react";
import capitalize from "lodash/capitalize"; import capitalize from "lodash/capitalize";
import { observer } from "mobx-react"; import { observer } from "mobx-react";
import type { DatasetTooltipLabel, PieChartData } from "../chart";
import { PieChart } from "../chart"; import { PieChart } from "../chart";
import { cssVar } from "../../utils"; import { cssVar } from "../../utils";
import type { ChartData } from "chart.js";
import { ThemeStore } from "../../theme.store"; import { ThemeStore } from "../../theme.store";
interface Props { interface Props {
@ -27,7 +27,7 @@ export class OverviewWorkloadStatus extends React.Component<Props> {
} }
const cssVars = cssVar(this.elem); const cssVars = cssVar(this.elem);
const chartData: Required<ChartData> = { const chartData: Required<PieChartData> = {
labels: [], labels: [],
datasets: [], datasets: [],
}; };
@ -43,10 +43,12 @@ export class OverviewWorkloadStatus extends React.Component<Props> {
} else { } else {
const data: number[] = []; const data: number[] = [];
const backgroundColor: string[] = []; const backgroundColor: string[] = [];
const tooltipLabels: DatasetTooltipLabel[] = [];
for (const [status, value] of statuses) { for (const [status, value] of statuses) {
data.push(value); data.push(value);
backgroundColor.push(cssVars.get(`--workload-status-${status.toLowerCase()}`).toString()); backgroundColor.push(cssVars.get(`--workload-status-${status.toLowerCase()}`).toString());
tooltipLabels.push(percent => `${capitalize(status)}: ${percent}`);
chartData.labels.push(`${capitalize(status)}: ${value}`); chartData.labels.push(`${capitalize(status)}: ${value}`);
} }
@ -54,6 +56,7 @@ export class OverviewWorkloadStatus extends React.Component<Props> {
data, data,
backgroundColor, backgroundColor,
label: "Status", label: "Status",
tooltipLabels,
}); });
} }

View File

@ -14,6 +14,17 @@ import { ThemeStore } from "../../theme.store";
interface Props extends ChartProps { interface Props extends ChartProps {
} }
export interface PieChartData extends ChartJS.ChartData {
datasets?: PieChartDataSets[];
}
export type DatasetTooltipLabel = (percent: string) => string | string;
interface PieChartDataSets extends ChartJS.ChartDataSets {
id?: string;
tooltipLabels?: DatasetTooltipLabel[];
}
@observer @observer
export class PieChart extends React.Component<Props> { export class PieChart extends React.Component<Props> {
render() { render() {
@ -26,15 +37,24 @@ export class PieChart extends React.Component<Props> {
mode: "index", mode: "index",
callbacks: { callbacks: {
title: () => "", title: () => "",
label: (tooltipItem, data) => { label: (tooltipItem, data: PieChartData) => {
const dataset: any = data["datasets"][tooltipItem.datasetIndex]; const dataset = data.datasets[tooltipItem.datasetIndex];
const metaData = Object.values<{ total: number }>(dataset["_meta"])[0]; const datasetData = dataset.data as number[];
const percent = Math.round((dataset["data"][tooltipItem["index"]] / metaData.total) * 100); const total = datasetData.reduce((acc, cur) => acc + cur, 0);
const label = dataset["label"]; const percent = Math.round((datasetData[tooltipItem.index] as number / total) * 100);
const percentLabel = isNaN(percent) ? "N/A" : `${percent}%`;
const tooltipLabel = dataset.tooltipLabels?.[tooltipItem.index];
let tooltip = `${dataset.label}: ${percentLabel}`;
if (isNaN(percent)) return `${label}: N/A`; if (tooltipLabel) {
if (typeof tooltipLabel === "function") {
tooltip = tooltipLabel(percentLabel);
} else {
tooltip = tooltipLabel;
}
}
return `${label}: ${percent}%`; return tooltip;
}, },
}, },
filter: ({ datasetIndex, index }, { datasets }) => { filter: ({ datasetIndex, index }, { datasets }) => {