mirror of
https://github.com/kremalicious/umami.git
synced 2024-12-19 15:53:39 +01:00
Custom tooltip and labels.
This commit is contained in:
parent
ce92c7897d
commit
18de85a06d
@ -1,9 +1,49 @@
|
|||||||
import React, { useRef, useEffect } from 'react';
|
import React, { useState, useRef, useEffect, useCallback } from 'react';
|
||||||
import ChartJS from 'chart.js';
|
import ChartJS from 'chart.js';
|
||||||
|
import { format } from 'date-fns';
|
||||||
|
import styles from './PageviewsChart.module.css';
|
||||||
|
|
||||||
export default function PageviewsChart({ data }) {
|
export default function PageviewsChart({ data, unit }) {
|
||||||
const canvas = useRef();
|
const canvas = useRef();
|
||||||
const chart = useRef();
|
const chart = useRef();
|
||||||
|
const [tooltip, setTooltip] = useState({});
|
||||||
|
|
||||||
|
const renderLabel = useCallback(
|
||||||
|
(label, index, values) => {
|
||||||
|
const d = new Date(values[index].value);
|
||||||
|
switch (unit) {
|
||||||
|
case 'day':
|
||||||
|
if (data.pageviews.length > 7) {
|
||||||
|
return index % 2 !== 0 ? format(d, 'MMM d') : '';
|
||||||
|
}
|
||||||
|
return format(d, 'EEE M/d');
|
||||||
|
default:
|
||||||
|
return label;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[unit, data],
|
||||||
|
);
|
||||||
|
|
||||||
|
const renderTooltip = model => {
|
||||||
|
const { caretX, caretY, opacity, title, body, labelColors } = model;
|
||||||
|
console.log(model);
|
||||||
|
|
||||||
|
if (!opacity) {
|
||||||
|
setTooltip({ opacity });
|
||||||
|
} else {
|
||||||
|
const [label, value] = body[0].lines[0].split(':');
|
||||||
|
|
||||||
|
setTooltip({
|
||||||
|
top: caretY,
|
||||||
|
left: caretX,
|
||||||
|
opacity,
|
||||||
|
title: title[0],
|
||||||
|
value,
|
||||||
|
label,
|
||||||
|
labelColor: labelColors[0].backgroundColor,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
function draw() {
|
function draw() {
|
||||||
if (!canvas.current) return;
|
if (!canvas.current) return;
|
||||||
@ -17,16 +57,16 @@ export default function PageviewsChart({ data }) {
|
|||||||
label: 'unique visitors',
|
label: 'unique visitors',
|
||||||
data: data.uniques,
|
data: data.uniques,
|
||||||
lineTension: 0,
|
lineTension: 0,
|
||||||
backgroundColor: 'rgb(146, 86, 217, 0.2)',
|
backgroundColor: 'rgb(146, 86, 217, 0.4)',
|
||||||
borderColor: 'rgb(122, 66, 191, 0.3)',
|
borderColor: 'rgb(122, 66, 191, 0.4)',
|
||||||
borderWidth: 1,
|
borderWidth: 1,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: 'page views',
|
label: 'page views',
|
||||||
data: data.pageviews,
|
data: data.pageviews,
|
||||||
lineTension: 0,
|
lineTension: 0,
|
||||||
backgroundColor: 'rgb(38, 128, 235, 0.2)',
|
backgroundColor: 'rgb(38, 128, 235, 0.4)',
|
||||||
borderColor: 'rgb(13, 102, 208, 0.3)',
|
borderColor: 'rgb(13, 102, 208, 0.4)',
|
||||||
borderWidth: 1,
|
borderWidth: 1,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
@ -36,7 +76,8 @@ export default function PageviewsChart({ data }) {
|
|||||||
duration: 300,
|
duration: 300,
|
||||||
},
|
},
|
||||||
tooltips: {
|
tooltips: {
|
||||||
intersect: false,
|
enabled: false,
|
||||||
|
custom: renderTooltip,
|
||||||
},
|
},
|
||||||
hover: {
|
hover: {
|
||||||
animationDuration: 0,
|
animationDuration: 0,
|
||||||
@ -46,16 +87,18 @@ export default function PageviewsChart({ data }) {
|
|||||||
{
|
{
|
||||||
type: 'time',
|
type: 'time',
|
||||||
distribution: 'series',
|
distribution: 'series',
|
||||||
offset: true,
|
|
||||||
time: {
|
time: {
|
||||||
displayFormats: {
|
unit,
|
||||||
day: 'ddd M/DD',
|
tooltipFormat: 'ddd MMMM DD YYYY',
|
||||||
},
|
},
|
||||||
tooltipFormat: 'ddd M/DD hA',
|
ticks: {
|
||||||
|
callback: renderLabel,
|
||||||
|
maxRotation: 0,
|
||||||
},
|
},
|
||||||
gridLines: {
|
gridLines: {
|
||||||
display: false,
|
display: false,
|
||||||
},
|
},
|
||||||
|
offset: true,
|
||||||
stacked: true,
|
stacked: true,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
@ -71,8 +114,16 @@ export default function PageviewsChart({ data }) {
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
chart.current.data.datasets[0].data = data.uniques;
|
const {
|
||||||
chart.current.data.datasets[1].data = data.pageviews;
|
data: { datasets },
|
||||||
|
options,
|
||||||
|
} = chart.current;
|
||||||
|
|
||||||
|
datasets[0].data = data.uniques;
|
||||||
|
datasets[1].data = data.pageviews;
|
||||||
|
options.scales.xAxes[0].time.unit = unit;
|
||||||
|
options.scales.xAxes[0].ticks.callback = renderLabel;
|
||||||
|
|
||||||
chart.current.update();
|
chart.current.update();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -84,8 +135,21 @@ export default function PageviewsChart({ data }) {
|
|||||||
}, [data]);
|
}, [data]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div>
|
<div className={styles.chart}>
|
||||||
<canvas ref={canvas} width={960} height={400} />
|
<canvas ref={canvas} width={960} height={400} />
|
||||||
|
<Tootip {...tooltip} />
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const Tootip = ({ top, left, opacity, title, value, label, labelColor }) => (
|
||||||
|
<div className={styles.tooltip} style={{ top, left, opacity }}>
|
||||||
|
<div className={styles.content}>
|
||||||
|
<div className={styles.title}>{title}</div>
|
||||||
|
<div className={styles.metric}>
|
||||||
|
<div className={styles.dot} style={{ backgroundColor: labelColor }} />
|
||||||
|
{value} {label}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
56
components/PageviewsChart.module.css
Normal file
56
components/PageviewsChart.module.css
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
.chart {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tooltip {
|
||||||
|
opacity: 0;
|
||||||
|
position: absolute;
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.content {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
color: #000;
|
||||||
|
text-shadow: -1px -1px 0 #fff, 1px -1px 0 #fff, -1px 1px 0 #fff, 1px 1px 0 #fff;
|
||||||
|
text-align: center;
|
||||||
|
width: 150px;
|
||||||
|
height: 50px;
|
||||||
|
transform: translate(-50%, -60px);
|
||||||
|
}
|
||||||
|
|
||||||
|
.content:after {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
bottom: 0;
|
||||||
|
margin: auto;
|
||||||
|
background: #000;
|
||||||
|
opacity: 0.05;
|
||||||
|
border-radius: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.title {
|
||||||
|
font-size: 12px;
|
||||||
|
font-weight: 600;
|
||||||
|
}
|
||||||
|
|
||||||
|
.metric {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
font-size: 14px;
|
||||||
|
font-weight: 600;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dot {
|
||||||
|
width: 10px;
|
||||||
|
height: 10px;
|
||||||
|
border-radius: 100%;
|
||||||
|
border: 1px solid #fff;
|
||||||
|
margin-right: 10px;
|
||||||
|
}
|
@ -34,7 +34,7 @@ export default function WebsiteStats({ websiteId, startDate, endDate, unit }) {
|
|||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
<WebsiteSummary data={{ pageviews, uniques }} />
|
<WebsiteSummary data={{ pageviews, uniques }} />
|
||||||
<PageviewsChart data={{ pageviews, uniques }} />
|
<PageviewsChart data={{ pageviews, uniques }} unit={unit} />
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user