165 lines
4.4 KiB
Vue
165 lines
4.4 KiB
Vue
<script>
|
|
import {Bar} from 'vue-chartjs';
|
|
import {
|
|
Chart,
|
|
Tooltip,
|
|
BarElement,
|
|
CategoryScale,
|
|
LinearScale,
|
|
} from 'chart.js';
|
|
import {chartJsColors} from '../utils/color.js';
|
|
import {createApp} from 'vue';
|
|
|
|
Chart.defaults.color = chartJsColors.text;
|
|
Chart.defaults.borderColor = chartJsColors.border;
|
|
|
|
Chart.register(
|
|
CategoryScale,
|
|
LinearScale,
|
|
BarElement,
|
|
Tooltip,
|
|
);
|
|
|
|
const sfc = {
|
|
components: {Bar},
|
|
props: {
|
|
locale: {
|
|
type: Object,
|
|
required: true,
|
|
},
|
|
},
|
|
data: () => ({
|
|
colors: {
|
|
barColor: 'green',
|
|
},
|
|
|
|
// possible keys:
|
|
// * avatar_link: (...)
|
|
// * commits: (...)
|
|
// * home_link: (...)
|
|
// * login: (...)
|
|
// * name: (...)
|
|
activityTopAuthors: window.config.pageData.repoActivityTopAuthors || [],
|
|
i18nCommitActivity: this,
|
|
}),
|
|
methods: {
|
|
graphPoints() {
|
|
return {
|
|
datasets: [{
|
|
label: this.locale.commitActivity,
|
|
data: this.activityTopAuthors.map((item) => item.commits),
|
|
backgroundColor: this.colors.barColor,
|
|
barThickness: 40,
|
|
borderWidth: 0,
|
|
tension: 0.3,
|
|
}],
|
|
labels: this.activityTopAuthors.map((item) => item.name),
|
|
};
|
|
},
|
|
getOptions() {
|
|
return {
|
|
responsive: true,
|
|
maintainAspectRatio: false,
|
|
animation: true,
|
|
scales: {
|
|
x: {
|
|
type: 'category',
|
|
grid: {
|
|
display: false,
|
|
},
|
|
ticks: {
|
|
// Disable the drawing of the labels on the x-asis and force them all
|
|
// of them to be 'shown', this avoids them being internally skipped
|
|
// for some data points. We rely on the internally generated ticks
|
|
// to know where to draw our own ticks. Set rotation to 90 degree
|
|
// and disable autoSkip. autoSkip is disabled to ensure no ticks are
|
|
// skipped and rotation is set to avoid messing with the width of the chart.
|
|
color: 'transparent',
|
|
minRotation: 90,
|
|
maxRotation: 90,
|
|
autoSkip: false,
|
|
},
|
|
},
|
|
y: {
|
|
ticks: {
|
|
stepSize: 1,
|
|
},
|
|
},
|
|
},
|
|
};
|
|
},
|
|
},
|
|
mounted() {
|
|
const refStyle = window.getComputedStyle(this.$refs.style);
|
|
this.colors.barColor = refStyle.backgroundColor;
|
|
|
|
for (const item of this.activityTopAuthors) {
|
|
const img = new Image();
|
|
img.src = item.avatar_link;
|
|
item.avatar_img = img;
|
|
}
|
|
|
|
Chart.register({
|
|
id: 'image_label',
|
|
afterDraw: (chart) => {
|
|
const xAxis = chart.boxes[0];
|
|
const yAxis = chart.boxes[1];
|
|
for (const [index] of xAxis.ticks.entries()) {
|
|
const x = xAxis.getPixelForTick(index);
|
|
const img = this.activityTopAuthors[index].avatar_img;
|
|
|
|
chart.ctx.save();
|
|
chart.ctx.drawImage(img, 0, 0, img.naturalWidth, img.naturalHeight, x - 10, yAxis.bottom + 10, 20, 20);
|
|
chart.ctx.restore();
|
|
}
|
|
},
|
|
beforeEvent: (chart, args) => {
|
|
const event = args.event;
|
|
if (event.type !== 'mousemove' && event.type !== 'click') return;
|
|
|
|
const yAxis = chart.boxes[1];
|
|
if (event.y < yAxis.bottom + 10 || event.y > yAxis.bottom + 30) {
|
|
chart.canvas.style.cursor = '';
|
|
return;
|
|
}
|
|
|
|
const xAxis = chart.boxes[0];
|
|
const pointIdx = xAxis.ticks.findIndex((_, index) => {
|
|
const x = xAxis.getPixelForTick(index);
|
|
return event.x >= x - 10 && event.x <= x + 10;
|
|
});
|
|
|
|
if (pointIdx === -1) {
|
|
chart.canvas.style.cursor = '';
|
|
return;
|
|
}
|
|
|
|
chart.canvas.style.cursor = 'pointer';
|
|
if (event.type === 'click' && this.activityTopAuthors[pointIdx].home_link) {
|
|
window.location.href = this.activityTopAuthors[pointIdx].home_link;
|
|
}
|
|
},
|
|
});
|
|
},
|
|
};
|
|
|
|
export function initRepoActivityTopAuthorsChart() {
|
|
const el = document.getElementById('repo-activity-top-authors-chart');
|
|
if (el) {
|
|
createApp(sfc, {
|
|
locale: {
|
|
commitActivity: el.getAttribute('data-locale-commit-activity'),
|
|
},
|
|
}).mount(el);
|
|
}
|
|
}
|
|
|
|
export default sfc; // activate the IDE's Vue plugin
|
|
</script>
|
|
<template>
|
|
<div>
|
|
<div class="activity-bar-graph" ref="style" style="width: 0; height: 0;"/>
|
|
<Bar height="150px" :data="graphPoints()" :options="getOptions()"/>
|
|
</div>
|
|
</template>
|