Files
vuejs-admin-dashboard-template/src/charts/BarChart01.vue
pasqualevitiello 9cb186991b 1.4.2
2023-02-13 15:44:58 +01:00

167 lines
5.5 KiB
Vue

<template>
<div class="px-5 py-3">
<ul ref="legend" class="flex flex-wrap"></ul>
</div>
<div class="grow">
<canvas ref="canvas" :data="data" :width="width" :height="height"></canvas>
</div>
</template>
<script>
import { ref, onMounted, onUnmounted } from 'vue'
import {
Chart, BarController, BarElement, LinearScale, TimeScale, Tooltip, Legend,
} from 'chart.js'
import 'chartjs-adapter-moment'
// Import utilities
import { tailwindConfig, formatValue } from '../utils/Utils'
Chart.register(BarController, BarElement, LinearScale, TimeScale, Tooltip, Legend)
export default {
name: 'BarChart01',
props: ['data', 'width', 'height'],
setup(props) {
const canvas = ref(null)
const legend = ref(null)
let chart = null
onMounted(() => {
const ctx = canvas.value
chart = new Chart(ctx, {
type: 'bar',
data: props.data,
options: {
layout: {
padding: {
top: 12,
bottom: 16,
left: 20,
right: 20,
},
},
scales: {
y: {
border: {
display: false,
},
ticks: {
maxTicksLimit: 5,
callback: (value) => formatValue(value),
},
},
x: {
type: 'time',
time: {
parser: 'MM-DD-YYYY',
unit: 'month',
displayFormats: {
month: 'MMM YY',
},
},
border: {
display: false,
},
grid: {
display: false,
},
},
},
plugins: {
legend: {
display: false,
},
tooltip: {
callbacks: {
title: () => false, // Disable tooltip title
label: (context) => formatValue(context.parsed.y),
},
},
},
interaction: {
intersect: false,
mode: 'nearest',
},
animation: {
duration: 500,
},
maintainAspectRatio: false,
resizeDelay: 200,
},
plugins: [{
id: 'htmlLegend',
afterUpdate(c, args, options) {
const ul = legend.value
if (!ul) return
// Remove old legend items
while (ul.firstChild) {
ul.firstChild.remove()
}
// Reuse the built-in legendItems generator
const items = c.options.plugins.legend.labels.generateLabels(c)
items.forEach((item) => {
const li = document.createElement('li')
li.style.marginRight = tailwindConfig().theme.margin[4]
// Button element
const button = document.createElement('button')
button.style.display = 'inline-flex'
button.style.alignItems = 'center'
button.style.opacity = item.hidden ? '.3' : ''
button.onclick = () => {
c.setDatasetVisibility(item.datasetIndex, !c.isDatasetVisible(item.datasetIndex))
c.update()
}
// Color box
const box = document.createElement('span')
box.style.display = 'block'
box.style.width = tailwindConfig().theme.width[3]
box.style.height = tailwindConfig().theme.height[3]
box.style.borderRadius = tailwindConfig().theme.borderRadius.full
box.style.marginRight = tailwindConfig().theme.margin[2]
box.style.borderWidth = '3px'
box.style.borderColor = item.fillStyle
box.style.pointerEvents = 'none'
// Label
const labelContainer = document.createElement('span')
labelContainer.style.display = 'flex'
labelContainer.style.alignItems = 'center'
const value = document.createElement('span')
value.style.color = tailwindConfig().theme.colors.gray[800]
value.style.fontSize = tailwindConfig().theme.fontSize['3xl'][0]
value.style.lineHeight = tailwindConfig().theme.fontSize['3xl'][1].lineHeight
value.style.fontWeight = tailwindConfig().theme.fontWeight.bold
value.style.marginRight = tailwindConfig().theme.margin[2]
value.style.pointerEvents = 'none'
const label = document.createElement('span')
label.style.color = tailwindConfig().theme.colors.gray[500]
label.style.fontSize = tailwindConfig().theme.fontSize.sm[0]
label.style.lineHeight = tailwindConfig().theme.fontSize.sm[1].lineHeight
const theValue = c.data.datasets[item.datasetIndex].data.reduce((a, b) => a + b, 0)
const valueText = document.createTextNode(formatValue(theValue))
const labelText = document.createTextNode(item.text)
value.appendChild(valueText)
label.appendChild(labelText)
li.appendChild(button)
button.appendChild(box)
button.appendChild(labelContainer)
labelContainer.appendChild(value)
labelContainer.appendChild(label)
ul.appendChild(li)
})
},
}],
})
})
onUnmounted(() => chart.destroy())
return {
canvas,
legend,
}
}
}
</script>