Candidate Vote Bar Chart

Adds a bar chart after the candidate table (for election commissions)

// ==UserScript==
// @name         Candidate Vote Bar Chart
// @namespace    http://tampermonkey.net/
// @version      1.0.2
// @description  Adds a bar chart after the candidate table (for election commissions)
// @author       Kasden45
// @match        https://wybory.gov.pl/prezydent2025/pl/obkw/*
// @icon         
// @grant        none
// @run-at       document-idle
// @license      MIT
// ==/UserScript==
(function() {
    'use strict';

    console.log('Script working')
function extractCandidateData(table) {
    const candidates = [];
    const rows = table.querySelectorAll('tbody tr');
    const totalVotes = parseInt(table.querySelector('tfoot td:last-child').textContent.replace(/\s+/g, ''), 10);

    rows.forEach(row => {
        const cells = row.querySelectorAll('td');
        const name = cells[1].querySelector('a').textContent.trim();
        const votes = parseInt(cells[2].textContent.trim(), 10);
        const percent = (votes / totalVotes) * 100;

        candidates.push({
            name: name,
            votes: votes,
            percent: percent
        });
    });

    return candidates.sort((a, b) => b.votes - a.votes);
}

// Create a bar chart
function createBarChart(candidateData) {
    const container = document.createElement('div');
    container.className = 'bar-chart-container';
    container.style.margin = '20px 0';
    container.style.width = '100%';

    const title = document.createElement('h4');
    title.textContent = 'Rozkład głosów kandydatów na Prezydenta RP';
    title.style.marginBottom = '15px';
    title.style.color = '#9d032a';
    container.appendChild(title);

    const totalVotes = candidateData.reduce((sum, candidate) => sum + candidate.votes, 0);
    const totalInfo = document.createElement('div');
    totalInfo.textContent = `Łączna liczba głosów: ${totalVotes.toLocaleString()}`;
    totalInfo.style.marginBottom = '15px';
    totalInfo.style.fontWeight = 'bold';
    container.appendChild(totalInfo);

    const chart = document.createElement('div');
    chart.className = 'vote-bars';
    chart.style.display = 'flex';
    chart.style.flexDirection = 'column';
    chart.style.gap = '8px';

    const maxVotes = Math.max(...candidateData.map(c => c.votes));

    candidateData.forEach(candidate => {
        const barContainer = document.createElement('div');
        barContainer.style.display = 'flex';
        barContainer.style.alignItems = 'center';
        barContainer.style.gap = '10px';

        const name = document.createElement('div');
        name.textContent = candidate.name;
        name.style.width = '250px';
        name.style.minWidth = '250px';
        name.style.fontWeight = 'bold';

        // Bar container for progress and label
        const barWrapper = document.createElement('div');
        barWrapper.style.display = 'flex';
        barWrapper.style.alignItems = 'center';
        barWrapper.style.gap = '10px';
        barWrapper.style.flexGrow = '1';

        // Progress bar
        const progress = document.createElement('progress');
        progress.value = candidate.percent;
        progress.max = 100;
        progress.style.width = '100%';
        progress.style.height = '25px';
        progress.style.accentColor = '#9d032a'; // Modern browsers support this

        // Label inside the bar
        const label = document.createElement('span');
        label.textContent = `${candidate.percent.toFixed(1)}% (${candidate.votes.toLocaleString()})`;
        label.style.marginLeft = '10px';
        label.style.whiteSpace = 'nowrap';
        label.style.fontWeight = 'bold';
        label.style.color = '#9d032a';

        barWrapper.appendChild(progress);
        barWrapper.appendChild(label);


        barContainer.appendChild(name);
        barContainer.appendChild(barWrapper);
        chart.appendChild(barContainer);
    });

    container.appendChild(chart);
    return container;
}

function addChartToPage() {
        const headers = document.querySelectorAll('h4');
        for (const header of headers) {
            if (/^15\.\s*/.test(header.textContent.trim())) {
                const next = header.nextElementSibling;
                if (next && next.matches('table.table.table-striped')) {
                    const candidateData = extractCandidateData(next);
                    const chart = createBarChart(candidateData);
                    next.parentNode.insertBefore(chart, next.nextSibling);
                    return true;
                }
            }
        }
        return false;
    }

    // Keep checking every 5 seconds until the header appears
    const interval = setInterval(() => {
        const success = addChartToPage();
        if (success) {
            clearInterval(interval); // Stop checking once found and inserted
        }
    }, 500);
    // Your code here...
})();
长期地址
遇到问题?请前往 GitHub 提 Issues,或加Q群1031348184

赞助商

Fishcpy

广告

Rainyun

注册一下就行

Rainyun

一年攒够 12 元