您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
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