Booru Revamped

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

Устаревшая версия за 24.12.2021. Перейдите к последней версии.

Чтобы установить этот скрипт, вы сначала должны установить расширение браузера, например Tampermonkey, Greasemonkey или Violentmonkey.

Для установки этого скрипта вам необходимо установить расширение, такое как Tampermonkey.

Чтобы установить этот скрипт, вы сначала должны установить расширение браузера, например Tampermonkey или Violentmonkey.

Чтобы установить этот скрипт, вы сначала должны установить расширение браузера, например Tampermonkey или Userscripts.

Чтобы установить этот скрипт, сначала вы должны установить расширение браузера, например Tampermonkey.

Чтобы установить этот скрипт, вы должны установить расширение — менеджер скриптов.

(у меня уже есть менеджер скриптов, дайте мне установить скрипт!)

Чтобы установить этот стиль, сначала вы должны установить расширение браузера, например Stylus.

Чтобы установить этот стиль, сначала вы должны установить расширение браузера, например Stylus.

Чтобы установить этот стиль, сначала вы должны установить расширение браузера, например Stylus.

Чтобы установить этот стиль, сначала вы должны установить расширение — менеджер стилей.

Чтобы установить этот стиль, сначала вы должны установить расширение — менеджер стилей.

Чтобы установить этот стиль, сначала вы должны установить расширение — менеджер стилей.

(у меня уже есть менеджер стилей, дайте мне установить скрипт!)

// ==UserScript==
// @name         Booru Revamped
// @namespace    westerhold78
// @version      1.6
// @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/*
// @require      https://code.jquery.com/jquery-3.6.0.min.js
// @icon         
// @copyright    2021, westerhold78
// @grant        GM_addStyle
// @grant        GM_xmlhttpRequest
// ==/UserScript==

const jQuery = window.jQuery;
const style = [];
const newTabForDetailPage = true;
const zoomScale = 2.5;
const zoomDelay = 0.4
const imageTypes = ['.jpg', '.png','.jpeg','.gif'];
const thumbnailPreviewClass = 'thumbnail-preview';
const hostname = window.location.hostname;
const pathname = window.location.pathname;
let rule34pahealSelector = '#image-list .shm-thumb-link img';

function imageExists(url) {
    return new Promise((resolve, reject) => {
        var image = new Image();
        image.onload = () => { resolve(url); }
        image.onerror = () => { reject(url); }
        image.src = url;
    });
}

function replaceExtension(extensionIndex, url) {
    if (extensionIndex > -1) {
        url = url.substring(0, url.lastIndexOf('.')) + imageTypes[extensionIndex];
    }
    return url;
}

function loadImage(image, url, imageWidth = 0, {extensionIndex = -1, replaceArr = {}} = {}) {
    url = replaceExtension(extensionIndex, url);
    imageExists(url)
    .then((_url) => {
        setImage(image, _url, imageWidth);
    })
    .catch((_url) => {
        if(extensionIndex > -1) {
            var condition = extensionIndex < imageTypes.length - 1;
            extensionIndex = (condition) ? ++extensionIndex : 0;
            _url = replaceExtension(extensionIndex, _url);
            if (condition) {
                loadImage(image, _url, imageWidth, {extensionIndex: extensionIndex, replaceArr: replaceArr});
            }
            else {
                for (const [pattern, value] of Object.entries(replaceArr)) {
                    if (!_url.includes(value) || value == '') {
                        _url = _url
                            .replace(pattern, value);
                    }
                }
                loadImage(image, _url, imageWidth, {extensionIndex: extensionIndex, replaceArr: replaceArr});
            }
        }
    });
}

function setImage(image, _url, imageWidth) {
    image.src = _url;
    image.width = imageWidth;
    image.classList.add('loaded');
}

function loadPicture(picture, url) {
    var source = picture.children[1];
    var image = picture.children[2];
    var imageWidth = image.innerWidth;

    imageExists(url)
    .then((url) => {
        source.srcset = url;
        image.width = imageWidth;
        picture.classList.add('loaded');
    })
    .catch((url) => {
        if (!url.includes('data/sample')) {
            url = url
                .replace('/data/', '/data/sample/sample-');
            loadPicture(picture, url);
        }
    });
}

function urlParam(name){
    const results = new RegExp('[\?&]' + name + '=([^&#]*)').exec(window.location.href);
    if (results === null){
       return null;
    }
    else{
       return results[1] ?? 0;
    }
};

