Booru Revamped

This adds a couple of changes to the layout and behaviour of the site

Stan na 17-04-2022. Zobacz najnowsza wersja.

Aby zainstalować ten skrypt, wymagana jest instalacje jednego z następujących rozszerzeń: Tampermonkey, Greasemonkey lub Violentmonkey.

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

Aby zainstalować ten skrypt, wymagana jest instalacje jednego z następujących rozszerzeń: Tampermonkey, Violentmonkey.

Aby zainstalować ten skrypt, wymagana będzie instalacja rozszerzenia Tampermonkey lub Userscripts.

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

Aby zainstalować ten skrypt, musisz zainstalować rozszerzenie menedżera skryptów użytkownika.

(Mam już menedżera skryptów użytkownika, pozwól mi to zainstalować!)

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

Będziesz musiał zainstalować rozszerzenie menedżera stylów użytkownika, aby zainstalować ten styl.

Będziesz musiał zainstalować rozszerzenie menedżera stylów użytkownika, aby zainstalować ten styl.

Musisz zainstalować rozszerzenie menedżera stylów użytkownika, aby zainstalować ten styl.

(Mam już menedżera stylów użytkownika, pozwól mi to zainstalować!)

"use strict";
// ==UserScript==
// @name         Booru Revamped
// @namespace    westerhold78
// @version      2.2
// @description  This adds a couple of changes to the layout and behaviour of the site
// @match        *://gelbooru.com/*
// @match        *://yande.re/*
// @match        *://danbooru.donmai.us/*
// @match        *://safebooru.org/*
// @match        *://konachan.com/*
// @match        *://konachan.net/*
// @match        *://rule34.paheal.net/*
// @match        *://rule34.xxx/*
// @icon         
// @copyright    2022, westerhold78
// @grant        GM_addStyle
// @grant        GM_xmlhttpRequest
// ==/UserScript==
const style = [];
const newTabForDetailPage = true;
const zoomScale = 2.5;
const zoomDelay = 0.4;
const imageTypes = ['jpg', 'png', 'jpeg', 'gif'];
const movieExtensions = ['mp4', 'webm'];
const rule34pahealSelectorImg = '#image-list .shm-thumb-link img';
const rule34pahealSelectorVideo = '#image-list .shm-thumb-link video';
const replaceMeString = '>>[[REPLACEME]]<<';
const { hash, hostname, pathname, searchParams } = new URL(window.location.href);
let anchorElement;
const configuration = {
    'yande.re': {
        pages: {
            list: {
                pathname: /^\/post$/,
            },
            post: {
                pathname: /^\/post\/show\/.*$/,
            },
        },
        selectors: {
            anchor: '#post-list-posts .inner .thumb',
            image: '#post-list-posts .inner .thumb img.preview',
        },
        sample: {
            type: 'selector',
            selector: '#image',
        },
        original: {
            type: 'selector',
            selector: '#highres',
        },
        postPageImageId: 'image',
        maxHeight: 'calc(100vh - 260px)',
        maxWidth: 'calc(100vw - 310px)',
    },
    'gelbooru.com': {
        pages: {
            list: {
                searchParams: {
                    page: 'post',
                    s: 'list',
                },
            },
            post: {
                searchParams: {
                    page: 'post',
                    s: 'view',
                },
            },
        },
        selectors: {
            anchor: '#container main .thumbnail-container .thumbnail-preview a',
            image: '#container main .thumbnail-container .thumbnail-preview a img',
        },
        sample: {
            type: 'selector',
            selector: '#image',
        },
        original: {
            type: 'ogImage',
        },
        postPageImageId: 'image',
        maxHeight: 'calc(100vh - 285px)',
        maxWidth: 'calc(100vw - 260px)',
    },
    'safebooru.org': {
        pages: {
            list: {
                searchParams: {
                    page: 'post',
                    s: 'list',
                },
            },
            post: {
                searchParams: {
                    page: 'post',
                    s: 'view',
                },
            },
        },
        selectors: {
            anchor: '#post-list .content .thumb a',
            image: '#post-list .content .thumb a img.preview',
        },
        sample: {
            type: 'selector',
            selector: '#image',
        },
        original: {
            type: 'ogImage',
        },
        postPageImageId: 'image',
        maxHeight: 'calc(100vh - 200px)',
        maxWidth: 'calc(100vw - 260px)',
    },
    'danbooru.donmai.us': {
        pages: {
            list: {
                pathname: /^\/posts$/,
            },
            post: {
                pathname: /^\/posts\/.*$/,
            },
        },
        selectors: {
            anchor: '#posts .posts-container article .post-preview-link',
            image: '#posts .posts-container article .post-preview-link img.post-preview-image',
        },
        sample: {
            type: 'ogImage',
        },
        original: {
            type: 'selector',
            selector: '#image',
        },
        postPageImageId: 'image',
        maxHeight: 'calc(100vh - 200px)',
        maxWidth: 'calc(100vw - 260px)',
    },
    'konachan.net': {
        pages: {
            list: {
                pathname: /^\/post$/,
            },
            post: {
                pathname: /^\/post\/show\/.*$/,
            },
        },
        selectors: {
            anchor: '#post-list-posts .inner .thumb',
            image: '#post-list-posts .inner .thumb img.preview',
        },
        sample: {
            type: 'selector',
            selector: '#image',
        },
        original: {
            type: 'selector',
            selector: '#highres',
        },
        postPageImageId: 'image',
        maxHeight: 'calc(100vh - 240px)',
        maxWidth: 'calc(100vw - 280px)',
    },
    'konachan.com': {
        pages: {
            list: {
                pathname: /^\/post$/,
            },
            post: {
                pathname: /^\/post\/show\/.*$/,
            },
        },
        selectors: {
            anchor: '#post-list-posts .inner .thumb',
            image: '#post-list-posts .inner .thumb img.preview',
        },
        sample: {
            type: 'ogImage',
        },
        original: {
            type: 'selector',
            selector: '#image',
        },
        postPageImageId: 'image',
        maxHeight: 'calc(100vh - 240px)',
        maxWidth: 'calc(100vw - 280px)',
    },
    'rule34.paheal.net': {
        pages: {
            list: {
                pathname: /^\/post\/list\/.*$/,
            },
            post: {
                pathname: /^\/post\/view\/.*$/,
            },
        },
        selectors: {
            anchor: '.shm-image-list .shm-thumb-link',
            image: '.shm-image-list .shm-thumb-link img',
        },
        sample: {
            type: 'ogImage',
        },
        original: {
            type: 'selector',
            selector: '#main_image',
        },
        postPageImageId: 'main_image',
        maxHeight: 'calc(100vh - 345px)',
        maxWidth: 'calc(100vw - 380px)',
    },
    'rule34.xxx': {
        pages: {
            list: {
                searchParams: {
                    page: 'post',
                    s: 'list',
                },
            },
            post: {
                searchParams: {
                    page: 'post',
                    s: 'view',
                },
            },
        },
        selectors: {
            anchor: '#post-list .content .image-list .thumb a',
            image: '#post-list .content .image-list .thumb a img',
        },
        sample: {
            type: 'selector',
            selector: '#image',
        },
        original: {
            type: 'ogImage',
            fallback: {
                type: 'selector',
                selector: '#gelcomVideoPlayer',
            },
        },
        postPageImageId: 'image',
        maxHeight: 'calc(100vh - 220px)',
        maxWidth: 'calc(100vw - 270px)',
    },
};
const stylesArr = {
    'yande.re': {
        overflowSelector: '#post-list-posts .inner',
        frameSelector: '#post-list .content .thumb video',
        scalingSelector: '#post-list .content .thumb *:is(img, video):hover',
        zIndexSelector: '#post-list .content .thumb *:is(img, video):hover',
    },
    'gelbooru.com': {
        scalingSelector: 'article.thumbnail-preview *:is(img, video):hover',
        frameSelector: 'article.thumbnail-preview video',
        zIndexSelector: 'article.thumbnail-preview *:is(img, video):hover',
    },
    'danbooru.donmai.us': {
        scalingSelector: 'a.post-preview-link *:is(img, video):hover',
        frameSelector: 'a.post-preview-link video',
        zIndexSelector: 'a.post-preview-link *:is(img, video):hover',
    },
    'safebooru.org': {
        scalingSelector: '#post-list .content .thumb a *:is(img, video):hover',
        zIndexSelector: '#post-list .content .thumb a *:is(img, video):hover',
    },
    'konachan.com': {
        overflowSelector: '#post-list-posts .inner',
        scalingSelector: '#post-list-posts .inner a.thumb *:is(img, video):hover',
        zIndexSelector: '#post-list-posts .inner a.thumb *:is(img, video):hover',
    },
    'konachan.net': {
        overflowSelector: '#post-list-posts .inner',
        scalingSelector: '#post-list-posts .inner a.thumb *:is(img, video):hover',
        zIndexSelector: '#post-list-posts .inner a.thumb *:is(img, video):hover',
    },
    'rule34.paheal.net': {
        scalingSelector: '#image-list .shm-image-list .shm-thumb-link *:is(img, video):hover',
        zIndexSelector: '#image-list .shm-image-list .shm-thumb-link *:is(img, video):hover',
    },
    'rule34.xxx': {
        scalingSelector: '#post-list .content .image-list .thumb a *:is(img, video):hover',
        zIndexSelector: '#post-list .content .image-list .thumb a *:is(img, video):hover',
    },
};
const element = stylesArr[hostname];
const { scalingSelector, frameSelector, overflowSelector, zIndexSelector } = element || {};
if (scalingSelector) {
    style.push(`${scalingSelector} {
      transform: scale(${zoomScale});
      -moz-transform: scale(${zoomScale});
      -webkit-transform: scale(${zoomScale});
      transition-delay: ${zoomDelay}s;
      transition-property: transform;
    }`);
}
if (frameSelector) {
    style.push(`${element.frameSelector} {
      border: 3px solid #0000ff;
  }`);
}
if (overflowSelector) {
    style.push(`${overflowSelector} {
      position: inherit;
      overflow: visible !important;
    }`);
}
if (zIndexSelector) {
    style.push(`${zIndexSelector} {
      position: relative;
      z-index: 9001;
    }`);
}
style.push(`[data-ext="mp4"] a video, [data-ext="webm"] a video {
    border: 3px solid #0000ff;
  }`);
