// ==UserScript==
// @name Desu X - Enhancement Script for Desuarchive.org
// @version 2.5
// @description Combines infinite scrolling, media preview on hover, download functionality, Fappe Tyme™ and gallery mode for desuarchive.org. Alt+G to activate gallery mode. 'F' to toggle fappe tyme. Press 'S' while hovering over a thumbnail or in gallery mode to download media with the original filename.
// @author kpganon
// @license MIT
// @namespace https://github.com/kpg-anon/scripts
// @match https://desuarchive.org/*
// @grant GM_addStyle
// @grant GM_xmlhttpRequest
// @grant GM_addElement
// ==/UserScript==
(function() {
'use strict';
GM_addStyle(`
@import url('https://fonts.googleapis.com/css?family=Roboto');
* {
font-family: 'Roboto', sans-serif !important;
}
.desux-search-page .paginate {
display: none !important;
}
.desux-search-page article.thread {
padding: 0 !important;
border-top: none !important;
}
.hidden-by-desux {
visibility: hidden;
position: absolute;
width: 1px;
height: 1px;
margin: -1px;
overflow: hidden;
clip: rect(0, 0, 0, 0);
clip-path: inset(50%);
white-space: nowrap;
}
[data-hidden="true"] {
opacity: 0;
pointer-events: none;
user-select: none;
position: absolute;
width: 1px;
height: 1px;
overflow: hidden;
clip: rect(0,0,0,0);
clip-path: inset(100%);
}
#hover-preview {
display: none;
position: fixed;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
z-index: 100;
pointer-events: none;
max-width: 100vw;
max-height: 100vh;
}
#hover-preview img,
#hover-preview video {
width: auto;
height: auto;
max-width: 100vw;
max-height: 100vh;
object-fit: contain;
}
#ig-galleryContainer {
position: fixed;
top: 0;
left: 0;
right: 0;
bottom: 0;
background-color: rgba(0, 0, 0, 0.8);
z-index: 99999;
display: flex;
align-items: center;
justify-content: center;
padding-bottom: 80px;
}
#ig-galleryImage {
max-width: 90%;
max-height: calc(100% - 80px - 20px);
transition: all 0.3s ease;
}
#ig-imageCounter {
position: absolute;
top: 10px;
left: 10px;
color: white;
font-size: 20px;
background-color: rgba(0, 0, 0, 0.6);
padding: 5px 10px;
border-radius: 5px;
z-index: 100000;
}
.ig-close-button {
position: fixed;
top: 0;
right: 0;
padding: 0;
background-color: transparent;
border: none;
}
.ig-close-button:hover {
background-color: rgba(0, 0, 0, 0.7);
filter: brightness(85%);
}
.ig-nav-button {
position: fixed;
background-color: transparent;
border: none;
padding: 0;
width: auto;
height: auto;
cursor: pointer;
display: flex;
align-items: center;
justify-content: center;
z-index: 10001;
top: 50%;
transform: translateY(-50%);
}
.ig-nav-button:hover {
background-color: rgba(0, 0, 0, 0.7);
filter: brightness(85%);
}
.ig-nav-button:active {
background-color: transparent;
}
.ig-nav-button.ig-prev-button {
left: -5px;
top: 50%;
transform: translateY(-50%);
}
.ig-nav-button.ig-prev-button:active .button-icon {
transform: scale(0.95);
}
.ig-nav-button.ig-next-button {
right: -5px;
top: 50%;
transform: translateY(-50%);
}
.ig-nav-button.ig-next-button:active .button-icon {
transform: scale(0.95);
}
#ig-thumbnailBar {
position: fixed;
bottom: 0;
left: 50%;
right: 0;
height: 75px;
transform: translateX(-50%);
display: flex;
overflow-x: scroll;
overflow-y: hidden;
background-color: rgba(0, 0, 0, 0.6);
padding: 10px 0;
white-space: nowrap;
scrollbar-width: thin;
scrollbar-color: #444 #282A36;
}
#ig-thumbnailBar::-webkit-scrollbar {
height: 12px;
background: #282A36;
}
#ig-thumbnailBar::-webkit-scrollbar-track {
background: #282A36;
}
#ig-thumbnailBar::-webkit-scrollbar-thumb {
background-color: #444;
border-radius: 10px;
border: 3px solid #282A36;
}
#ig-thumbnailBar::-webkit-scrollbar-thumb:hover {
background: #555;
}
.ig-thumbnail {
height: 60px;
object-fit: cover;
margin: 0 5px;
cursor: pointer;
transition: transform 0.3s ease, outline 0.3s ease;
}
.ig-thumbnail:hover {
opacity: 0.7;
}
.ig-thumbnail.ig-active {
transform: scale(1.05);
outline: 3px solid green;
}
.ig-download-button {
position: fixed;
top: 5px;
width: 40px;
height: 40px;
cursor: pointer;
display: flex;
align-items: center;
justify-content: center;
border: none;
padding: 0;
background-color: transparent;
transition: background-color 0.3s ease, transform 0.3s ease;
z-index: 10002;
}
.ig-download-button .button-icon {
width: 100%;
height: 100%;
transition: transform 0.3s ease, opacity 0.3s ease;
}
.ig-download-button:hover {
background-color: transparent;
box-shadow: none;
}
.ig-download-button:hover .button-icon {
transform: scale(1.05);
opacity: 0.7;
}
.ig-download-button:active .button-icon {
transform: scale(0.95);
`);
const prefix = 'ig-';
let hoveredMediaLink = null;
let hoveredMediaFilename = null;
let images = [];
let currentIndex = 0;
let galleryContainer, galleryImage, counter, thumbnailBar;
let loading = false;
const preview = document.createElement('div');
preview.id = 'hover-preview';
document.body.appendChild(preview);
function togglePostVisibility() {
const posts = document.querySelectorAll('.post_wrapper');
posts.forEach(post => {
const hasImage = post.querySelector('.post_file') !== null;
if (!hasImage) {
if(post.getAttribute('data-hidden') === 'true') {
post.removeAttribute('data-hidden');
} else {
post.setAttribute('data-hidden', 'true');
}
}
});
}
function addPageSpecificClass() {
const urlPath = window.location.pathname;
const bodyClass = document.body.classList;
if(urlPath.includes('/search/')) {
bodyClass.add('desux-search-page');
} else if(urlPath.match(/\/\w+\/thread\/\d+/)) {
bodyClass.add('desux-thread-page');
}
}
function attachHoverPreviewAndDownload() {
document.querySelectorAll('.thread .thread_image_link, .post_wrapper .thread_image_link').forEach(anchor => {
anchor.addEventListener('mouseover', function() {
const href = this.href;
const isVideo = href.endsWith('.webm') || href.endsWith('.mp4');
preview.innerHTML = '';
const media = isVideo ? document.createElement('video') : document.createElement('img');
media.src = href;
if (isVideo) {
media.autoplay = true;
media.loop = true;
media.muted = true;
}
preview.appendChild(media);
preview.style.display = 'block';
const postContainer = this.closest('.post_wrapper') || this.closest('.thread');
const filenameElement = postContainer.querySelector('.post_file_filename');
hoveredMediaLink = href;
hoveredMediaFilename = filenameElement ? (filenameElement.getAttribute('title') || filenameElement.textContent).trim() : getFilenameFromUrl(href);
});
anchor.addEventListener('mouseout', function() {
preview.innerHTML = '';
preview.style.display = 'none';
hoveredMediaLink = null;
hoveredMediaFilename = null;
});
});
}
function getFilenameFromUrl(url) {
return url.split('/').pop();
}
const closeImage = '';
const downloadImage = '';
const navigateLeftImage = '';
const navigateRightImage = '';
function collectMediaItems() {
const mediaLinks = document.querySelectorAll('.thread .thread_image_link, .post_wrapper .thread_image_link');
mediaLinks.forEach((mediaLink) => {
const postWrapper = mediaLink.closest('.post_wrapper, .thread');
if (postWrapper) {
const isVideo = mediaLink.href.endsWith('.webm');
const thumbnail = mediaLink.querySelector('img').src;
const postLink = postWrapper.querySelector('a[data-function="quote"]');
const postId = postLink ? postLink.getAttribute('data-post') : postWrapper.id;
images.push({
src: mediaLink.href,
isVideo,
thumbnail,
postId
});
}
});
}
function getCurrentPageNumber() {
const matches = window.location.pathname.match(/page\/(\d+)/);
const pageNumber = matches ? parseInt(matches[1], 10) : 1;
// console.log('Current page number:', pageNumber);
return pageNumber;
}
let currentPageNumber = getCurrentPageNumber();
if (window.location.pathname.includes('/search/')) {
$('#footer').hide();
}
function loadMoreContent() {
if (loading || !window.location.pathname.includes('/search/')) return;
loading = true;
// console.log('Loading more content...');
const nextPageUrl = constructNextPageUrl(currentPageNumber);
// console.log('Next page URL:', nextPageUrl);
$.ajax({
url: nextPageUrl,
type: 'GET',
success: function(response) {
// console.log('Content loaded successfully');
const $response = $(response);
$response.find('article.backlink_container, section.section_title, h3.section_title, div.paginate').remove();
const newContent = $response.find('.thread').parent();
if (newContent.length === 0) {
// console.log('No more content to load');
$('#footer').show();
loading = false;
return;
}
$('.thread').last().parent().append(newContent.html());
attachHoverPreviewAndDownload();
collectMediaItems();
currentPageNumber++;
loading = false;
},
error: function(xhr, status, error) {
// console.error('Error loading content:', status, error);
loading = false;
}
});
}
function constructNextPageUrl(currentPageNumber) {
let basePath = window.location.href;
let nextPageNumber = currentPageNumber + 1;
// console.log('Constructing URL for page:', nextPageNumber);
if (basePath.includes('/page/')) {
basePath = basePath.replace(/\/page\/\d+/, `/page/${nextPageNumber}`);
} else {
if (!basePath.endsWith('/')) {
basePath += '/';
}
basePath += `page/${nextPageNumber}/`;
}
// console.log('Next page URL:', basePath);
return basePath;
}
function createGallery() {
if (galleryContainer) {
galleryContainer.remove();
galleryContainer = null;
return;
}
galleryContainer = GM_addElement(document.body, 'div', { id: prefix + 'galleryContainer' });
galleryImage = GM_addElement(galleryContainer, 'img', { id: prefix + 'galleryImage' });
counter = GM_addElement(galleryContainer, 'div', { id: prefix + 'imageCounter' });
const closeButton = createButton(closeImage, closeGallery, prefix + 'close-button');
const downloadButton = createButton(downloadImage, downloadCurrentMedia, prefix + 'download-button');
const nextButton = createButton(navigateRightImage, () => navigateGallery(1), prefix + 'nav-button', prefix + 'next-button');
const prevButton = createButton(navigateLeftImage, () => navigateGallery(-1), prefix + 'nav-button', prefix + 'prev-button');
thumbnailBar = GM_addElement(galleryContainer, 'div', { id: prefix + 'thumbnailBar' });
images.forEach((media, index) => {
const thumb = GM_addElement(thumbnailBar, 'img', {
src: media.thumbnail,
class: prefix + 'thumbnail'
});
thumb.addEventListener('click', () => {
currentIndex = index;
updateGallery();
});
});
galleryContainer.append(counter, closeButton, downloadButton, nextButton, prevButton, thumbnailBar);
updateGallery();
}
function createButton(base64Image, onClick, ...classes) {
const button = GM_addElement(document.body, 'button', { class: classes.join(' ') });
button.addEventListener('click', onClick);
GM_addElement(button, 'img', {
src: base64Image,
class: 'button-icon'
});
return button;
}
function navigateGallery(direction) {
const totalImages = images.length;
currentIndex = (currentIndex + direction + totalImages) % totalImages;
updateGallery();
}
function closeGallery() {
if (galleryContainer) {
galleryContainer.remove();
galleryContainer = null;
}
}
function updateGallery() {
if (images.length > 0 && galleryContainer) {
const currentMedia = images[currentIndex];
if (galleryContainer.contains(galleryImage)) {
galleryContainer.removeChild(galleryImage);
}
galleryImage = currentMedia.isVideo ? GM_addElement(galleryContainer, 'video', { id: prefix + 'galleryImage', controls: true, autoplay: true, loop: true, muted: true }) : GM_addElement(galleryContainer, 'img', { id: prefix + 'galleryImage' });
galleryImage.src = currentMedia.src;
counter.textContent = `${currentIndex + 1}/${images.length}`;
updateThumbnails();
}
}
function updateThumbnails() {
const thumbnails = Array.from(thumbnailBar.children);
thumbnails.forEach((thumb, index) => {
thumb.classList.toggle(prefix + 'active', index === currentIndex);
});
const selectedThumbnail = thumbnails[currentIndex];
thumbnailBar.scrollLeft = selectedThumbnail.offsetLeft - thumbnailBar.offsetWidth / 2 + selectedThumbnail.offsetWidth / 2;
}
function getOriginalFilename(postId) {
let postElement = document.querySelector(`.thread[id="${postId}"]`);
let filenameLink;
if (postElement) {
filenameLink = postElement.querySelector('.post_file_filename');
} else {
postElement = document.querySelector(`.post_wrapper a[data-function="quote"][data-post="${postId}"]`);
const postWrapperParent = postElement ? postElement.closest('.post_wrapper') : null;
filenameLink = postWrapperParent ? postWrapperParent.querySelector('.post_file_filename') : null;
}
if (filenameLink) {
if (filenameLink.getAttribute('title')) {
return filenameLink.getAttribute('title').trim();
} else if (filenameLink.textContent) {
return filenameLink.textContent.trim();
}
}
const currentMedia = images.find(img => img.postId === postId);
if (currentMedia && currentMedia.src) {
const urlParts = currentMedia.src.split('.');
const ext = urlParts[urlParts.length - 1];
return `default-filename.${ext}`;
}
return "default-filename";
}
function downloadCurrentMedia() {
const currentMedia = images[currentIndex];
let postId = currentMedia.postId;
if (!postId || postId === document.querySelector('.thread').id) {
postId = document.querySelector('.thread').id;
}
const originalFilename = getOriginalFilename(postId);
GM_xmlhttpRequest({
method: 'GET',
url: currentMedia.src,
responseType: 'blob',
onload: function(response) {
saveBlob(response.response, originalFilename);
},
onerror: function(error) {
console.error('Error downloading file:', error);
}
});
}
function saveBlob(blob, filename) {
const a = GM_addElement(document.body, "a", { href: window.URL.createObjectURL(blob), download: filename });
a.click();
document.body.removeChild(a);
}
document.addEventListener('keydown', function(e) {
if (e.key.toLowerCase() === 's' && !/input|textarea/i.test(document.activeElement.tagName)) {
e.preventDefault();
if (galleryContainer && galleryContainer.style.display !== 'none') {
downloadCurrentMedia();
} else if (hoveredMediaLink) {
GM_xmlhttpRequest({
method: 'GET',
url: hoveredMediaLink,
responseType: 'blob',
onload: function(response) {
const blobUrl = URL.createObjectURL(response.response);
const downloadLink = document.createElement('a');
downloadLink.href = blobUrl;
downloadLink.download = hoveredMediaFilename || 'download';
document.body.appendChild(downloadLink);
downloadLink.click();
document.body.removeChild(downloadLink);
URL.revokeObjectURL(blobUrl);
},
onerror: function() {
alert('Download failed.');
}
});
}
}
if (e.altKey && e.key.toLowerCase() === 'g') {
if (!images.length) collectMediaItems();
createGallery();
}
if (e.key === 'Escape' && galleryContainer && galleryContainer.style.display !== 'none') {
closeGallery();
}
if (galleryContainer && galleryContainer.style.display !== 'none') {
if (e.key === 'ArrowRight') {
navigateGallery(1);
} else if (e.key === 'ArrowLeft') {
navigateGallery(-1);
}
}
if ((e.key === 'F' || e.key === 'f') && !e.ctrlKey && !e.altKey && !e.metaKey && !e.shiftKey &&
e.target.tagName !== 'INPUT' && e.target.tagName !== 'TEXTAREA' &&
/https:\/\/desuarchive\.org\/.*\/thread\/.*/.test(window.location.href)) {
togglePostVisibility();
e.preventDefault();
}
});
window.addEventListener('scroll', function() {
if (window.location.pathname.includes('/search/') && window.scrollY + window.innerHeight >= document.body.scrollHeight - 90) {
loadMoreContent();
}
});
addPageSpecificClass();
attachHoverPreviewAndDownload();
collectMediaItems();
})();