// ==UserScript==
// @name 4chan-masonry
// @namespace 0000xFFFF
// @version 1.3.3
// @description View all media (images+videos) from a 4chan thread in a masonry grid layout.
// @author 0000xFFFF
// @license MIT
// @match *://boards.4chan.org/*/thread/*
// @match *://boards.4channel.org/*/thread/*
// @grant GM_addStyle
// @icon 
// ==/UserScript==
const GRID_ROWS_MIN = 1;
const GRID_ROWS_MAX = 15;
const GRID_ROWS_DEFAULT = 4;
const CONCURRENT_LOADS_IMAGE = 3;
const CONCURRENT_LOADS_VIDEO = 1;
const LOAD_DELAY_IMAGE = 10;
const LOAD_DELAY_VIDEO = 300;
const PRELOAD_VIEWPORT_BUFFER = 200; // Load images within 200px of viewport
function GM_addStyle(css) {
const style = document.createElement("style");
style.textContent = css;
(document.head || document.documentElement).appendChild(style);
return style;
}
// Usage
var MasonryCss = `
#fcm_overlay {
position: fixed;
top: 0;
left: 0;
width: 100vw;
height: 100vh;
background: rgba(0,0,0,0.95);
z-index: 10000;
overflow: auto;
box-sizing: border-box;
}
#fcm_topbar {
position: fixed;
top: 0;
left: 0;
right: 0;
display: flex;
justify-content: space-between;
align-items: center;
padding: 15px 20px;
background-color: rgba(0, 0, 0, 0.8);
color: #fff;
z-index: 999;
transition: transform 0.3s ease-in-out;
will-change: transform;
backdrop-filter: blur(5px);
}
.fcm_slider_container {
display: flex;
align-items: center;
gap: 15px;
}
.fcm_close {
padding: 8px 12px;
background: #d32f2f;
color: white;
border: none;
border-radius: 8px;
cursor: pointer;
font-size: 14px;
font-weight: bold;
transition: all 0.2s ease;
min-width: 40px;
display: flex;
align-items: center;
justify-content: center;
}
.fcm_close:hover {
background: #b71c1c;
transform: scale(1.05);
}
#fcm_masonry {
display: block;
gap: 10px;
margin-top: 90px;
column-gap: 5px;
}
.fcm_shortcut_4chanx {
margin: 0 0 0 5px;
padding: 0;
cursor: pointer;
justify-content: center;
}
.fcm_shortcut_4chanx img {
width: 16px;
height: 13px;
}
.fcm_button_regular {
padding: 12px 18px;
display: flex;
gap: 5px;
background: #2d5016;
color: white;
border: none;
border-radius: 8px;
cursor: pointer;
font-size: 14px;
font-weight: bold;
box-shadow: 0 4px 15px rgba(0,0,0,0.3);
transition: all 0.3s ease;
white-space: nowrap;
}
.fcm_button_regular:hover {
background: #4a7c21;
transform: translateY(-2px);
box-shadow: 0 6px 20px rgba(0,0,0,0.4);
}
.fcm_button_regular img {
height: 15px;
}
.fcm_container {
display: flex;
margin: 15px 0 15px 0;
}
.fcm_grid_container {
max-width: 97vw;
margin: 0 auto;
}
.fcm_value_display {
color: white;
font-weight: bold;
font-size: 16px;
min-width: 20px;
}
.fcm_media_wrapper {
position: relative;
overflow: hidden;
border-radius: 8px;
cursor: pointer;
transition: transform 0.1s ease, box-shadow 0.1s ease;
}
.fcm_media_wrapper:hover {
transform: scale(1.05);
box-shadow: 0 8px 25px rgba(0,0,0,0.6);
z-index: 100;
}
.fcm_media_img,
.fcm_media_thumb,
.fcm_media_video {
width: 100%;
display: block;
object-fit: cover;
cursor: pointer;
}
.fcm_media_loading, .fcm_media_thumb {
}
.fcm_media_img {
object-fit: contain;
transition: opacity 0.3s ease;
}
.fcm_media_video {
object-fit: contain;
display: none;
}
.fcm_play_btn {
position: absolute;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
width: 50px;
height: 50px;
background: rgba(0,0,0,0.5);
border-radius: 50%;
display: flex;
align-items: center;
justify-content: center;
font-size: 24px;
color: white;
pointer-events: none;
opacity: 0.4;
}
.fcm_tooltip {
position: absolute;
bottom: 0;
left: 0;
right: 0;
background: rgba(0,0,0,0.8);
color: white;
padding: 8px;
font-size: 12px;
transform: translateY(100%);
transition: transform 0.3s ease;
word-break: break-all;
}
.fcm_media_wrapper:hover .fcm_tooltip {
transform: translateY(0);
}
`;
GM_addStyle(MasonryCss);
const userscript_icon_1 = ""
const userscript_icon_2 = "";
const userscript_icon_3 = ""
let gridRows = GRID_ROWS_DEFAULT;
let isGridOpen = false;
let gridOverlay = null;
let loadQueue = [];
let activeLoadsImage = 0;
let activeLoadsVideo = 0;
let preloadCache = new Map(); // Cache to avoid duplicate requests
// Rate limiting function
function delay(ms) {
return new Promise(resolve => setTimeout(resolve, ms));
}
// Check if element is near viewport
function isNearViewport(element, buffer = PRELOAD_VIEWPORT_BUFFER) {
const rect = element.getBoundingClientRect();
const windowHeight = window.innerHeight;
const windowWidth = window.innerWidth;
return (
rect.bottom >= -buffer &&
rect.top <= windowHeight + buffer &&
rect.right >= -buffer &&
rect.left <= windowWidth + buffer
);
}
// Queue-based image preloader
async function preloadMedia(url, priority = 'low', type = 'image') {
if (preloadCache.has(url)) {
return preloadCache.get(url);
}
return new Promise((resolve, reject) => {
loadQueue.push({ url, resolve, reject, priority, type });
processLoadQueue();
});
}
// Process the load queue with rate limiting
async function processLoadQueue() {
if ((activeLoadsImage >= CONCURRENT_LOADS_IMAGE || activeLoadsVideo >= CONCURRENT_LOADS_VIDEO) || loadQueue.length === 0) {
return;
}
// Sort by priority (high priority first)
loadQueue.sort((a, b) => {
const priorityOrder = { 'high': 3, 'medium': 2, 'low': 1 };
return priorityOrder[b.priority] - priorityOrder[a.priority];
});
const item = loadQueue.shift();
let delayMs = 0;
if (item.type === 'image') {
activeLoadsImage++;
delayMs = LOAD_DELAY_IMAGE;
}
else {
activeLoadsVideo++;
delayMs = LOAD_DELAY_VIDEO;
}
try {
await delay(delayMs); // Rate limiting delay
const result = await loadMedia(item.url, item.type);
preloadCache.set(item.url, result);
item.resolve(result);
} catch (error) {
console.warn(`Failed to load ${item.url}:`, error);
item.reject(error);
} finally {
if (item.type === 'image') {
activeLoadsImage--;
}
else {
activeLoadsVideo--;
}
// Process next item in queue
setTimeout(processLoadQueue, 50);
}
}
function updateQueuePriority(url, newPriority) {
const item = loadQueue.find(i => i.url === url);
if (item) {
item.priority = newPriority;
}
}
function updatePriorities() {
document.querySelectorAll('.fcm_media_img').forEach(img => {
const url = img.dataset.fullUrl;
if (!url) return; // Already loaded
if (preloadCache.has(url)) return; // Already done
const item = loadQueue.find(i => i.url === url);
if (item) {
item.priority = isNearViewport(img) ? 'medium' : 'low';
}
});
}
function throttle(fn, delay) {
let lastCall = 0;
let timeout;
return function (...args) {
const now = Date.now();
if (now - lastCall < delay) {
clearTimeout(timeout);
timeout = setTimeout(() => {
lastCall = Date.now();
fn.apply(this, args);
}, delay - (now - lastCall));
} else {
lastCall = now;
fn.apply(this, args);
}
};
}
const throttledUpdatePriorities = throttle(updatePriorities, 800);
window.addEventListener('scroll', throttledUpdatePriorities);
window.addEventListener('resize', throttledUpdatePriorities);
// Actual media loading function
function loadMedia(url, type) {
return new Promise((resolve, reject) => {
if (type === 'image') {
const img = new Image();
const timeout = setTimeout(() => {
reject(new Error('Image load timeout'));
}, 10000); // 10 second timeout
img.onload = () => {
clearTimeout(timeout);
resolve(img);
};
img.onerror = () => {
clearTimeout(timeout);
reject(new Error('Image load failed'));
};
img.src = url;
} else if (type === 'video') {
const video = document.createElement('video');
const timeout = setTimeout(() => {
reject(new Error('Video load timeout'));
}, 15000); // 15 second timeout for videos
video.addEventListener('loadeddata', () => {
clearTimeout(timeout);
resolve(video);
});
video.addEventListener('error', () => {
clearTimeout(timeout);
reject(new Error('Video load failed'));
});
video.src = url;
video.preload = 'metadata'; // Only load metadata initially
}
});
}
// Intersection Observer for lazy loading
const imageObserver = new IntersectionObserver((entries) => {
entries.forEach(entry => {
if (entry.isIntersecting) {
const img = entry.target;
const fullUrl = img.dataset.fullUrl;
if (fullUrl && !img.dataset.loading) {
img.dataset.loading = 'true';
preloadMedia(fullUrl, isNearViewport(img) ? 'medium' : 'low').then((fullImg) => {
img.style.opacity = '0';
setTimeout(() => {
img.src = fullImg.src;
img.style.opacity = '1';
img.removeAttribute('data-full-url');
img.removeAttribute('data-loading');
img.classList.remove("fcm_media_loading");
}, 200);
}).catch(() => {
img.removeAttribute('data-loading');
img.classList.remove("fcm_media_loading");
});
}
}
});
}, {
rootMargin: '200px' // Start loading 200px before entering viewport
});
function createOptimizedVideoElement(mediaData, thumbImg, playBtn, mediaWrapper) {
let video = null;
let videoLoaded = false;
let hoverTimeout = null;
let isHovering = false;
const createVideo = () => {
if (!video) {
video = document.createElement('video');
video.className = "fcm_media_video";
video.loop = true;
video.muted = true;
video.playsInline = true;
video.style.display = 'none';
video.addEventListener('canplay', () => {
videoLoaded = true;
if (isNearViewport(video) && !video.controls) {
showVideo();
}
});
video.addEventListener('loadstart', () => {
console.log('Video loading started:', mediaData.url);
});
mediaWrapper.appendChild(video);
}
return video;
};
const showVideo = () => {
if (video && videoLoaded) {
thumbImg.style.display = 'none';
playBtn.style.display = 'none';
video.style.display = 'block';
video.play().catch(() => { });
}
};
const hideVideo = () => {
if (video && !video.controls) {
video.pause();
video.style.display = 'none';
thumbImg.style.display = 'block';
playBtn.style.display = 'flex';
}
};
// Hover enter - start loading after delay
mediaWrapper.addEventListener('mouseenter', () => {
isHovering = true;
clearTimeout(hoverTimeout);
hoverTimeout = setTimeout(() => {
if (isHovering) { // Double-check still hovering
const vid = createVideo();
if (videoLoaded) {
// Video already loaded, show immediately
showVideo();
}
else if (!vid.src) {
// Queue the video load instead of hitting server immediately
preloadMedia(mediaData.url, 'high', 'video')
.then((videoEl) => {
vid.src = mediaData.url;
})
.catch(() => {
console.warn("Failed to queue video load:", mediaData.url);
});
}
// If video is loading, showVideo() will be called from 'canplay' event
}
}, 100); // Reduced delay for better responsiveness
});
// Hover leave - hide video and cancel loading if needed
mediaWrapper.addEventListener('mouseleave', () => {
isHovering = false;
clearTimeout(hoverTimeout);
//hideVideo();
});
// Click handler - permanent video with controls
mediaWrapper.addEventListener('click', (e) => {
if (!video || video.controls) return; // Already clicked or no video
e.preventDefault();
const vid = createVideo();
// Set up for permanent display
vid.muted = false;
vid.controls = true;
vid.style.display = 'block';
thumbImg.style.display = 'none';
playBtn.style.display = 'none';
if (!vid.src) {
vid.src = mediaData.url;
}
// Play when ready
if (videoLoaded) {
vid.play().catch(() => { });
} else {
vid.addEventListener('canplay', () => {
vid.play().catch(() => { });
}, { once: true });
}
});
}
// Replace the image creation part in createMasonryGrid function
function createOptimizedMediaElement(mediaData) {
const mediaWrapper = document.createElement('div');
mediaWrapper.className = "fcm_media_wrapper";
if (mediaData.isVideo) {
// Thumbnail image
const thumbImg = document.createElement('img');
thumbImg.src = mediaData.thumbnail;
thumbImg.className = "fcm_media_thumb";
thumbImg.loading = 'lazy';
mediaWrapper.appendChild(thumbImg);
// Play button overlay
const playBtn = document.createElement('div');
playBtn.className = "fcm_play_btn";
playBtn.innerHTML = '►';
mediaWrapper.appendChild(playBtn);
// Use optimized video creation
createOptimizedVideoElement(mediaData, thumbImg, playBtn, mediaWrapper);
} else {
const img = document.createElement('img');
img.className = "fcm_media_img fcm_media_loading";
img.src = mediaData.thumbnail;
img.loading = 'lazy';
img.dataset.fullUrl = mediaData.url;
mediaWrapper.addEventListener('mouseenter', () => {
updateQueuePriority(mediaData.url, 'high');
});
// Use Intersection Observer for lazy loading
imageObserver.observe(img);
mediaWrapper.appendChild(img);
}
// Filename tooltip
const tooltip = document.createElement('div');
tooltip.textContent = mediaData.originalName;
tooltip.className = "fcm_tooltip";
// Middle click handler
mediaWrapper.addEventListener('mousedown', (event) => {
if (event.button === 1) {
window.open(mediaData.url, '_blank');
}
});
mediaWrapper.appendChild(tooltip);
return mediaWrapper;
}
// Add cleanup function for when grid is closed
function cleanupPreloading() {
// Cancel pending loads
loadQueue.forEach(item => {
item.reject(new Error('Cleanup cancelled'));
});
loadQueue = [];
activeLoads = 0;
// this is handles with refresh
// // Clear cache periodically to prevent memory leaks
// if (preloadCache.size > 100) {
// preloadCache.clear();
// }
// Disconnect observer
imageObserver.disconnect();
}
function initUI() {
const button = document.createElement('span');
button.title = "Masonry Grid";
button.addEventListener('click', function(e) {
e.preventDefault();
openGrid();
});
const forchanX_header = document.getElementById("header-bar")
if (forchanX_header) { // if 4chan X is detected
const element = document.getElementById("shortcut-watcher");
button.id = "shortcut-masonry";
button.className = "shortcut brackets-wrap fcm_shortcut_4chanx";
const img = document.createElement("img");
img.src = userscript_icon_1;
button.appendChild(img);
img.addEventListener('mouseover', () => {
if (!img.classList.contains('clicked')) {
img.src = userscript_icon_2;
}
});
img.addEventListener('mouseout', () => {
if (!img.classList.contains('clicked')) {
img.src = userscript_icon_1;
}
});
img.addEventListener('mousedown', () => {
img.classList.add('pressed');
img.src = userscript_icon_3;
});
img.addEventListener('mouseup', () => {
img.classList.remove('pressed');
if (img.matches(':hover')) {
img.src = userscript_icon_2;
} else {
img.src = userscript_icon_1;
}
});
// also handle case where mouse is released outside the image
document.addEventListener('mouseup', () => {
if (img.classList.contains('pressed')) {
img.classList.remove('pressed');
if (img.matches(':hover')) {
img.src = 'icon2.png';
} else {
img.src = 'icon1.png';
}
}
});
forchanX_header.appendChild(button);
element.parentElement.insertBefore(button, element);
}
else {
button.className = "fcm_button_regular";
const img = document.createElement("img");
img.src = userscript_icon_3;
button.appendChild(img);
const span = document.createElement("span");
span.innerHTML = "Masonry Grid";
button.appendChild(span);
const containerDiv = document.createElement('div');
containerDiv.id = "4chan_grid_cont";
containerDiv.className = "fcm_container";
containerDiv.appendChild(button);
const threadElement = document.querySelector(".thread");
threadElement.parentElement.insertBefore(containerDiv, threadElement);
}
findMediaLinks(button);
}
function findMediaLinks(button = null) {
const mediaLinks = [];
const files = document.querySelectorAll('div.file');
files.forEach((fileDiv, index) => {
const fileText = fileDiv.querySelector('.fileText');
const fileThumb = fileDiv.querySelector('.fileThumb');
const fileThumbImage = fileThumb.querySelector("img");
let link = null;
if (fileText) { link = fileText.querySelector('a'); }
if (link && link.href) {
const url = link.href.startsWith('//') ? 'https:' + link.href : link.href;
const isImage = /\.(jpg|jpeg|png|gif|webp|bmp|svg)(\?|$)/i.test(url);
const isVideo = /\.(mp4|webm|mkv|avi|mov)(\?|$)/i.test(url);
if (isImage || isVideo) {
const postId = url.split('/').pop().split('?')[0];
let originalName = link.title.trim() || link.textContent.trim() || postId;
// if 4chan-X is used fix the name fetching
const fnfull = link.querySelector('.fnfull');
if (fnfull) { originalName = fnfull.textContent.trim(); }
// get file info text
const fileInfo = fileDiv.querySelector(".file-info");
let width = null;
let height = null;
if (fileInfo) {
const fileInfoClone = fileInfo.cloneNode(true);
fileInfoClone.querySelectorAll("a").forEach(a => a.remove());
const info = fileInfoClone.textContent.trim();
const size = info.split(", ")[1].replace(")", "");
const width_and_height = size.split("x");
width = width_and_height[0];
height = width_and_height[1];
}
const newElement = {
url: url,
originalName: originalName,
postId: postId,
index: index + 1,
isVideo: isVideo,
thumbnail: (fileThumbImage.src),
width: width,
height: height
};
mediaLinks.push(newElement);
}
}
});
if (button) {
button.title = `Masonry Grid (${mediaLinks.length})`;
}
return mediaLinks;
}
function findMediaLinksFromImgAndVideoElements() {
const mediaLinks = []
const imgElements = document.querySelectorAll('img[src*="jpg"], img[src*="jpeg"], img[src*="png"], img[src*="gif"], img[src*="webp"], img[src*="bmp"]');
const videoElements = document.querySelectorAll('video[src*="mp4"], video[src*="webm"], video[src*="mkv"], video[src*="avi"], video[src*="mov"]');
const mediaElements = [...imgElements, ...videoElements];
mediaElements.forEach((img_or_vid, index) => {
const url = img_or_vid.src;
const filename = url.split('/').pop().split('?')[0];
const isVideo = /\.(mp4|webm|mkv|avi|mov)(\?|$)/i.test(url);
mediaLinks.push({
url: url,
originalName: filename,
postId: filename,
index: index + 1,
isVideo: isVideo
});
});
return mediaLinks;
}
function updateMasonryGrid() {
const gridContainer = document.getElementById('fcm_masonry');
if (gridContainer) {
gridContainer.style.columnCount = gridRows;
}
}
function createTopBar() {
const topbar = document.createElement("div");
topbar.id = "fcm_topbar";
const sliderContainer = document.createElement('div');
sliderContainer.className = "fcm_slider_container";
const slider = document.createElement('input');
slider.type = 'range';
slider.id = "setting_slider_cols";
slider.min = GRID_ROWS_MIN;
slider.max = GRID_ROWS_MAX;
slider.step = "1";
slider.value = gridRows;
const valueDisplay = document.createElement('span');
valueDisplay.textContent = gridRows;
valueDisplay.className = "fcm_value_display";
slider.addEventListener('input', (e) => {
gridRows = parseInt(e.target.value);
valueDisplay.textContent = gridRows;
updateMasonryGrid();
});
function slider_left() { slider.value = Math.max(Number(slider.value) - Number(slider.step), slider.min); }
function slider_right() { slider.value = Math.min(Number(slider.value) + Number(slider.step), slider.max); }
slider.addEventListener('keydown', (event) => {
switch (event.key) {
case 'ArrowLeft': slider_left(); break;
case 'ArrowRight': slider_right(); break;
}
});
slider.focus();
slider.select();
sliderContainer.appendChild(slider);
sliderContainer.appendChild(valueDisplay);
topbar.appendChild(sliderContainer);
// Close button
const closeButton = document.createElement('button');
closeButton.className = "fcm_close";
closeButton.innerHTML = '✕';
closeButton.addEventListener('click', closeGrid);
topbar.appendChild(closeButton);
return topbar;
}
function createMasonryGrid(mediaLinks) {
const overlay = document.createElement('div');
overlay.id = 'fcm_overlay';
const topbar = createTopBar();
let lastScroll = 0;
overlay.addEventListener("scroll", () => {
const currentScroll = overlay.scrollTop;
if (currentScroll > lastScroll) { topbar.style.transform = "translateY(-100%)"; }
else { topbar.style.transform = "translateY(0)"; }
lastScroll = currentScroll;
});
overlay.appendChild(topbar);
const container = document.createElement('div');
container.className = "fcm_grid_container";
const handleParentClick = (event) => {
event.preventDefault();
if (event.target === event.currentTarget) {
closeGrid();
}
};
overlay.addEventListener('click', handleParentClick);
container.addEventListener('click', handleParentClick);
topbar.addEventListener('click', handleParentClick);
// Grid container
const gridContainer = document.createElement('div');
gridContainer.id = 'fcm_masonry';
gridContainer.style.columnCount = gridRows;
// Create image elements
mediaLinks.forEach((mediaData, index) => {
const mediaWrapper = createOptimizedMediaElement(mediaData);
gridContainer.appendChild(mediaWrapper);
});
container.appendChild(gridContainer);
overlay.appendChild(container);
return overlay;
}
function openGrid() {
if (isGridOpen) return;
const mediaLinks = findMediaLinks();
if (mediaLinks.length === 0) {
alert('No media found on this page!');
return;
}
gridOverlay = createMasonryGrid(mediaLinks);
document.body.appendChild(gridOverlay);
isGridOpen = true;
// Prevent body scrolling
document.body.style.overflow = 'hidden';
// Close on escape key
document.addEventListener('keydown', handleEscapeKey);
}
function closeGrid() {
if (!isGridOpen || !gridOverlay) return;
cleanupPreloading();
document.body.removeChild(gridOverlay);
gridOverlay = null;
isGridOpen = false;
document.body.style.overflow = 'auto';
document.removeEventListener('keydown', handleEscapeKey);
}
function handleEscapeKey(e) {
if (e.key === 'Escape') {
closeGrid();
}
}
async function init() {
if (document.readyState === 'loading') {
document.addEventListener('DOMContentLoaded', init);
return;
}
setTimeout(async () => {
try {
initUI();
} catch (error) {
console.error('Error initializing userscript:', error);
}
}, 500);
}
init();