Pixiv Previewer

preview image in pixiv.

// ==UserScript==
// @name         Pixiv Previewer
// @namespace    https://github.com/lintmx
// @version      0.0.1
// @description  preview image in pixiv.
// @author       lintmx
// @license      MIT
// @match        https://www.pixiv.net*
// @grant        none
// @run-at       document-idle
// ==/UserScript==

const isDev = false;

const Logger = {
    debug: (...msg) => isDev && console.log('[Pixiv Previewer]', 'DEBUG:', ...msg),
    info: (...msg) => console.log('[Pixiv Previewer]', 'INFO:', ...msg),
    error: (...msg) => console.log('[Pixiv Previewer]', 'ERROR:', ...msg),
};

const Icon = {
    Expand:
        '',
    Redirect:
        '',
};

const Previewer = {
    addButton: () => {
        Logger.debug('Start Add Button');
        const illustList = document.querySelectorAll('div[type=illust]>div:not(.pixiv-previewer-btn-exist)');

        for (const illust of illustList) {
            const previewButton = document.createElement('div');
            previewButton.className = 'pixiv-previewer-view-btn';
            previewButton.style.cssText = [
                'position: absolute',
                'left: 0px',
                'bottom: 0px',
                'width: 32px',
                'height: 32px',
                'cursor: pointer',
                'background: no-repeat center/50%',
                'background-image: url("' + Icon.Expand + '")',
            ].join(';');
            previewButton.innerHTML = ' ';
            previewButton.addEventListener('click', (e) => {
                const clickImgElm = e.target.parentNode.querySelector('a>div:last-child>img');
                const pageSizeElm = e.target.parentNode.querySelector(
                    'a>div:first-child>div:last-child>div>span:last-child'
                );

                if (!clickImgElm) return;

                const imgUrl = clickImgElm.src;
                const pageSize = pageSizeElm ? pageSizeElm.innerText : '1';

                Previewer.showPreview(imgUrl, pageSize);
            });

            illust.appendChild(previewButton);
            illust.classList.add('pixiv-previewer-btn-exist');
        }
    },
    showPreview: (url, pageSize = 1) => {
        const { Time, IllustId } = Previewer.convert(url);
        const Cover = document.querySelector('#pixiv-previewer-cover');
        Cover.style.display = 'flex';
        Cover.dataset.time = Time;
        Cover.dataset.id = IllustId;
        Cover.dataset.size = pageSize;
        Cover.dataset.page = 0;
        document.querySelector('#pixiv-previewer-redirect').href = 'https://www.pixiv.net/artworks/' + IllustId;
        document.querySelector(
            '#pixiv-previewer-image'
        ).src = `https://i.pximg.net/img-master/img/${Time}/${IllustId}_p0_master1200.jpg`;
    },
    convert: (baseUrl = '') => {
        const [, Time = '', IllustId = ''] = baseUrl.match(
            /^https:\/\/i\.pximg\.net\/.+?\/img\/([0-9\/]+?)\/(\d+?)_p.+?\.[a-z]+$/
        );

        return { Time, IllustId };
    },
};

(function () {
    ('use strict');
    Logger.debug('Script Start.');

    const rootElm = document.querySelector('#root');

    // 全屏预览遮罩
    const previewCover = document.createElement('div');
    previewCover.id = 'pixiv-previewer-cover';
    previewCover.style.cssText = [
        'z-index: 445',
        'width: 100%',
        'height: 100%',
        'position: fixed',
        'left: 0px',
        'right: 0px',
        'top: 0px',
        'bottom: 0px',
        'display: none',
        'justify-content: center',
        'align-items: center',
        'flex-direction: column',
        'background: rgba(0, 0, 0, 0.6)',
    ].join(';');
    previewCover.addEventListener('click', (e) => {
        Logger.debug('cover click.');
        e.target.style.display = 'none';
    });

    const previewRedirect = document.createElement('a');
    previewRedirect.id = 'pixiv-previewer-redirect';
    previewRedirect.style.cssText = [
        'width: 32px',
        'height: 32px',
        'background: red',
        'margin-top: 16px',
        'display: block',
        'background: no-repeat center',
        'background-image: url("' + Icon.Redirect + '")',
    ].join(';');
    previewCover.appendChild(previewRedirect);

    const previewImage = document.createElement('img');
    previewImage.id = 'pixiv-previewer-image';
    previewImage.style.cssText = ['max-width: 90%', 'max-height: 90%', 'margin: auto'].join(';');
    previewImage.addEventListener('click', (e) => {
        e.stopPropagation();

        const Cover = document.querySelector('#pixiv-previewer-cover');
        const { time, id, size } = Cover.dataset;
        let { page = 0 } = Cover.dataset;

        let next = Number.parseInt(page, 10) + 1;
        if (next + 1 > size) {
            next = 0;
        }

        if (next != page) {
            document.querySelector(
                '#pixiv-previewer-image'
            ).src = `https://i.pximg.net/img-master/img/${time}/${id}_p${next}_master1200.jpg`;
            Cover.dataset.page = next;
        }

        Logger.debug('click image');
    });
    previewCover.appendChild(previewImage);

    rootElm.appendChild(previewCover);

    setInterval(Previewer.addButton, 1000);

    Logger.debug('Script End.');
})();