Booru Revamped

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

"use strict";
// ==UserScript==
// @name         Booru Revamped
// @namespace    westerhold78
// @version      2.3
// @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