您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
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... })();