function getPathFragments(name){
    return pathname.replace(/^\//, '').split('/')
};

// If you need to use another JavaScript library alongside jQuery, return control of $ back to the other library with a call to $.noConflict(). Old references of $ are saved during jQuery initialization; noConflict() simply restores them.
jQuery.noConflict();
(function( $ ) {
    $(function() {
        // Code that uses jQuery's $ can follow here.
        'use strict';

        switch (hostname) {
            case 'yande.re': {
                const pathFragments = getPathFragments();

                if (pathFragments[0] === 'post' && pathFragments[1] === 'show') { // post detail page

                    // scroll to sidebar
                    const sidebar = $(".sidebar")[0];
                    sidebar.scrollIntoView({
                       behavior: 'smooth'
                    });
                }
                break;
            }
        }

        if(urlParam('s') == 'view') {
            var image = $('#image');
            var top = image.offset().top;
            var url = $(image).attr('src');
            if(url.includes('samples/')) {
                url = url.replace('samples', 'images')
                    .replace('sample_', '');
                loadImage(image, url);
            }
            window.scrollTo({ top: top, behavior: 'smooth' });
        }


        if(urlParam('s') == 'post') {
            alert('kutjes')
        }

        //Replace thumbnails
        $(`.${thumbnailPreviewClass} img, #post-list-posts img, #post-list .content img, #posts-container article picture, ${rule34pahealSelector}`).mouseover(function(event) {
            var image = event.target;
            if (!$(image).hasClass("loaded")) {
                var imageInnerWidth = $(image).innerWidth();

                // remove tooltip
                $(image).attr('title','');

                //replace url
                var patternFileExtension = /(\.[0-9a-z]+$)/i;
                var patternDomain = /^(http:\/\/www\.|https:\/\/www\.|http:\/\/|https:\/\/)?(.*\.)?(gelbooru\.com|yande\.re|donmai\.us|konachan\.com|konachan\.net)\/?(?:thumbnails)?(.*)/;
                var url = $(image).attr('src');
                switch (hostname) {
                    case 'yande.re':
                        url = url
                            .replace(patternDomain, 'https://$2$3/$4')
                            .replace('assets.', 'files.')
                            .replace(/data\/preview\/(.*)\/(.*)\//, 'sample/')
                            .replace(patternFileExtension, '/$1');
                        loadImage(image, url, imageInnerWidth, {extensionIndex: -1, replaceArr: {'sample':'image'}});
                        break;
                    case 'konachan.com': case 'konachan.net':
                        url = url
                            .replace(patternDomain, 'https://$2$3/$4')
                            .replace('assets.', 'files.')
                            .replace(/data\/preview\/(.*)\/(.*)\//, 'sample/')
                            .replace(patternFileExtension, '/$1');
                        loadImage(image, url, imageInnerWidth, {extensionIndex: -1, replaceArr: {'sample':'image'}});
                        break;
                    case 'rule34.xxx': case 'safebooru.org':
                        url = url
                            .replace('thumbnails', 'samples')
                            .replace('thumbnail_', 'sample_');
                        loadImage(image, url, imageInnerWidth, {extensionIndex: 0, replaceArr: {'samples':'images', 'sample_': ''}});
                        break;
                    case 'rule34.paheal.net':
                        url = url
                            .replace('_thumbs', '_images');
                        loadImage(image, url, imageInnerWidth);
                        break;
                    case 'gelbooru.com':
                        url = url
                            .replace(patternDomain, 'https://$2gelbooru.com/samples/$4')
                            .replace('thumbnail_', 'sample_');
                        loadImage(image, url, imageInnerWidth, {extensionIndex: 0, replaceArr: {'samples':'images', 'sample_': ''}});
                        break;
                    case 'danbooru.donmai.us':
                        var picture = event.currentTarget;
                        if (!picture.classList.contains('loaded')) {
                            url = picture.children[2].src
                                .replace(patternDomain, 'https://danbooru.donmai.us/data/$4')
                                .replace('sample/', 'sample/sample-')
                                .replace(/preview\/(.*)\/(.*)\//, '');
                            loadPicture(picture, url);
                        }
                        break;
                }
            }
        });

        // Open detail page in new tab
        if(newTabForDetailPage) {
            $('.thumbnail-preview a, #post-list-posts li .thumb, #post-list .content .thumb a, #posts-container article a').on('click', function(e){
                e.preventDefault();
                var url = $(this).attr('href');
                window.open(url, '_blank');
            });
        }

        // Open image on details page in new tab
        $('#image').click(function() {
            const sampleUrl = $(this)[0].src
            const url = $('.highres-show')[0]
            window.open(url, '_self');
        });
    });
})(jQuery);

style.push(
    `.${thumbnailPreviewClass} img:hover, #post-list .content .thumb img:hover, #posts img:hover, ${rule34pahealSelector}:hover { transform: scale(${zoomScale}); -moz-transform: scale(${zoomScale}); -webkit-transform: scale(${zoomScale}); transition-delay: ${zoomDelay}s; transition-property: transform; }`,
    `#post-list-posts .inner, article.post-preview { position: inherit; overflow: visible !important; }`,
    `img#image { object-fit: contain; cursor: pointer; width: unset !important; max-width: 100%; max-height: 90vh; }`,
    `#resized_notice { display: none; }`
);

GM_addStyle(style.join("\n"));