rudeshark.net/packages/client/src/pages/admin/overview.pie.vue

88 lines
1.7 KiB
Vue
Raw Normal View History

<template>
2023-04-08 02:01:42 +02:00
<canvas ref="chartEl"></canvas>
</template>
<script lang="ts" setup>
2023-04-08 02:01:42 +02:00
import { onMounted, onUnmounted, ref, shallowRef } from "vue";
import { Chart } from "chart.js";
import number from "@/filters/number";
import { defaultStore } from "@/store";
import { useChartTooltip } from "@/scripts/use-chart-tooltip";
import { initChart } from "@/scripts/init-chart";
initChart();
const props = defineProps<{
2023-04-08 02:01:42 +02:00
data: {
name: string;
value: number;
color: string;
onClick?: () => void;
}[];
}>();
const chartEl = shallowRef<HTMLCanvasElement>(null);
const { handler: externalTooltipHandler } = useChartTooltip({
2023-04-08 02:01:42 +02:00
position: "middle",
});
let chartInstance: Chart;
onMounted(() => {
chartInstance = new Chart(chartEl.value, {
2023-04-08 02:01:42 +02:00
type: "doughnut",
data: {
2023-04-08 02:01:42 +02:00
labels: props.data.map((x) => x.name),
datasets: [
{
backgroundColor: props.data.map((x) => x.color),
borderColor: getComputedStyle(
2023-07-06 03:28:27 +02:00
document.documentElement,
2023-04-08 02:01:42 +02:00
).getPropertyValue("--panel"),
borderWidth: 2,
hoverOffset: 0,
data: props.data.map((x) => x.value),
},
],
},
options: {
layout: {
padding: {
2022-06-30 13:15:14 +02:00
left: 16,
right: 16,
top: 16,
bottom: 16,
},
},
2022-06-30 15:02:08 +02:00
onClick: (ev) => {
2023-04-08 02:01:42 +02:00
const hit = chartInstance.getElementsAtEventForMode(
ev,
"nearest",
{ intersect: true },
2023-07-06 03:28:27 +02:00
false,
2023-04-08 02:01:42 +02:00
)[0];
2022-06-30 15:02:08 +02:00
if (hit && props.data[hit.index].onClick) {
props.data[hit.index].onClick();
}
},
plugins: {
legend: {
display: false,
},
tooltip: {
enabled: false,
2023-04-08 02:01:42 +02:00
mode: "index",
animation: {
duration: 0,
},
external: externalTooltipHandler,
},
},
},
});
});
</script>
2023-04-08 02:01:42 +02:00
<style lang="scss" scoped></style>