Greasy Fork

Greasy Fork is available in English.

下载Bilibili动态页面图片

下载“Bilibili动态”时间线页面的图片

当前为 2023-09-18 提交的版本,查看 最新版本

// ==UserScript==
// @name         Bilibili Download Pictures
// @name:zh-CN   下载Bilibili动态页面图片
// @version      0.8.0
// @description  Download pictures from bilibili timeline
// @description:zh-CN 下载“Bilibili动态”时间线页面的图片
// @author       OWENDSWANG
// @icon         https://avatars.githubusercontent.com/u/9076865?s=40&v=4
// @license      MIT
// @homepage     http://greasyfork.icu/scripts/421885
// @supportURL   https://github.com/owendswang/Download-Pictures-from-Bilibili-Timeline
// @match        https://t.bilibili.com/*
// @match        https://space.bilibili.com/*/dynamic*
// @match        https://www.bilibili.com/opus/*
// @grant        GM_download
// @namespace http://greasyfork.icu/users/738244
// ==/UserScript==

(function() {
    'use strict';

    // Your code here...
    let notLoaded = true;
    let cardsTotal = 0;
    let skeletonsTotal = 0;
    /*function mostViewedItemMouseClick(event) {
        notLoaded = true;
        cardsTotal = 0;
        let feed = document.body.querySelector('div.bili-dyn-list');
        if (feed) {
            // console.log('feed');
            feed.addEventListener('mouseover', feedMouseOver);
        }
    }*/
    function addOpusDownloadButton(card) {
        if(card.getElementsByClassName('download-button').length == 0) {
            // console.log(card);
            const buttonBar = card.getElementsByClassName('bili-tabs__nav__items')[0];
            let downloadButton = document.createElement('div');
            downloadButton.textContent = '下载';
            downloadButton.classList.add('bili-tabs__nav__item');
            downloadButton.addEventListener('click', function(event) {
                const content = document.body.querySelector('div.opus-module-content');
                const list = content.querySelectorAll('div.bili-album__preview__picture__img');
                // console.log(list);
                for (const item of list) {
                    let imgUrl = item.style.backgroundImage.split(/"|@/)[1] || item.querySelector('img').src.split('@')[0];
                    if (imgUrl.startsWith('//')) {
                        imgUrl = 'https:' + imgUrl;
                    }
                    const imgName = imgUrl.split('/')[imgUrl.split('/').length - 1];
                    // console.log(imgUrl);
                    // console.log(imgName);
                    GM_download(imgUrl, imgName);
                }
                const topAlbum = document.body.querySelector('div.opus-module-top__album');
                if (topAlbum) {
                    const topAlbumIndicatorList = topAlbum.querySelectorAll('div.horizontal-scroll-album__indicator > div > img');
                    const topAlbumList = topAlbum.querySelectorAll('div.horizontal-scroll-album__pic__img > img');
                    let topList = topAlbumList;
                    if (topAlbumIndicatorList.length > 0) topList = topAlbumIndicatorList;
                    for (const item of topList) {
                        let imgUrl = item.src.split(/@/)[0];
                        if (imgUrl.startsWith('//')) {
                            imgUrl = 'https:' + imgUrl;
                        }
                        const imgName = imgUrl.split('/')[imgUrl.split('/').length - 1];
                        // console.log(imgUrl);
                        // console.log(imgName);
                        GM_download(imgUrl, imgName);
                    }
                }
            });
            buttonBar.appendChild(downloadButton);
        }
    }
    function addDownloadButton(card) {
        if(card.getElementsByClassName('download-button').length == 0) {
            card.querySelectorAll('div.bili-dyn-item__footer > div.bili-dyn-item__action').forEach((ele) => { ele.style.marginRight = '48px'; });
            let buttonBar = card.getElementsByClassName('bili-dyn-item__footer')[0];
            let downloadButton = document.createElement('div');
            downloadButton.classList.add('bili-dyn-item__action');
            downloadButton.classList.add('download-button');
            let span = document.createElement('div');
            span.classList.add('bili-dyn-action');
            let icon = document.createElement('i');
            icon.style.width = '20px';
            icon.style.height = '20px';
            icon.style.transform = 'scale(0.8)';
            icon.style.backgroundImage = 'url(\'\')';
            let text = document.createElement('span');
            if (navigator.language == 'zh-CN') {
                text.textContent = '下载';
            } else {
                text.textContent = 'Download';
            }
            span.appendChild(icon);
            span.appendChild(text);
            downloadButton.appendChild(span);
            buttonBar.appendChild(downloadButton);
            downloadButton = buttonBar.getElementsByClassName('download-button')[0];
            downloadButton.addEventListener('mouseover', function(event) {
                this.querySelector('i').style.backgroundImage = 'url(\'\')';
                // console.log('over');
            });
            downloadButton.addEventListener('mouseout', function(event) {
                this.querySelector('i').style.backgroundImage = 'url(\'\')';
                // console.log('out');
            });
            downloadButton.addEventListener('click', function(event) {
                // console.log('click');
                event.preventDefault();
                const content = this.closest('div.bili-dyn-item__main');
                const list = content.querySelectorAll('div.bili-album__preview__picture__img');
                // console.log(list);
                if (list.length > 0) {
                    for (let j = 0; j < list.length; j++) {
                        let imgUrl;
                        if (list[j].querySelector('img')) {
                            imgUrl = list[j].querySelector('img').src.split(/@/)[0];
                        } else {
                            imgUrl = list[j].style.backgroundImage.split(/"|@/)[1];
                        }
                        // console.log(imgUrl);
                        if (imgUrl.startsWith('//')) {
                            imgUrl = 'https:' + imgUrl;
                        }
                        const imgName = imgUrl.split('/')[imgUrl.split('/').length - 1];
                        // console.log(imgName);
                        GM_download(imgUrl, imgName);
                    }
                }
            });
        }
    }
    function oldHandleCard(card) {
        if (card.getElementsByClassName('imagesbox').length > 0 && card.getElementsByClassName('download-button').length == 0) {
            // console.log('added download button');
            let buttonBar = card.getElementsByClassName('button-bar')[0];
            let buttons = buttonBar.getElementsByClassName('single-button');
            buttons[buttons.length - 1].classList.remove('p-rel');
            let downloadButton = document.createElement('div');
            downloadButton.classList.add('single-button', 'c-pointer', 'p-rel', 'download-button');
            downloadButton.style.display = 'inline-block';
            downloadButton.style.lineHeight = '48px';
            downloadButton.style.width = '92px';
            downloadButton.style.fontSize = '12px';
            let span = document.createElement('span');
            span.classList.add('text-bar');
            let icon = document.createElement('i');
            icon.style.background = 'center/contain no-repeat';
            icon.style.backgroundSize = '15px';
            icon.style.backgroundImage = 'url(\'\')';
            icon.style.width = '20px';
            icon.style.height = '20px';
            icon.style.display = 'inline-block';
            icon.style.verticalAlign = 'text-top';
            icon.style.margin = '0';
            icon.style.padding = '0';
            icon.style.marginRight = '4px';
            icon.style.marginTop = '-2px';
            let text = document.createElement('span');
            text.classList.add('text-offset');
            if (navigator.language == 'zh-CN') {
                text.textContent = '下载';
            } else {
                text.textContent = 'Download';
            }
            span.appendChild(icon);
            span.appendChild(text);
            downloadButton.appendChild(span);
            buttonBar.appendChild(downloadButton);
            downloadButton = buttonBar.getElementsByClassName('download-button')[0];
            downloadButton.addEventListener('mouseover', function(event) {
                this.querySelector('span.text-bar').style.color = '#00a1d6';
                this.querySelector('i').style.backgroundImage = 'url(\'\')';
                // console.log('over');
            });
            downloadButton.addEventListener('mouseout', function(event) {
                this.querySelector('span.text-bar').style.color = null;
                this.querySelector('i').style.backgroundImage = 'url(\'\')';
                // console.log('out');
            });
            downloadButton.addEventListener('click', function(event) {
                // console.log('click');
                const content = this.closest('div.main-content');
                let list = content.querySelectorAll('div.img-content');
                if (list.length > 0) {
                    for (let j = 0; j < list.length; j++) {
                        let imgUrl = list[j].style.backgroundImage.split(/"|@/)[1];
                        if (imgUrl.startsWith('//')) {
                            imgUrl = 'https:' + imgUrl;
                        }
                        const imgName = imgUrl.split('/')[imgUrl.split('/').length - 1];
                        // console.log(imgUrl);
                        // console.log(imgName);
                        GM_download(imgUrl, imgName);
                    }
                } else {
                    list = content.querySelectorAll('li > img');
                    if (list.length > 0) {
                        for (let k = 0; k < list.length; k++) {
                            let imgUrl = list[k].src.split('@')[0];
                            if (imgUrl.startsWith('//')) {
                                imgUrl = 'https:' + imgUrl;
                            }
                            const imgName = imgUrl.split('/')[imgUrl.split('/').length - 1];
                            GM_download(imgUrl, imgName);
                        }
                    } else {
                        const singleImg = content.querySelector('div.boost-img-container > img');
                        let imgUrl = singleImg.src.split('@')[0];
                        const imgName = imgUrl.split('/')[imgUrl.split('/').length - 1];
                        GM_download(imgUrl, imgName);
                    }
                }
            });
        }
    }
    function handleOpusCard(card) {
         if (card.getElementsByClassName('bili-album').length > 0 || card.getElementsByClassName('')) {
             addOpusDownloadButton(card);
         }
    }
    function handleCard(card) {
        if (card.getElementsByClassName('bili-album').length > 0) {
            // console.log('add download button');
            card.getElementsByClassName('bili-album__preview__picture__img').forEach((img) => {
                img.addEventListener('click', function(event) {
                 addDownloadButton(card);
                });
            });
            addDownloadButton(card);
        }
    }
    function oldFeedMouseOver(event) {
        const cards = document.body.querySelectorAll('div.feed-card div.card');
        const skeletonCards = document.body.querySelectorAll('div.feed-card div.card > div.skeleton');
        // console.log(skeletonCards.length);
        if (notLoaded || skeletonCards.length != skeletonsTotal || cards.length > cardsTotal) {
            // console.log('cards');
            cardsTotal = cards.length;
            // console.log(cardsTotal);
            skeletonsTotal = skeletonCards.length;
            for (let i = 0; i < cardsTotal; i++) {
                // console.log('card');
                oldHandleCard(cards[i])
                // startIndex += 1;
            }
            if (cardsTotal > 0) {
                notLoaded = false;
            }
            // document.body.removeEventListener('mouseover', bodyMouseOver);
        }
    }
    function feedMouseOver(event) {
        const cards = document.body.querySelectorAll('div.bili-dyn-list div.bili-dyn-item');
        // console.log(cards.length);
        if (notLoaded || cards.length > cardsTotal) {
            // console.log('cards');
            cardsTotal = cards.length;
            // console.log(cardsTotal);
            for (let i = 0; i < cardsTotal; i++) {
                // console.log('card');
                handleCard(cards[i])
                // startIndex += 1;
            }
            if (cardsTotal > 0) {
                notLoaded = false;
            }
            // document.body.removeEventListener('mouseover', bodyMouseOver);
        }
    }
    function bodyMouseOver(event) {
        if (notLoaded) {
            // console.log('body');
            const mostViewedItems = document.body.querySelectorAll('div.bili-dyn-up-list__item');
            // console.log(mostViewedItems.length);
            /*for (let l = 0; l < mostViewedItems.length; l++) {
                mostViewedItems[l].addEventListener('click', mostViewedItemMouseClick);
            }*/
            let feed = document.body.querySelector('div.bili-dyn-list');
            if (feed) {
                // console.log('feed');
                // feed.addEventListener('mouseover', feedMouseOver);
                feedMouseOver();
            } else if(document.body.querySelector('div.bili-dyn-item')) {
                const card = document.body.querySelector('div.bili-dyn-item');
                if (card) {
                    handleCard(card);
                    notLoaded = false;
                    // document.body.removeEventListener('mouseover', bodyMouseOver);
                }
            } else if (document.body.querySelector('div.opus-detail')) {
                const card = document.body.querySelector('div.opus-detail');
                if (card) {
                    handleOpusCard(card);
                    notLoaded = false;
                    // document.body.removeEventListener('mouseover', bodyMouseOver);
                }
            } else if(document.body.querySelector('div.feed-card')) {
                // feed = document.querySelector('div.feed-card');
                // feed.addEventListener('mouseover', oldFeedMouseOver);
                oldFeedMouseOver();
            }
        }
    }
    //document.body.addEventListener('mouseover', bodyMouseOver);
    bodyMouseOver();
    new MutationObserver((mutationList, observer) => {
        for (const mutation of mutationList) {
            // console.log(mutation.target);
            if (mutation.type === 'childList') {
                for (const node of mutation.addedNodes) {
                    //console.log(node);
                }
                if (mutation.target.tagName === 'DIV' && mutation.target.className === 'bili-dyn-list__items') {
                    // console.log(mutation.addedNodes);
                    for (const node of mutation.addedNodes) {
                        handleCard(node);
                    }
                }
            }
        }
    }).observe(document.body, { attributes: false, childList: true, subtree: true });
})();