Greasy Fork

Greasy Fork is available in English.

8chan Reports Page Enhancer

Enhances the usability and appearance of the 8chan board moderation reports page with a dynamic, compact grid layout, smaller image thumbnails, combined report labels in a single div, styled Moderate buttons, scrollable post content, and an orange-themed Ban button to open the ban modal.

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。

您需要先安装用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         8chan Reports Page Enhancer
// @namespace    http://tampermonkey.net/
// @version      1.12
// @description  Enhances the usability and appearance of the 8chan board moderation reports page with a dynamic, compact grid layout, smaller image thumbnails, combined report labels in a single div, styled Moderate buttons, scrollable post content, and an orange-themed Ban button to open the ban modal.
// @author       Grok
// @match        *://8chan.moe/openReports.js*
// @grant        GM_addStyle
// @license      MIT
// @run-at       document-start
// ==/UserScript==

(function() {
    'use strict';

    // Inject CSS styles
    GM_addStyle(`
        /* General Layout */
        body {
            font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Arial, sans-serif;
            background-color: #f5f6f5;
            color: #333;
            line-height: 1.4;
            margin: 0;
            font-size: 0.9em;
        }

        .titleWrapper {
            max-width: 100%;
            margin: 15px auto;
            padding: 0 10px;
            box-sizing: border-box;
        }

        .titleFieldset {
            background: #fff;
            border: 1px solid #ddd;
            border-radius: 6px;
            padding: 15px;
            box-shadow: 0 1px 3px rgba(0,0,0,0.05);
        }

        legend {
            font-size: 1.3em;
            font-weight: 600;
            color: #222;
            padding: 0 8px;
        }

        /* Report Cells */
        #reportDiv {
            display: flex !important;
            flex-wrap: wrap !important;
            gap: 10px;
            margin-top: 15px;
            justify-content: space-between;
            width: 100%;
            box-sizing: border-box;
        }

        .reportCell {
            background: #fff;
            border: 1px solid #e0e0e0;
            border-radius: 5px;
            padding: 10px;
            transition: box-shadow 0.2s;
            flex: 1 1 calc(50% - 8px);
            box-sizing: border-box;
            min-width: 250px;
            font-size: 0.85em;
            display: flex;
            flex-direction: column;
        }

        .reportCell:hover {
            box-shadow: 0 3px 6px rgba(0,0,0,0.1);
        }

        /* Combined Div */
        .reportCell .combined-div {
            display: flex;
            flex-wrap: nowrap;
            gap: 12px;
            align-items: center;
            font-size: 0.9em;
            margin-bottom: 5px;
            white-space: nowrap;
        }

        .reportCell .combined-div span {
            display: inline-flex;
            align-items: center;
        }

        /* Hide Reports and Category labels as fallback */
        .reportCell label:has(.totalLabel),
        .reportCell label.categoryDiv {
            display: none !important;
        }

        .boardLabel, .totalLabel, .categoryLabel {
            font-weight: 500;
            color: #1a73e8;
        }

        .reasonLabel {
            background: #f1f3f4;
            padding: 3px 6px;
            border-radius: 3px;
            display: inline-block;
            font-size: 0.9em;
        }

        /* Checkbox Styles */
        .closureCheckbox {
            margin-right: 6px;
            vertical-align: middle;
            width: 16px;
            height: 16px;
            cursor: pointer;
            accent-color: #1a73e8;
            box-shadow: 0 0 6px rgba(26, 115, 232, 0.6);
        }

        .deletionCheckBox {
            margin-right: 6px;
            vertical-align: middle;
            width: 14px;
            height: 14px;
            cursor: pointer;
            accent-color: #d32f2f;
        }

        /* Moderate and Ban Buttons */
        .reportCell label {
            display: flex;
            align-items: center;
            gap: 8px;
            flex-wrap: wrap;
        }

        .link {
            color: #d32f2f;
            font-weight: 500;
            text-decoration: none;
            padding: 3px 6px;
            border-radius: 3px;
            font-size: 0.9em;
            background-color: #f0e0e0;
            border: 1px solid #e03030;
            transition: background-color 0.2s, border-color 0.2s;
        }

        .banButton {
            color: #f57c00;
            font-weight: 500;
            text-decoration: none;
            padding: 3px 6px;
            border-radius: 3px;
            font-size: 0.9em;
            background-color: #ffe0b2;
            border: 1px solid #f57c00;
            transition: background-color 0.2s, border-color 0.2s;
        }

        .link:hover, .banButton:hover {
            background-color: #fff3e0;
            border-color: #888;
            text-decoration: none;
        }

        /* Post Content */
        .postingDiv {
            flex: 1;
            min-width: 0;
            max-height: 25rem;
            overflow-y: auto;
        }

        .postingDiv .innerPost {
            background: #fafafa;
            padding: 8px;
            border-radius: 3px;
            margin-top: 8px;
            font-size: 0.9em;
            min-width: 150px;
            padding-bottom: 12px;
        }

        .labelBoard {
            font-size: 1em;
            color: #0288d1;
            margin: 0 0 8px;
        }

        .postInfo {
            font-size: 0.85em;
            color: #555;
            white-space: normal;
        }

        .labelId {
            padding: 2px 5px;
            border-radius: 3px;
            color: #fff;
            font-size: 0.85em;
        }

        .panelIp, .panelASN, .panelBypassId {
            font-size: 0.8em;
            color: #666;
            margin-top: 4px;
            word-break: break-all;
        }

        .divMessage {
            margin-top: 8px;
            font-size: 0.9em;
            color: #333;
            padding: 6px;
            background: #f5f5f5;
            border-radius: 3px;
            min-width: 150px;
            word-wrap: break-word;
        }

        .quoteLink {
            color: #388e3c;
            text-decoration: none;
            font-weight: 500;
            font-size: 0.9em;
        }

        .quoteLink:hover {
            text-decoration: underline;
        }

        /* Uploads */
        .panelUploads {
            margin-top: 8px;
            display: flex;
            flex-wrap: wrap;
        }

        .uploadCell {
            margin-bottom: 8px;
            flex: 0 0 auto;
        }

        .imgLink img {
            border-radius: 3px;
            max-width: 60px !important;
            max-height: 60px !important;
            width: auto !important;
            height: auto !important;
            object-fit: contain;
            display: block;
            margin: 0;
        }

        .originalNameLink, .nameLink {
            color: #0288d1;
            text-decoration: none;
            font-size: 0.85em;
        }

        .originalNameLink:hover, .nameLink:hover {
            text-decoration: underline;
        }

        .uploadDetails, .divHash, .sizeLabel, .dimensionLabel {
            font-size: 0.8em;
            color: #555;
        }

        .unlinkLink, .unlinkAndDeleteLink {
            font-size: 0.85em;
        }

        /* Forms */
        #filterForm, form[action="/closeReports.js"] {
            background: #f9f9f9;
            padding: 10px;
            border-radius: 5px;
            margin-bottom: 15px;
        }

        #filterForm label, form[action="/closeReports.js"] label {
            display: block;
            margin-bottom: 8px;
            font-weight: 500;
            color: #444;
            font-size: 0.9em;
        }

        #filterCategoriesDiv {
            margin: 8px 0;
        }

        .categoryCheckbox {
            margin-right: 6px;
            width: 14px;
            height: 14px;
        }

        input[type="text"], select {
            padding: 6px;
            border: 1px solid #ccc;
            border-radius: 3px;
            font-size: 0.9em;
            width: 180px;
            margin-left: 8px;
        }

        input[type="text"]:focus, select:focus {
            border-color: #1a73e8;
            outline: none;
            box-shadow: 0 0 0 2px rgba(26,115,232,0.2);
        }

        button, #filterSubmitButton, #closeReportsFormButton {
            background: #1a73e8;
            color: #fff;
            border: none;
            padding: 6px 12px;
            border-radius: 3px;
            font-size: 0.9em;
            cursor: pointer;
            transition: background 0.2s;
        }

        button:hover, #filterSubmitButton:hover, #closeReportsFormButton:hover {
            background: #1565c0;
        }

        /* Checkboxes */
        .closeReportsField {
            margin-right: 6px;
            vertical-align: middle;
            width: 14px;
            height: 14px;
        }

        /* Navigation */
        .navHeader {
            background: #fff;
            border-bottom: 1px solid #ddd;
            padding: 8px 15px;
            position: sticky;
            top: 0;
            z-index: 1000;
        }

        .navLinkSpan a {
            color: #0288d1;
            text-decoration: none;
            margin: 0 4px;
            font-size: 0.9em;
        }

        .navLinkSpan a:hover {
            text-decoration: underline;
        }

        /* Responsive Adjustments for More Columns */
        @media (min-width: 900px) {
            .reportCell {
                flex: 1 1 calc(33.33% - 8px);
            }
        }

        @media (min-width: 1200px) {
            .reportCell {
                flex: 1 1 calc(25% - 8px);
            }
        }

        @media (max-width: 600px) {
            .reportCell {
                flex: 1 1 100% !important;
                min-width: 100% !important;
                max-width: 100% !important;
            }

            input[type="text"], select {
                width: 100%;
            }

            .divMessage, .innerPost {
                min-width: 100% !important;
            }

            .reportCell .combined-div {
                flex-direction: column;
                gap: 5px;
                align-items: flex-start;
                white-space: normal;
            }

            .reportCell label {
                flex-direction: column;
                align-items: flex-start;
                gap: 4px;
            }
        }
    `);

    // JavaScript to combine labels and add Ban button
    function processReportCells() {
        const cells = document.querySelectorAll('.reportCell:not(.processed)');
        if (cells.length) {
            console.log(`Processing ${cells.length} reportCell(s) at`, Date.now());
        }
        cells.forEach(cell => {
            // Mark as processed
            cell.classList.add('processed');

            // Combine labels into a single div
            const labels = cell.querySelectorAll('label:not([for])');
            if (labels.length >= 3) {
                const boardSpan = labels[0].querySelector('.boardLabel');
                const totalSpan = labels[1].querySelector('.totalLabel');
                const categorySpan = labels[2].querySelector('.categoryLabel');

                if (boardSpan && totalSpan && categorySpan) {
                    const combinedDiv = document.createElement('div');
                    combinedDiv.className = 'combined-div';
                    combinedDiv.innerHTML = `
                        <span class="boardLabel">/${boardSpan.textContent}/</span>
                        Reports: <span class="totalLabel">${totalSpan.textContent}</span>
                        Category: <span class="categoryLabel">${categorySpan.textContent}</span>
                    `;
                    cell.insertBefore(combinedDiv, labels[0]);
                    labels[0].remove();
                    labels[1].remove();
                    labels[2].remove();
                }
            }

            // Add Ban button
            const reportLabel = cell.querySelector('label');
            const deletionCheckBox = cell.querySelector('.deletionCheckBox');
            if (reportLabel && deletionCheckBox) {
                const banButton = document.createElement('a');
                banButton.className = 'banButton';
                banButton.textContent = 'Ban';
                banButton.href = '#';
                banButton.setAttribute('data-ban-id', deletionCheckBox.name);
                banButton.setAttribute('aria-label', `Ban post ${deletionCheckBox.name}`);
                banButton.addEventListener('click', (e) => {
                    e.preventDefault();
                    e.stopPropagation(); // Prevent bubbling to label/checkbox
                    console.log('Ban button clicked for post:', deletionCheckBox.name);

                    // Find extraMenuButton
                    const extraMenuButton = cell.querySelector('.extraMenuButton');
                    if (extraMenuButton) {
                        console.log('Clicking extraMenuButton for:', deletionCheckBox.name);
                        extraMenuButton.click();

                        // Wait for extraMenu to appear
                        setTimeout(() => {
                            const extraMenu = document.querySelector('.extraMenu');
                            if (extraMenu) {
                                const banOption = Array.from(extraMenu.querySelectorAll('li')).find(li => li.textContent === 'Ban');
                                if (banOption) {
                                    console.log('Found Ban option, triggering click for:', deletionCheckBox.name);
                                    const originalDisplay = extraMenu.style.display;
                                    extraMenu.style.display = 'block';
                                    banOption.click();
                                    setTimeout(() => {
                                        extraMenu.style.display = originalDisplay;
                                    }, 0);
                                } else {
                                    console.error('Ban option not found in .extraMenu');
                                }
                            } else {
                                console.error('extraMenu not found after clicking extraMenuButton');
                            }
                        }, 50); // Wait 50ms for extraMenu to appear
                    } else {
                        console.error('extraMenuButton not found in reportCell');
                    }
                });
                reportLabel.appendChild(banButton);
                console.log('Added Ban button for post:', deletionCheckBox.name);
            }
        });
    }

    // Debounce function to batch processReportCells calls
    function debounce(fn, wait) {
        let timeout;
        return function(...args) {
            clearTimeout(timeout);
            timeout = setTimeout(() => fn(...args), wait);
        };
    }

    // Process cells with debounce
    const debouncedProcessReportCells = debounce(processReportCells, 100);

    // Observe #reportDiv for .reportCell additions
    function observeReportDiv() {
        const reportDiv = document.querySelector('#reportDiv');
        if (reportDiv) {
            console.log('Found #reportDiv, starting observer at', Date.now());
            const observer = new MutationObserver((mutations) => {
                if (mutations.some(m => Array.from(m.addedNodes).some(node => node.nodeType === 1 && node.matches('.reportCell')))) {
                    debouncedProcessReportCells();
                }
            });
            observer.observe(reportDiv, { childList: true, subtree: true });
            // Check existing cells immediately
            processReportCells();
        } else {
            // Retry if #reportDiv not found
            setTimeout(observeReportDiv, 100);
        }
    }

    // Start observing as early as possible
    observeReportDiv();
})();