const CHART_TARGET_SIZE = 200 const projectsCanvas = document.getElementById('chart-projects') const osCanvas = document.getElementById('chart-os') const editorsCanvas = document.getElementById('chart-editor') const languagesCanvas = document.getElementById('chart-language') const machinesCanvas = document.getElementById('chart-machine') const projectContainer = document.getElementById('project-container') const osContainer = document.getElementById('os-container') const editorContainer = document.getElementById('editor-container') const languageContainer = document.getElementById('language-container') const machineContainer = document.getElementById('machine-container') const containers = [projectContainer, osContainer, editorContainer, languageContainer, machineContainer] const canvases = [projectsCanvas, osCanvas, editorsCanvas, languagesCanvas, machinesCanvas] const data = [wakapiData.projects, wakapiData.operatingSystems, wakapiData.editors, wakapiData.languages, wakapiData.machines] let topNPickers = [...document.getElementsByClassName('top-picker')] topNPickers.sort(((a, b) => parseInt(a.attributes['data-entity'].value) - parseInt(b.attributes['data-entity'].value))) topNPickers.forEach(e => { const idx = parseInt(e.attributes['data-entity'].value) e.max = Math.min(data[idx].length, 10) e.value = e.max }) let charts = [] let showTopN = [] let resizeCount = 0 Chart.defaults.global.defaultFontColor = "#E2E8F0" Chart.defaults.global.defaultColor = "#E2E8F0" String.prototype.toHHMMSS = function () { var sec_num = parseInt(this, 10) var hours = Math.floor(sec_num / 3600) var minutes = Math.floor((sec_num - (hours * 3600)) / 60) var seconds = sec_num - (hours * 3600) - (minutes * 60) if (hours < 10) { hours = '0' + hours } if (minutes < 10) { minutes = '0' + minutes } if (seconds < 10) { seconds = '0' + seconds } return hours + ':' + minutes + ':' + seconds } String.prototype.toHHMM = function () { const sec_num = parseInt(this, 10) const hours = Math.floor(sec_num / 3600) const minutes = Math.floor((sec_num - (hours * 3600)) / 60) return hours + ':' + minutes } function draw(subselection) { function getTooltipOptions(key) { return { mode: 'single', callbacks: { label: (item) => { let d = wakapiData[key][item.index] return `${d.key}: ${d.total.toString().toHHMMSS()}` }, title: () => 'Total Time' } } } function shouldUpdate(index) { return !subselection || (subselection.includes(index) && data[index].length >= showTopN[index]) } charts .filter((c, i) => shouldUpdate(i)) .forEach(c => c.destroy()) let projectChart = !projectsCanvas.classList.contains('hidden') && shouldUpdate(0) ? new Chart(projectsCanvas.getContext('2d'), { type: 'horizontalBar', data: { datasets: [{ data: wakapiData.projects .slice(0, Math.min(showTopN[0], wakapiData.projects.length)) .map(p => parseInt(p.total)), backgroundColor: wakapiData.projects.map(p => { const c = hexToRgb(getRandomColor(p.key)) return `rgba(${c.r}, ${c.g}, ${c.b}, 0.6)` }), hoverBackgroundColor: wakapiData.projects.map(p => { const c = hexToRgb(getRandomColor(p.key)) return `rgba(${c.r}, ${c.g}, ${c.b}, 0.8)` }), borderColor: wakapiData.projects.map(p => { const c = hexToRgb(getRandomColor(p.key)) return `rgba(${c.r}, ${c.g}, ${c.b}, 1)` }), borderWidth: 2 }], labels: wakapiData.projects .slice(0, Math.min(showTopN[0], wakapiData.projects.length)) .map(p => p.key) }, options: { legend: { display: false }, scales: { xAxes: [{ scaleLabel: { display: true, labelString: 'Minutes' } }] }, tooltips: getTooltipOptions('projects'), maintainAspectRatio: false, onResize: onChartResize } }) : null let osChart = !osCanvas.classList.contains('hidden') && shouldUpdate(1) ? new Chart(osCanvas.getContext('2d'), { type: 'pie', data: { datasets: [{ data: wakapiData.operatingSystems .slice(0, Math.min(showTopN[1], wakapiData.operatingSystems.length)) .map(p => parseInt(p.total)), backgroundColor: wakapiData.operatingSystems.map(p => { const c = hexToRgb(osColors[p.key.toLowerCase()] || getRandomColor(p.key)) return `rgba(${c.r}, ${c.g}, ${c.b}, 0.6)` }), hoverBackgroundColor: wakapiData.operatingSystems.map(p => { const c = hexToRgb(osColors[p.key.toLowerCase()] || getRandomColor(p.key)) return `rgba(${c.r}, ${c.g}, ${c.b}, 0.8)` }), borderColor: wakapiData.operatingSystems.map(p => { const c = hexToRgb(osColors[p.key.toLowerCase()] || getRandomColor(p.key)) return `rgba(${c.r}, ${c.g}, ${c.b}, 1)` }), }], labels: wakapiData.operatingSystems .slice(0, Math.min(showTopN[1], wakapiData.operatingSystems.length)) .map(p => p.key) }, options: { tooltips: getTooltipOptions('operatingSystems'), maintainAspectRatio: false, onResize: onChartResize } }) : null let editorChart = !editorsCanvas.classList.contains('hidden') && shouldUpdate(2) ? new Chart(editorsCanvas.getContext('2d'), { type: 'pie', data: { datasets: [{ data: wakapiData.editors .slice(0, Math.min(showTopN[2], wakapiData.editors.length)) .map(p => parseInt(p.total)), backgroundColor: wakapiData.editors.map(p => { const c = hexToRgb(editorColors[p.key.toLowerCase()] || getRandomColor(p.key)) return `rgba(${c.r}, ${c.g}, ${c.b}, 0.6)` }), hoverBackgroundColor: wakapiData.editors.map(p => { const c = hexToRgb(editorColors[p.key.toLowerCase()] || getRandomColor(p.key)) return `rgba(${c.r}, ${c.g}, ${c.b}, 0.8)` }), borderColor: wakapiData.editors.map(p => { const c = hexToRgb(editorColors[p.key.toLowerCase()] || getRandomColor(p.key)) return `rgba(${c.r}, ${c.g}, ${c.b}, 1)` }), }], labels: wakapiData.editors .slice(0, Math.min(showTopN[2], wakapiData.editors.length)) .map(p => p.key) }, options: { tooltips: getTooltipOptions('editors'), maintainAspectRatio: false, onResize: onChartResize } }) : null let languageChart = !languagesCanvas.classList.contains('hidden') && shouldUpdate(3) ? new Chart(languagesCanvas.getContext('2d'), { type: 'pie', data: { datasets: [{ data: wakapiData.languages .slice(0, Math.min(showTopN[3], wakapiData.languages.length)) .map(p => parseInt(p.total)), backgroundColor: wakapiData.languages.map(p => { const c = hexToRgb(languageColors[p.key.toLowerCase()] || getRandomColor(p.key)) return `rgba(${c.r}, ${c.g}, ${c.b}, 0.6)` }), hoverBackgroundColor: wakapiData.languages.map(p => { const c = hexToRgb(languageColors[p.key.toLowerCase()] || getRandomColor(p.key)) return `rgba(${c.r}, ${c.g}, ${c.b}, 0.8)` }), borderColor: wakapiData.languages.map(p => { const c = hexToRgb(languageColors[p.key.toLowerCase()] || getRandomColor(p.key)) return `rgba(${c.r}, ${c.g}, ${c.b}, 1)` }), }], labels: wakapiData.languages .slice(0, Math.min(showTopN[3], wakapiData.languages.length)) .map(p => p.key) }, options: { tooltips: getTooltipOptions('languages'), maintainAspectRatio: false, onResize: onChartResize } }) : null let machineChart = !machinesCanvas.classList.contains('hidden') && shouldUpdate(4) ? new Chart(machinesCanvas.getContext('2d'), { type: 'pie', data: { datasets: [{ data: wakapiData.machines .slice(0, Math.min(showTopN[4], wakapiData.machines.length)) .map(p => parseInt(p.total)), backgroundColor: wakapiData.machines.map(p => { const c = hexToRgb(getRandomColor(p.key)) return `rgba(${c.r}, ${c.g}, ${c.b}, 0.6)` }), hoverBackgroundColor: wakapiData.machines.map(p => { const c = hexToRgb(getRandomColor(p.key)) return `rgba(${c.r}, ${c.g}, ${c.b}, 0.8)` }), borderColor: wakapiData.machines.map(p => { const c = hexToRgb(getRandomColor(p.key)) return `rgba(${c.r}, ${c.g}, ${c.b}, 1)` }), }], labels: wakapiData.machines .slice(0, Math.min(showTopN[4], wakapiData.machines.length)) .map(p => p.key) }, options: { tooltips: getTooltipOptions('machines'), maintainAspectRatio: false, onResize: onChartResize } }) : null getTotal(wakapiData.operatingSystems) charts = [projectChart, osChart, editorChart, languageChart, machineChart].filter(c => !!c) if (!subselection) { charts.forEach(c => c.options.onResize(c.chart)) equalizeHeights() } } function parseTopN() { showTopN = topNPickers.map(e => parseInt(e.value)) } function togglePlaceholders(mask) { const placeholderElements = containers.map(c => c.querySelector('.placeholder-container')) for (let i = 0; i < mask.length; i++) { if (!mask[i]) { canvases[i].classList.add('hidden') placeholderElements[i].classList.remove('hidden') } else { canvases[i].classList.remove('hidden') placeholderElements[i].classList.add('hidden') } } } function getPresentDataMask() { return data.map(list => (list ? list.reduce((acc, e) => acc + e.total, 0) : 0) > 0) } function getContainer(chart) { return chart.canvas.parentNode } function onChartResize(chart) { let container = getContainer(chart) let targetHeight = Math.min(chart.width, CHART_TARGET_SIZE) let actualHeight = chart.height - chart.chartArea.top let containerTargetHeight = container.clientHeight += (targetHeight - actualHeight) container.style.height = parseInt(containerTargetHeight) + 'px' resizeCount++ watchEqualize() } function watchEqualize() { if (resizeCount === charts.length) { equalizeHeights() resizeCount = 0 } } function equalizeHeights() { let maxHeight = 0 charts.forEach(c => { let container = getContainer(c) if (maxHeight < container.clientHeight) { maxHeight = container.clientHeight } }) charts.forEach(c => { let container = getContainer(c) container.style.height = parseInt(maxHeight) + 'px' }) } function getTotal(items) { const el = document.getElementById('total-span') if (!el) return const total = items.reduce((acc, d) => acc + d.total, 0) const formatted = total.toString().toHHMM() el.innerText = `${formatted.split(':')[0]} hours, ${formatted.split(':')[1]} minutes` } function getRandomColor(seed) { seed = seed ? seed : '1234567' Math.seedrandom(seed) var letters = '0123456789ABCDEF'.split('') var color = '#' for (var i = 0; i < 6; i++) { color += letters[Math.floor(Math.random() * 16)] } return color } // https://stackoverflow.com/a/5624139/3112139 function hexToRgb(hex) { var shorthandRegex = /^#?([a-f\d])([a-f\d])([a-f\d])$/i; hex = hex.replace(shorthandRegex, function(m, r, g, b) { return r + r + g + g + b + b; }); var result = /^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i.exec(hex); return result ? { r: parseInt(result[1], 16), g: parseInt(result[2], 16), b: parseInt(result[3], 16) } : null; } function showApiKeyPopup(event) { const el = document.getElementById('api-key-popup') el.classList.remove('hidden') el.classList.add('block') event.stopPropagation() } function copyApiKey(event) { const el = document.getElementById('api-key-container') el.select() el.setSelectionRange(0, 9999) document.execCommand('copy') event.stopPropagation() } // https://koddsson.com/posts/emoji-favicon/ const favicon = document.querySelector('link[rel=icon]') if (favicon) { const emoji = favicon.getAttribute('data-emoji') if (emoji) { const canvas = document.createElement('canvas') canvas.height = 64 canvas.width = 64 const ctx = canvas.getContext('2d') ctx.font = '64px serif' ctx.fillText(emoji, 0, 64) favicon.href = canvas.toDataURL() } } // Click outside window.addEventListener('click', function (event) { if (event.target.classList.contains('popup')) { return } document.querySelectorAll('.popup').forEach(el => { el.classList.remove('block') el.classList.add('hidden') }) }) window.addEventListener('load', function () { topNPickers.forEach(e => e.addEventListener('change', () => { parseTopN() draw([parseInt(e.attributes['data-entity'].value)]) })) parseTopN() togglePlaceholders(getPresentDataMask()) draw() })