Greasy Fork

来自缓存

Greasy Fork is available in English.

Yamibo漫区封面卡片

Yamibo漫画区帖子列表实现卡片风格瀑布流展示

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

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

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

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

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

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

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

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

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

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

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

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

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

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

// ==UserScript==
// @name         Yamibo漫区封面卡片
// @namespace    https://bbs.yamibo.com/
// @version      1.1
// @description  Yamibo漫画区帖子列表实现卡片风格瀑布流展示
// @author       hitori酱
// @match        https://bbs.yamibo.com/forum.php*fid=30*
// @match        https://bbs.yamibo.com/forum-30-*
// @match        https://bbs.yamibo.com/forum-37-*
// @icon         https://www.yamibo.com/favicon.ico
// @grant        none
// @run-at       document-end
// @noframes
// @license      MIT License
// @updateURL
// @downloadURL
// ==/UserScript==

(function() {
    'use strict';

    const DOMAIN = 'https://bbs.yamibo.com/';
    const PLACEHOLDER = 'https://s2.loli.net/2025/11/10/nTuPOVqdFQHLosz.jpg';

    // Base64 SVG 图标
    const ICONS = {
        view: '',
        reply: '',
        favorite: ''
    };

    function setRealImage(card, imgUrl) {
        const img = card.querySelector('img');
        if (img) {
            img.src = imgUrl;
            img.removeAttribute('data-src');
        }
    }

    function createCard(threadData) {
        const { title, url, category, heat, readPerm, views, replies, favorite } = threadData;
        const card = document.createElement('div');
        card.className = 'comic-card';

        // 图片
        const imgWrapper = document.createElement('div');
        imgWrapper.className = 'img-wrapper';
        const img = document.createElement('img');
        img.src = PLACEHOLDER;
        img.alt = title;
        imgWrapper.appendChild(img);

        // 浮层
        const tagCategory = document.createElement('div');
        tagCategory.className = 'tag-category';
        tagCategory.innerText = category;
        imgWrapper.appendChild(tagCategory);

        // 热门标签
        if (heat && (heat.includes('热') || heat.includes('火'))) {
            const hotTag = document.createElement('div');
            hotTag.className = 'tag-hot';
            hotTag.innerText = '火';
            imgWrapper.appendChild(hotTag);
        }

        const tagRead = document.createElement('div');
        tagRead.className = 'tag-readperm';
        tagRead.innerText = readPerm ? `[阅读权限 ${readPerm}]` : '';
        imgWrapper.appendChild(tagRead);

        card.appendChild(imgWrapper);

        // 数据行
        const dataRow = document.createElement('div');
        dataRow.className = 'data-row';

        const viewDiv = document.createElement('div');
        viewDiv.innerHTML = `<img src="${ICONS.view}" class="icon"> ${views||0}`;
        dataRow.appendChild(viewDiv);

        const replyDiv = document.createElement('div');
        replyDiv.innerHTML = `<img src="${ICONS.reply}" class="icon"> ${replies||0}`;
        dataRow.appendChild(replyDiv);

        const favDiv = document.createElement('div');
        favDiv.innerHTML = `<img src="${ICONS.favorite}" class="icon fav-icon"> ${favorite||0}`;
        dataRow.appendChild(favDiv);

        card.appendChild(dataRow);

        // 标题
        const titleDiv = document.createElement('div');
        titleDiv.className = 'title';
        titleDiv.innerText = title;
        card.appendChild(titleDiv);

        card.onclick = () => window.open(url, '_blank');
        return card;
    }

    function initGrid() {
        const threadList = document.querySelector('#threadlist');
        if (!threadList) return;

        // 默认收起标签区域
        const threadTypes = document.querySelector('#thread_types');
        if (threadTypes) {
            threadTypes.style.height = '';
            const foldLi = threadTypes.querySelector('li.fold');
            if (foldLi && foldLi.onclick) {
                foldLi.onclick();
            }
        }

        const oldTable = document.querySelector('#threadlisttableid');
        if (oldTable) oldTable.style.display = 'none';

        const stickyThreads = document.querySelectorAll('tbody[id^="stickthread_"]');
        stickyThreads.forEach(tbody => tbody.style.display = 'none');

        const nextPageBtn = document.querySelector('#autopbn');
        if (nextPageBtn) nextPageBtn.style.display = 'none';

        // 隐藏列表标题框右侧的表头文字标签
        const threadHeader = document.querySelector('#threadlist .th');
        if (threadHeader) {
            const headerCells = threadHeader.querySelectorAll('td, th');
            headerCells.forEach((cell, index) => {
                // 隐藏作者、回复/查看、最后发表等标签列
                if (index > 0) { // 保留第一列(主题标题),隐藏其他列
                    cell.style.display = 'none';
                }
            });
        }

        const grid = document.createElement('div');
        grid.id = 'comic-grid';
        threadList.appendChild(grid);

        const style = document.createElement('style');
        style.innerHTML = `
#comic-grid {
    display: flex;
    flex-wrap: wrap;
    gap: 15px;
    justify-content: center;
    padding: 10px;
}
.comic-card {
    width: calc(16.66% - 15px);
    display: flex;
    flex-direction: column;
    background: #fff;
    border-radius: 8px;
    box-shadow: 0 4px 12px rgba(0,0,0,0.1);
    overflow: visible;
    cursor: pointer;
    transition: transform 0.3s, box-shadow 0.3s;
}
.comic-card:hover {
    transform: translateY(-6px);
    box-shadow: 0 8px 20px rgba(0,0,0,0.2);
}
.img-wrapper {
    position: relative;
    width: 100%;
    aspect-ratio: 4/5;
    background: #f8f8f8;
    overflow: hidden;
    flex-shrink: 0;
}
.img-wrapper img {
    width: 100%;
    height: 100%;
    object-fit: cover;
    object-position: top;
    display: block;
    border-radius: 8px 8px 0 0;
}
.tag-category {
    position: absolute;
    top: 5px; left: 5px;
    background: rgba(153,153,153,0.6);
    padding: 2px 5px;
    border-radius: 3px;
    font-size: 12px;
    z-index: 10;
    color: #a0522d;
}
.tag-hot {
    position: absolute;
    top: 5px; right: 5px;
    background: rgba(255, 107, 53, 0.9);
    color: #fff;
    padding: 2px 5px;
    border-radius: 3px;
    font-size: 12px;
    z-index: 10;
}

.tag-readperm {
    position: absolute;
    bottom: 5px; right: 5px;
    background: rgba(153,153,153,0.6);
    color: #a0522d;
    padding: 1px 6px;
    border-radius: 3px;
    font-size: 12px;
    z-index: 10;
}
.data-row {
    display: flex;
    justify-content: space-around;
    padding: 5px 0;
    font-size: 12px;
    color: #666;
}
.data-row .icon {
    width: 16px;
    height: 16px;
    vertical-align: middle;
    margin-right: 3px;
}
.data-row .fav-icon {
    width: 16px;
    height: 16px;
    vertical-align: middle;
    margin-right: 4px;
    object-fit: contain;
}
.title {
    padding: 8px;
    font-size: 15px;
    line-height: 1.4;
    text-align: left;
    word-break: break-word;
    overflow-wrap: break-word;
    min-height: 1.4em;
    overflow: visible;
    display: block;
    flex: 1 1 auto;
}
        `;
        document.head.appendChild(style);

        // 等高逻辑:基于标题实际需要高度计算
        function equalizeRowHeights() {
            const cards = Array.from(grid.children);
            let rows = {};
            cards.forEach(card => {
                card.style.height = '';
                const top = card.offsetTop;
                if (!rows[top]) rows[top] = [];
                rows[top].push(card);
            });
            Object.values(rows).forEach(rowCards => {
                // 计算固定部分高度:图片 + 数据行
                const fixedHeight = rowCards[0].querySelector('.img-wrapper').offsetHeight +
                                  rowCards[0].querySelector('.data-row').offsetHeight;

                // 找出本行标题需要的最大高度
                const maxTitleHeight = Math.max(...rowCards.map(c => {
                    const title = c.querySelector('.title');
                    return title.scrollHeight + 16; // 16px是padding
                }));

                const finalHeight = fixedHeight + maxTitleHeight;
                rowCards.forEach(c => {
                    c.style.height = finalHeight + 'px';
                });
            });
        }

        loadThreads(grid, equalizeRowHeights);

        window.addEventListener('resize', () => setTimeout(equalizeRowHeights, 300));
    }

    function getThreads() {
        return document.querySelectorAll('tbody[id^="normalthread_"]');
    }

    function findFirstImage(doc) {
        let img;
        
        // 优先查找 ignore_js_op 内的图片(通常是漫画内容)
        img = doc.querySelector('ignore_js_op img');
        if (img && (img.getAttribute('file') || img.getAttribute('zoomfile'))) return img;
        
        // 查找带有 aid 属性的附件图片(支持多种容器)
        img = doc.querySelector('img[aid]');
        if (img && (img.getAttribute('file') || img.getAttribute('zoomfile'))) {
            return img; // 有file或zoomfile属性的aid图片直接返回
        }
        
        // 查找 zoom 类的图片
        img = doc.querySelector('img.zoom');
        if (img && (img.getAttribute('file') || img.getAttribute('zoomfile'))) return img;
        
        // 在 postmessage 容器中查找图片
        img = doc.querySelector('.postmessage img');
        if (img && (img.getAttribute('src') || img.getAttribute('file') || img.getAttribute('zoomfile'))) {
            const src = img.getAttribute('src') || '';
            // 过滤表情图片和占位图片
            if (!src.includes('static/image/smiley/') && !src.includes('static/image/common/none.gif')) {
                return img;
            }
        }
        
        // 最后尝试第一个帖子内容区的图片,但要过滤表情图片
        img = doc.querySelector('td.t_f img');
        if (img && (img.getAttribute('src') || img.getAttribute('file') || img.getAttribute('zoomfile'))) {
            const src = img.getAttribute('src') || '';
            // 过滤表情图片路径
            if (!src.includes('static/image/smiley/') && !src.includes('static/image/common/none.gif')) {
                return img;
            }
        }



        // 尝试后续楼层的内容(第2-5楼)- 支持多种容器格式
        const postContainers = doc.querySelectorAll('.postmessage, td.t_f');
        for (let i = 1; i < Math.min(postContainers.length, 5); i++) {
            const post = postContainers[i];
            
            // 优先查找 ignore_js_op 内的图片
            img = post.querySelector('ignore_js_op img');
            if (img && (img.getAttribute('file') || img.getAttribute('zoomfile'))) {
                return img;
            }
            
            // 查找带 aid 属性的图片
            img = post.querySelector('img[aid]');
            if (img && (img.getAttribute('file') || img.getAttribute('zoomfile'))) {
                return img; // 有file或zoomfile属性的aid图片直接返回
            }
            
            // 然后查找普通图片,但过滤表情图片
            img = post.querySelector('img');
            if (img && (img.getAttribute('src') || img.getAttribute('file') || img.getAttribute('zoomfile'))) {
                const src = img.getAttribute('src') || '';
                if (!src.includes('static/image/smiley/') && !src.includes('static/image/common/none.gif')) {
                    return img;
                }
            }
        }

        
        // 如果楼层查找失败,尝试其他备用方式
        
        // 5. 查找保存照片区域的图片
        img = doc.querySelector('div.savephotop img');
        if (img && (img.getAttribute('src') || img.getAttribute('file') || img.getAttribute('zoomfile'))) {
            const src = img.getAttribute('src') || '';
            if (!src.includes('static/image/common/none.gif')) return img;
        }

        // 6. 查找嵌套在各种容器内的 savephotop 图片
        img = doc.querySelector('.pcb .savephotop img');
        if (img && (img.getAttribute('src') || img.getAttribute('file') || img.getAttribute('zoomfile'))) {
            const src = img.getAttribute('src') || '';
            if (!src.includes('static/image/common/none.gif')) return img;
        }

        // 7. 查找 pattl 内的图片
        img = doc.querySelector('.pattl img');
        if (img && (img.getAttribute('src') || img.getAttribute('file') || img.getAttribute('zoomfile'))) {
            const src = img.getAttribute('src') || '';
            if (!src.includes('static/image/common/none.gif')) return img;
        }

        // 8. 查找 tattl attm 内的图片
        img = doc.querySelector('.tattl.attm img');
        if (img && (img.getAttribute('src') || img.getAttribute('file') || img.getAttribute('zoomfile'))) {
            const src = img.getAttribute('src') || '';
            if (!src.includes('static/image/common/none.gif')) return img;
        }

        // 9. 最后尝试任何带有 file 或 zoomfile 属性的图片(优先使用这些属性)
        const fileImgs = doc.querySelectorAll('img[file], img[zoomfile]');
        for (const img of fileImgs) {
            if (img.getAttribute('file') || img.getAttribute('zoomfile')) {
                const src = img.getAttribute('src') || '';
                if (!src.includes('static/image/common/none.gif')) return img;
            }
        }

        return null;
    }

    async function fetchFavorite(url) {
        try {
            const resp = await fetch(url);
            const html = await resp.text();
            const parser = new DOMParser();
            const doc = parser.parseFromString(html, 'text/html');
            const fav = doc.querySelector('#favoritenumber');
            return fav ? parseInt(fav.innerText) : 0;
        } catch(e) {
            return 0;
        }
    }

    async function loadThreads(grid, equalizeRowHeights) {
        const threads = getThreads();
        let loadedCount = 0;

        for (const tbody of threads) {
            const link = tbody.querySelector('th a.s.xst');
            if (!link) continue;
            const title = link.innerText;
            const url = link.href;

            const categoryElem = tbody.querySelector('th em a');
            const category = categoryElem ? categoryElem.innerText : '';

            const readPermElem = tbody.querySelector('th span.xw1');
            const readPerm = readPermElem ? readPermElem.innerText : '';

            const heatElem = tbody.querySelector('th .tbox.theatlevel');
            const heat = heatElem ? heatElem.innerText : '';

            const repliesElem = tbody.querySelector('td.num a.xi2');
            const replies = repliesElem ? repliesElem.innerText : '';

            const viewsElem = tbody.querySelector('td.num em');
            const views = viewsElem ? viewsElem.innerText : '';

            // 初始 favorite 0,异步更新
            let threadData = { title, url, category, heat, readPerm, views, replies, favorite: 0 };
            const card = createCard(threadData);
            grid.appendChild(card);

            fetch(url).then(resp => resp.text()).then(async html => {
                const parser = new DOMParser();
                const doc = parser.parseFromString(html, 'text/html');
                const firstImg = findFirstImage(doc);
                if (firstImg) {
                    let srcRel = firstImg.getAttribute('file') || firstImg.getAttribute('zoomfile') || firstImg.getAttribute('src');
                    if (srcRel) {
                        const imgUrl = srcRel.startsWith('http') ? srcRel : new URL(srcRel, DOMAIN).href;
                        if (/\.(jpe?g|png|webp)$/i.test(imgUrl)) {
                            setRealImage(card, imgUrl);
                        }
                    }
                }

                // 更新收藏
                const favElem = doc.querySelector('#favoritenumber');
                if (favElem) {
                    const favNum = parseInt(favElem.innerText) || 0;
                    const favDiv = card.querySelector('.fav-icon').parentNode;
                    favDiv.innerHTML = `<img src="${ICONS.favorite}" class="icon fav-icon"> ${favNum}`;
                }

                // 每次更新后重新计算等高
                loadedCount++;
                if (loadedCount === threads.length) {
                    setTimeout(equalizeRowHeights, 100);
                }
            });
        }
    }

    window.addEventListener('load', initGrid);
})();