GM_addStyle(style.join('\n'));
// -- - - - - -
const onLoad = () => {
    // handleHostname()
    handlePaths();
};
// start of the application code
if (document.readyState !== 'loading') {
    onLoad();
}
else {
    document.addEventListener('DOMContentLoaded', () => {
        onLoad();
    });
}
function handlePaths() {
    var _a, _b, _c, _d;
    const pathnameMatch = (_b = (_a = Object.entries(configuration[hostname].pages).filter(([pageName, page]) => { var _a, _b, _c; return ((_c = (_b = (_a = page.pathname) === null || _a === void 0 ? void 0 : _a.exec(pathname)) === null || _b === void 0 ? void 0 : _b.length) !== null && _c !== void 0 ? _c : 0) > 0; })) === null || _a === void 0 ? void 0 : _a[0]) === null || _b === void 0 ? void 0 : _b[0];
    const searchParamsMatch = (_d = (_c = Object.entries(configuration[hostname].pages).filter(([pageName, page]) => {
        var _a;
        const searchParamsLength = Object.entries(page.searchParams || []).filter(([key, param]) => {
            return param === searchParams.get(key);
        }).length;
        const pageSearchParamsLength = Object.keys((_a = page.searchParams) !== null && _a !== void 0 ? _a : []).length;
        return (searchParamsLength === pageSearchParamsLength &&
            searchParamsLength > 0 &&
            pageSearchParamsLength > 0);
    })) === null || _c === void 0 ? void 0 : _c[0]) === null || _d === void 0 ? void 0 : _d[0];
    if (pathnameMatch === 'list' ||
        searchParamsMatch === 'list' ||
        (hostname === 'danbooru.donmai.us' && pathname === '/')) {
        // on the list page
        openImageInNewTab();
        document
            .querySelectorAll(configuration[hostname].selectors.image)
            .forEach(async (image) => {
            image.addEventListener('mouseover', async (event) => await onImageMouseOver(event));
        });
    }
    else if (pathnameMatch === 'post' || searchParamsMatch === 'post') {
        // on the post page
        const original = getUrl(document, 'original');
        const imageElement = document.getElementById(configuration[hostname].postPageImageId);
        imageElement.onclick = null;
        imageElement === null || imageElement === void 0 ? void 0 : imageElement.addEventListener('click', () => {
            window.open(original, '_self');
        });
        imageElement.style.maxHeight = configuration[hostname].maxHeight;
        imageElement.style.maxWidth = configuration[hostname].maxWidth;
        imageElement.style.height = 'unset';
        imageElement.style.width = 'unset';
        imageElement === null || imageElement === void 0 ? void 0 : imageElement.scrollIntoView({
            behavior: 'smooth',
            block: 'end',
        });
    }
}
function openImageInNewTab() {
    // Open detail page in new tab
    if (newTabForDetailPage) {
        const thumbs = document.querySelectorAll(configuration[hostname].selectors.anchor);
        Array.from(thumbs).map(thumb => thumb.setAttribute('target', '_blank'));
    }
}
async function onImageMouseOver(event) {
    var _a;
    const image = event.target;
    // remove tooltip
    image.title = '';
    if (!image.classList.contains('loaded')) {
        const urlList = await getURLList(image);
        if (hostname === 'rule34.paheal.net') {
            const parent = (_a = image.parentElement) === null || _a === void 0 ? void 0 : _a.parentElement;
            const sibling = parent === null || parent === void 0 ? void 0 : parent.children[2];
            urlList.original = sibling.href;
        }
        const type = await getType(image);
        if (type === 'video') {
            await swapVideo(image, urlList.original);
        }
        else {
            await swapImageURLs(image, urlList);
        }
    }
}
async function getType(image) {
    var _a, _b;
    switch (hostname) {
        case 'gelbooru.com':
            return movieExtensions.includes(image.classList[0]) ? 'video' : 'image';
        case 'danbooru.donmai.us': {
            const url = new URL(((_a = image.parentElement) === null || _a === void 0 ? void 0 : _a.parentElement).href);
            const doc = await getDocFromURL(url.href);
            const type = getPostType(doc);
            return type;
        }
        case 'rule34.paheal.net': {
            const extension = ((_b = image.parentElement) === null || _b === void 0 ? void 0 : _b.parentElement)
                .dataset['ext'];
            return movieExtensions.includes(extension) ? 'video' : 'image';
        }
        case 'rule34.xxx': {
            const url = new URL(image.parentElement.href);
            const doc = await getDocFromURL(url.href);
            const type = getPostType(doc);
            return type;
        }
        default:
            return null;
    }
}
async function swapImageURLs(image, urlList) {
    const { sample, original } = urlList;
    if (sample !== undefined) {
        setAsyncImage(image, sample).then(img => {
            if (original !== undefined) {
                setImage(image, original, image.width);
            }
        });
    }
    else {
        if (original !== undefined) {
            setImage(image, original, image.width);
        }
    }
}
async function getURLList(image) {
    var _a;
    switch (hostname) {
        case 'danbooru.donmai.us': {
            const url = ((_a = image === null || image === void 0 ? void 0 : image.parentElement) === null || _a === void 0 ? void 0 : _a.parentElement)
                .href;
            const list = await fetchImageURLGeneric(url);
            return list;
        }
        default: {
            const url = (image === null || image === void 0 ? void 0 : image.parentElement).href;
            const list = await fetchImageURLGeneric(url);
            return list !== null && list !== void 0 ? list : {};
        }
    }
}
function swapVideo(image, url) {
    const video = document.createElement('video');
    video.autoplay = true;
    video.loop = true;
    video.muted = true;
    video.poster = image.src;
    video.height = image.height;
    video.width = image.width;
    const source = document.createElement('source');
    source.type = 'video/mp4';
    source.src = url;
    video.appendChild(source);
    image.replaceWith(video);
}
async function fetchImageURLGeneric(url) {
    const doc = await getDocFromURL(url);
    const sample = getUrl(doc, 'sample');
    const original = getUrl(doc, 'original');
    const returnValue = {
        ...(original !== undefined && original.length > 0 && { original }),
        ...(sample !== undefined && sample !== original && { sample }),
    };
    return returnValue;
}
async function getDocFromURL(url) {
    const doc = await fetch(url)
        .then(response => {
        // The API call was successful!
        return response.text();
    })
        .then(html => {
        // Convert the HTML string into a document object
        const parser = new DOMParser();
        const doc = parser.parseFromString(html, 'text/html');
        return doc;
    });
    return doc;
}
function getUrl(doc, quality) {
    var _a, _b;
    const settings = configuration[hostname][quality];
    switch (settings.type) {
        case 'ogImage':
            return doc
                .querySelector("meta[property='og:image']")
                .getAttribute('content');
        case 'selector': {
            const element = doc.querySelector(settings.selector);
            return ((_b = (_a = element === null || element === void 0 ? void 0 : element.src) !== null && _a !== void 0 ? _a : element === null || element === void 0 ? void 0 : element.href) !== null && _b !== void 0 ? _b : element === null || element === void 0 ? void 0 : element.getElementsByTagName('source')[0].src);
        }
        default:
            return '';
    }
}
function getPostType(doc) {
    var _a;
    const element = (_a = doc.querySelector(configuration[hostname]['original'].selector)) !== null && _a !== void 0 ? _a : doc.querySelector(configuration[hostname]['original'].fallback.selector);
    return element instanceof HTMLVideoElement
        ? 'video'
        : element instanceof HTMLImageElement
            ? 'image'
            : null;
}
function setImage(image, url, width) {
    return new Promise((resolve, reject) => {
        var _a;
        image.onload = () => resolve();
        image.onerror = () => reject();
        image.width = width;
        image.src = url;
        if (hostname === 'danbooru.donmai.us') {
            (_a = image.previousElementSibling) === null || _a === void 0 ? void 0 : _a.remove();
        }
    });
}
async function setAsyncImage(image, url) {
    return new Promise((resolve, reject) => {
        setTimeout(async () => {
            try {
                if (url === undefined) {
                    return;
                }
                await setImage(image, url, image.width);
                return url;
            }
            catch (error) {
                console.error(error);
                reject(error);
                return;
            }
            finally {
                resolve(url);
            }
        });
    });
}
//# sourceMappingURL=index.js.map