// ==UserScript==
// @name 4chan Gallery
// @namespace http://tampermonkey.net/
// @version 2025-01-12 (3.6)
// @description 4chan grid-based image gallery with zoom mode support for threads that allows you to browse images, and soundposts (images with sounds, webms with sounds) along with other utility features.
// @author TheDarkEnjoyer
// @match https://boards.4chan.org/*/thread/*
// @match https://boards.4chan.org/*/archive
// @match https://boards.4channel.org/*/thread/*
// @match https://boards.4channel.org/*/archive
// @match https://warosu.org/*/*
// @match https://archived.moe/*/*
// @match https://archive.palanq.win/*/*
// @match https://archive.4plebs.org/*/*
// @match https://desuarchive.org/*/*
// @match https://thebarchive.com/*/*
// @match https://archiveofsins.com/*/*
// @icon 
// @grant none
// @license GNU GPLv3
// ==/UserScript==
(function () {
"use strict";
// injectVideoJS();
const defaultSettings = {
Load_High_Res_Images_By_Default: {
value: false,
info: "When opening the gallery, load high quality images by default (no thumbnails)",
},
Add_Placeholder_Image_For_Zoom_Mode: {
value: true,
info: "Add a placeholder image for zoom mode so even if the thread has no images, you can still open the zoom mode",
},
Play_Webms_On_Hover: {
value: true,
info: "Autoplay webms on hover, pause on mouse leave",
},
Switch_Catbox_To_Pixstash_For_Soundposts: {
value: false,
info: "Switch all catbox.moe links to pixstash.moe links for soundposts",
},
Show_Arrow_Buttons_In_Zoom_Mode: {
value: true,
info: "Show clickable arrow buttons on screen edges in zoom mode",
},
Grid_Columns: {
value: 3,
info: "Number of columns in the grid view",
},
Grid_Cell_Max_Height: {
value: 200,
info: "Maximum height of each cell in pixels",
},
Embed_External_Links: {
value: false,
info: "Embed catbox/pixstash links found in post comments",
},
Strictly_Load_GIFs_As_Thumbnails_On_Hover: {
value: false,
info: "Only load GIF thumbnails until hovered"
},
Open_Close_Gallery_Key: {
value: "i",
info: "Key to open/close the gallery"
},
Hide_Gallery_Button: {
value: false,
info: "Hide the gallery button (You can still open the gallery with the keybind, default is 'i')"
},
};
let threadURL = window.location.href;
let lastScrollPosition = 0;
let gallerySize = { width: 0, height: 0 };
let gridContainer; // Add this line
// store settings in local storage
if (!localStorage.getItem("gallerySettings")) {
localStorage.setItem("gallerySettings", JSON.stringify(defaultSettings));
}
let settings = JSON.parse(localStorage.getItem("gallerySettings"));
// check if settings has all the keys from defaultSettings, if not, add the missing keys
let missingSetting = false;
for (const setting in defaultSettings) {
if (!settings.hasOwnProperty(setting)) {
settings[setting] = defaultSettings[setting];
missingSetting = true;
}
}
// update the settings in local storage if there are missing settings
if (missingSetting) {
localStorage.setItem("gallerySettings", JSON.stringify(settings));
}
function setStyles(element, styles) {
for (const property in styles) {
element.style[property] = styles[property];
}
}
function getPosts(websiteUrl, doc) {
switch (websiteUrl) {
case "warosu.org":
return doc.querySelectorAll(".comment, .highlight");
case "archived.moe":
case "archive.palanq.win":
case "archive.4plebs.org":
case "desuarchive.org":
case "thebarchive.com":
case "archiveofsins.com":
return doc.querySelectorAll(".post, .thread");
case "boards.4chan.org":
case "boards.4channel.org":
default:
return doc.querySelectorAll(".postContainer");
}
}
function getDocument(thread, threadURL) {
return new Promise((resolve, reject) => {
if (thread === threadURL) {
resolve(document);
} else {
fetch(thread)
.then((response) => response.text())
.then((html) => {
const parser = new DOMParser();
const doc = parser.parseFromString(html, "text/html");
resolve(doc);
})
.catch((error) => {
reject(error);
});
}
});
}
function injectVideoJS() {
const link = document.createElement("link");
link.href = "https://vjs.zencdn.net/8.10.0/video-js.css";
link.rel = "stylesheet";
document.head.appendChild(link);
// theme
const theme = document.createElement("link");
theme.href = "https://unpkg.com/@videojs/themes@1/dist/city/index.css";
theme.rel = "stylesheet";
document.head.appendChild(theme);
const script = document.createElement("script");
script.src = "https://vjs.zencdn.net/8.10.0/video.min.js";
document.body.appendChild(script);
("VideoJS injected successfully!");
}
function createArrowButton(direction) {
const button = document.createElement('button');
setStyles(button, {
position: 'fixed',
top: '50%',
[direction]: '20px',
transform: 'translateY(-50%)',
zIndex: '10001',
backgroundColor: 'rgba(28, 28, 28, 0.7)',
color: '#d9d9d9',
padding: '15px',
border: 'none',
borderRadius: '50%',
cursor: 'pointer',
display: settings.Show_Arrow_Buttons_In_Zoom_Mode.value ? 'block' : 'none'
});
button.innerHTML = direction === 'left' ? '◀' : '▶';
button.onclick = () => {
const event = new KeyboardEvent('keydown', { key: direction === 'left' ? 'ArrowLeft' : 'ArrowRight' });
document.dispatchEvent(event);
};
return button;
}
// Modify createMediaCell to accept mode and postURL parameters
function createMediaCell(url, commentText, mode, postURL, board, threadID, postID) {
if (!gridContainer) {
gridContainer = document.createElement("div");
setStyles(gridContainer, {
display: "grid",
gridTemplateColumns: `repeat(${settings.Grid_Columns.value}, 1fr)`,
gap: "10px",
padding: "20px",
backgroundColor: "#1c1c1c",
color: "#d9d9d9",
maxWidth: "80%",
maxHeight: "80%",
overflowY: "auto",
resize: "both",
overflow: "auto",
border: "1px solid #d9d9d9",
});
}
const cell = document.createElement("div");
setStyles(cell, {
border: "1px solid #d9d9d9",
position: "relative",
});
// Make the cell draggable
cell.draggable = true;
cell.addEventListener("dragstart", (e) => {
e.dataTransfer.setData("text/plain", [...gridContainer.children].indexOf(cell));
e.dataTransfer.dropEffect = "move";
});
// Allow drops on this cell
cell.addEventListener("dragover", (e) => {
e.preventDefault();
e.dataTransfer.dropEffect = "move";
});
cell.addEventListener("drop", (e) => {
e.preventDefault();
const draggedIndex = e.dataTransfer.getData("text/plain");
const containerChildren = [...gridContainer.children];
const draggedCell = containerChildren[draggedIndex];
if (draggedCell !== cell) {
const dropIndex = containerChildren.indexOf(cell);
if (draggedIndex < dropIndex) {
gridContainer.insertBefore(draggedCell, containerChildren[dropIndex].nextSibling);
} else {
gridContainer.insertBefore(draggedCell, containerChildren[dropIndex]);
}
}
});
const mediaContainer = document.createElement("div");
setStyles(mediaContainer, {
position: "relative",
display: "flex",
justifyContent: "center",
alignItems: "center",
});
const buttonDiv = document.createElement("div");
setStyles(buttonDiv, {
display: "flex",
justifyContent: "space-between",
alignItems: "center",
padding: "5px",
});
// Add view post button for external media
const viewPostButton = document.createElement("button");
viewPostButton.textContent = "View Original";
setStyles(viewPostButton, {
backgroundColor: "#1c1c1c",
color: "#d9d9d9",
padding: "5px 10px",
borderRadius: "3px",
border: "none",
cursor: "pointer",
boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
});
viewPostButton.addEventListener("click", () => {
window.location.href = postURL;
gallerySize = {
width: gridContainer.offsetWidth,
height: gridContainer.offsetHeight,
};
});
buttonDiv.appendChild(viewPostButton);
if (url.match(/\.(webm|mp4)$/i)) {
const video = document.createElement("video");
video.src = url;
video.controls = true;
video.title = commentText;
video.setAttribute("fileName", url.split('/').pop());
video.setAttribute("board", board);
video.setAttribute("threadID", threadID);
video.setAttribute("postID", postID);
setStyles(video, {
maxWidth: "100%",
maxHeight: `${settings.Grid_Cell_Max_Height.value}px`,
objectFit: "contain",
cursor: "pointer",
});
mediaContainer.appendChild(video);
const openInNewTabButton = document.createElement("button");
openInNewTabButton.textContent = "Open";
setStyles(openInNewTabButton, {
backgroundColor: "#1c1c1c",
color: "#d9d9d9",
padding: "5px 10px",
borderRadius: "3px",
border: "none",
cursor: "pointer",
boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
});
openInNewTabButton.onclick = () => {
window.open(url, "_blank");
};
buttonDiv.appendChild(openInNewTabButton);
} else if (url.match(/\.(jpg|jpeg|png|gif)$/i)) {
// Only create image cell if mode is "all"
if (mode === "all") {
const image = document.createElement("img");
image.src = url;
image.title = commentText;
image.setAttribute("fileName", url.split('/').pop());
image.setAttribute("actualSrc", url);
image.setAttribute("thumbnailUrl", url);
image.setAttribute("board", board);
image.setAttribute("threadID", threadID);
image.setAttribute("postID", postID);
setStyles(image, {
maxWidth: "100%",
maxHeight: `${settings.Grid_Cell_Max_Height.value}px`,
objectFit: "contain",
cursor: "pointer",
});
image.loading = "lazy";
if (
settings.Strictly_Load_GIFs_As_Thumbnails_On_Hover.value &&
url.match(/\.gif$/i)
) {
image.src = url;
image.addEventListener("mouseover", () => {
image.src = url;
});
image.addEventListener("mouseout", () => {
image.src = url;
});
}
mediaContainer.appendChild(image);
} else {
return; // Skip non-webm/soundpost media in webm mode
}
}
cell.appendChild(mediaContainer);
cell.appendChild(buttonDiv);
gridContainer.appendChild(cell);
}
const loadButton = () => {
const isArchivePage = window.location.pathname.includes("/archive");
let addFakeImage = settings.Add_Placeholder_Image_For_Zoom_Mode.value;
const button = document.createElement("button");
button.textContent = "Open Image Gallery";
button.id = "openImageGallery";
setStyles(button, {
position: "fixed",
bottom: "20px",
right: "20px",
zIndex: "1000",
backgroundColor: "#1c1c1c",
color: "#d9d9d9",
padding: "10px 20px",
borderRadius: "5px",
border: "none",
cursor: "pointer",
boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
visibility: settings.Hide_Gallery_Button.value ? "hidden" : "visible",
});
const openImageGallery = () => {
// new check to see if gallery is already in the DOM
const existingGallery = document.getElementById("imageGallery");
if (existingGallery) {
existingGallery.style.display = "flex";
return;
}
addFakeImage = settings.Add_Placeholder_Image_For_Zoom_Mode.value;
const gallery = document.createElement("div");
gallery.id = "imageGallery";
setStyles(gallery, {
position: "fixed",
top: "0",
left: "0",
width: "100%",
height: "100%",
backgroundColor: "rgba(0, 0, 0, 0.8)",
display: "flex",
justifyContent: "center",
alignItems: "center",
zIndex: "9999",
});
gridContainer = document.createElement("div");
setStyles(gridContainer, {
display: "grid",
gridTemplateColumns: `repeat(${settings.Grid_Columns.value}, 1fr)`,
gap: "10px",
padding: "20px",
backgroundColor: "#1c1c1c",
color: "#d9d9d9",
maxWidth: "80%",
maxHeight: "80%",
overflowY: "auto",
resize: "both",
overflow: "auto",
border: "1px solid #d9d9d9",
});
// Add dragover & drop listeners to the grid container
gridContainer.addEventListener("dragover", (e) => {
e.preventDefault();
});
gridContainer.addEventListener("drop", (e) => {
e.preventDefault();
const draggedIndex = e.dataTransfer.getData("text/plain");
const targetCell = e.target.closest("div[draggable='true']");
if (!targetCell) return;
const containerChildren = [...gridContainer.children];
const dropIndex = containerChildren.indexOf(targetCell);
if (draggedIndex >= 0 && dropIndex >= 0) {
const draggedCell = containerChildren[draggedIndex];
if (draggedIndex < dropIndex) {
gridContainer.insertBefore(draggedCell, containerChildren[dropIndex].nextSibling);
} else {
gridContainer.insertBefore(draggedCell, containerChildren[dropIndex]);
}
}
});
// Restore the previous grid container size
if (gallerySize.width > 0 && gallerySize.height > 0) {
gridContainer.style.width = `${gallerySize.width}px`;
gridContainer.style.height = `${gallerySize.height}px`;
}
let mode = "all"; // Default mode is "all"
let autoPlayWebms = false; // Default auto play webms without sound is false
const mediaTypeButtonContainer = document.createElement("div");
setStyles(mediaTypeButtonContainer, {
position: "absolute",
top: "10px",
left: "10px",
display: "flex",
gap: "10px",
});
// Toggle mode button
const toggleModeButton = document.createElement("button");
toggleModeButton.textContent = "Toggle Mode (All)";
setStyles(toggleModeButton, {
backgroundColor: "#1c1c1c",
color: "#d9d9d9",
padding: "10px 20px",
borderRadius: "5px",
border: "none",
cursor: "pointer",
boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
});
toggleModeButton.addEventListener("click", () => {
mode = mode === "all" ? "webm" : "all";
toggleModeButton.textContent = `Toggle Mode (${mode === "all" ? "All" : "Webm & Images with Sound"})`;
gridContainer.innerHTML = ""; // Clear the grid
loadPosts(mode, addFakeImage); // Reload posts based on the new mode
});
// Toggle auto play webms button
const toggleAutoPlayButton = document.createElement("button");
toggleAutoPlayButton.textContent = "Auto Play Webms without Sound";
setStyles(toggleAutoPlayButton, {
backgroundColor: "#1c1c1c",
color: "#d9d9d9",
padding: "10px 20px",
borderRadius: "5px",
border: "none",
cursor: "pointer",
boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
});
toggleAutoPlayButton.addEventListener("click", () => {
autoPlayWebms = !autoPlayWebms;
toggleAutoPlayButton.textContent = autoPlayWebms
? "Stop Auto Play Webms"
: "Auto Play Webms without Sound";
gridContainer.innerHTML = ""; // Clear the grid
loadPosts(mode, addFakeImage); // Reload posts based on the new mode and auto play setting
});
mediaTypeButtonContainer.appendChild(toggleModeButton);
mediaTypeButtonContainer.appendChild(toggleAutoPlayButton);
gallery.appendChild(mediaTypeButtonContainer);
// settings button on the top right corner of the screen
const settingsButton = document.createElement("button");
settingsButton.id = "settingsButton";
settingsButton.textContent = "Settings";
setStyles(settingsButton, {
position: "absolute",
top: "20px",
right: "20px",
backgroundColor: "#007bff", // Primary color
color: "#fff",
padding: "10px 20px",
borderRadius: "5px",
border: "none",
cursor: "pointer",
boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
transition: "background-color 0.3s ease",
});
settingsButton.addEventListener("click", () => {
const settingsContainer = document.createElement("div");
settingsContainer.id = "settingsContainer";
setStyles(settingsContainer, {
position: "fixed",
top: "0",
left: "0",
width: "100%",
height: "100%",
backgroundColor: "rgba(0, 0, 0, 0.8)",
display: "flex",
justifyContent: "center",
alignItems: "center",
zIndex: "9999",
animation: "fadeIn 0.3s ease",
});
const settingsBox = document.createElement("div");
setStyles(settingsBox, {
backgroundColor: "#000000", // Background color
color: "#ffffff", // Text color
padding: "30px",
borderRadius: "10px",
border: "1px solid #6c757d", // Secondary color
maxWidth: "80%",
maxHeight: "80%",
overflowY: "auto",
boxShadow: "0 4px 6px rgba(0, 0, 0, 0.1)",
});
const settingsTitle = document.createElement("h2");
settingsTitle.id = "settingsTitle";
settingsTitle.textContent = "Settings";
setStyles(settingsTitle, {
textAlign: "center",
marginBottom: "20px",
});
const settingsList = document.createElement("ul");
settingsList.id = "settingsList";
setStyles(settingsList, {
listStyleType: "none",
padding: "0",
margin: "0",
});
// include default settings as existing settings inside the input fields
// have an icon next to the setting that explains what the setting does
for (const setting in settings) {
// remove settings that are not in the default settings
if (!(setting in defaultSettings)) {
delete settings[setting];
continue;
}
const settingItem = document.createElement("li");
setStyles(settingItem, {
display: "flex",
alignItems: "center",
marginBottom: "15px",
});
const settingLabel = document.createElement("label");
settingLabel.textContent = setting.replace(/_/g, " ");
settingLabel.title = settings[setting].info;
setStyles(settingLabel, {
flex: "1",
display: "flex",
alignItems: "center",
});
const settingIcon = document.createElement("span");
settingIcon.className = "material-icons-outlined";
settingIcon.textContent = settings[setting].icon;
settingIcon.style.marginRight = "10px";
settingLabel.prepend(settingIcon);
settingItem.appendChild(settingLabel);
const settingInput = document.createElement("input");
const settingValueType = typeof defaultSettings[setting].value;
if (settingValueType === "boolean") {
settingInput.type = "checkbox";
settingInput.checked = settings[setting].value;
} else if (settingValueType === "number") {
settingInput.type = "number";
settingInput.value = settings[setting].value;
} else {
settingInput.type = "text";
settingInput.value = settings[setting].value;
}
setStyles(settingInput, {
padding: "8px 12px",
borderRadius: "5px",
border: "1px solid #6c757d", // Secondary color
flex: "2",
});
settingInput.addEventListener("focus", () => {
setStyles(settingInput, {
borderColor: "#007bff", // Primary color
boxShadow: "0 0 0 2px rgba(0, 123, 255, 0.25)",
outline: "none",
});
});
settingInput.addEventListener("blur", () => {
setStyles(settingInput, {
borderColor: "#6c757d", // Secondary color
boxShadow: "none",
});
});
if (settingValueType === "boolean") {
settingInput.style.marginRight = "10px";
}
settingItem.appendChild(settingInput);
settingsList.appendChild(settingItem);
}
const saveButton = document.createElement("button");
saveButton.id = "saveButton";
saveButton.textContent = "Save";
setStyles(saveButton, {
backgroundColor: "#007bff", // Primary color
color: "#fff",
padding: "10px 20px",
borderRadius: "5px",
border: "none",
cursor: "pointer",
boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
transition: "background-color 0.3s ease",
marginRight: "10px",
});
saveButton.addEventListener("click", () => {
const newSettings = {};
// First copy default settings structure
for (const key in defaultSettings) {
newSettings[key] = { ...defaultSettings[key] };
}
const inputs = document.querySelectorAll("#settingsList input");
inputs.forEach((input) => {
const settingName = input.previousSibling.textContent.replace(
/ /g,
"_"
);
if (settingName in defaultSettings) {
newSettings[settingName].value = input.type === "checkbox" ? input.checked : input.value;
}
});
localStorage.setItem("gallerySettings", JSON.stringify(newSettings));
settings = newSettings;
settingsContainer.remove();
const gallery = document.querySelector('#imageGallery');
if (gallery) {
document.body.removeChild(gallery);
setTimeout(() => {
document.querySelector('#openImageGallery').click();
}, 20);
}
});
// Close button
const closeButton = document.createElement("button");
closeButton.id = "closeButton";
closeButton.textContent = "Close";
setStyles(closeButton, {
backgroundColor: "#007bff", // Primary color
color: "#fff",
padding: "10px 20px",
borderRadius: "5px",
border: "none",
cursor: "pointer",
boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
transition: "background-color 0.3s ease",
});
closeButton.addEventListener("click", () => {
settingsContainer.remove();
});
settingsBox.appendChild(settingsTitle);
settingsBox.appendChild(settingsList);
settingsBox.appendChild(saveButton);
settingsBox.appendChild(closeButton);
settingsContainer.appendChild(settingsBox);
gallery.appendChild(settingsContainer);
});
// Hover effect for settings button
settingsButton.addEventListener("mouseenter", () => {
settingsButton.style.backgroundColor = "#0056b3";
});
settingsButton.addEventListener("mouseleave", () => {
settingsButton.style.backgroundColor = "#007bff";
});
gallery.appendChild(settingsButton);
const loadPosts = (mode, addFakeImage) => {
const checkedThreads = isArchivePage
? // Get all checked threads in the archive page or the current link if it's not an archive page
Array.from(
document.querySelectorAll(
".flashListing input[type='checkbox']:checked"
)
).map((checkbox) => {
let archiveSite =
checkbox.parentNode.parentNode.querySelector("a").href;
return archiveSite;
})
: [threadURL];
const loadPostsFromThread = (thread, addFakeImage) => {
// get the website url without the protocol and next slash
let websiteUrl = thread.replace(/(^\w+:|^)\/\//, "").split("/")[0];
// const board = thread.split("/thread/")[0].split("/").pop();
// const threadNo = `${parseInt(thread.split("thread/").pop())}`
getDocument(thread, threadURL).then((doc) => {
let posts;
// use a case statement to deal with different websites
posts = getPosts(websiteUrl, doc);
// add thread and website url as attributes to every post
posts.forEach((post) => {
post.setAttribute("thread", thread);
post.setAttribute("websiteUrl", websiteUrl);
});
if (addFakeImage) {
// Add a fake image to the grid container to allow zoom mode to open even if the thread has no images
let placeholder_imageURL = "https://files.pixstash.moe/ecl8vh.png";
let examplePost = document.createElement("div");
examplePost.innerHTML = `
<div class="postContainer", id="1231232">
<div class="fileText">
<a href="${placeholder_imageURL}" download="${placeholder_imageURL}">OpenZoomMode[sound=https://files.catbox.moe/brugtt.mp3].jpg</a>
</div>
<div class="fileThumb">
<img src="${placeholder_imageURL}" alt="Thumbnail">
</div>
<div class="postMessage">
Just a placeholder image for zoom mode
</div>
</div>
`;
examplePost.setAttribute("thread", "https://boards.4chan.org/b/thread/123456789");
examplePost.setAttribute("websiteUrl", "boards.4chan.org");
posts = [examplePost, ...posts];
}
posts.forEach((post) => {
let mediaLinkFlag = false;
let board;
let threadID;
let postID;
let postURL;
let thumbnailUrl;
let mediaLink;
let fileName;
let comment;
let isVideo;
let isImage;
let soundLink;
let encodedSoundPostLink;
let temp;
let hasEmbeddedMediaLink = false;
let matches;
websiteUrl = post.getAttribute("websiteUrl");
thread = post.getAttribute("thread");
// case statement for different websites
switch (websiteUrl) {
case "warosu.org":
let thumbnailElement = post.querySelector(".thumb");
fileName = post
.querySelector(".fileinfo")
?.innerText.split(", ")[2];
thumbnailUrl = thumbnailElement?.src;
mediaLink = thumbnailElement?.parentNode.href;
comment = post.querySelector("blockquote");
threadID = post.getAttribute("thread").match(/thread\/(\d+)/)
if (threadID) {
threadID = threadID[1];
} else {
threadID = post.querySelector(".js").href.match(/thread\/(\d+)/)[1];
}
postID = post.id.replace("pc", "").replace("p", "");
break;
case "archived.moe":
case "archive.palanq.win":
case "archive.4plebs.org":
case "desuarchive.org":
case "thebarchive.com":
case "archiveofsins.com":
thumbnailUrl = post.querySelector(".post_image")?.src;
mediaLink = post.querySelector(".thread_image_link")?.href;
fileName = post.querySelector(
".post_file_filename"
)?.title;
comment = post.querySelector(".text");
threadID = post.querySelector(".post_data > a")?.href.match(
/thread\/(\d+)/
)[1];
postID = post.id
break;
case "boards.4chan.org":
case "boards.4channel.org":
default:
if (post.querySelector(".fileText")) {
// if they have 4chanX installed, there will be a fileText-orignal class
if (post.querySelector(".download-button")) {
temp = post.querySelector(".download-button");
mediaLink = temp.href;
fileName = temp.download;
} else {
if (post.classList.contains("opContainer")) {
mediaLink = post.querySelector(".fileText a");
temp = mediaLink;
} else {
mediaLink = post.querySelector(".fileText");
temp = mediaLink.querySelector("a");
}
if (mediaLink.title === "") {
if (temp.title === "") {
fileName = temp.innerText;
} else {
fileName = temp.title;
}
} else {
fileName = mediaLink.title;
}
mediaLink = temp.href;
}
thumbnailUrl = post.querySelector(".fileThumb img")?.src;
}
comment = post.querySelector(".postMessage");
threadID = thread.match(/thread\/(\d+)/)[1];
postID = post.id.replace("pc", "").replace("p", "");
}
const fileExtRegex = /\.(webm|mp4|jpg|png|gif)$/i;
const linkRegex = /https:\/\/(files|litter)\.(catbox|pixstash)\.moe\/[a-z0-9]+\.(jpg|png|gif|webm|mp4)/g;
if (mediaLink) {
const ext = mediaLink.match(fileExtRegex)?.[1]?.toLowerCase();
isVideo = ext === 'webm' || ext === 'mp4';
isImage = ext === 'jpg' || ext === 'png' || ext === 'gif';
soundLink = fileName.match(/\[sound=(.+?)\]/);
mediaLinkFlag = true;
}
if (settings.Embed_External_Links.value && comment) {
matches = Array.from(comment.innerText.matchAll(linkRegex)).map(match => match[0]);
if (matches.length > 0) {
if (!mediaLinkFlag) {
mediaLink = matches[0];
fileName = mediaLink.split("/").pop();
thumbnailUrl = mediaLink;
if (hasEmbeddedMediaLink) {
matches.shift();
}
const ext = mediaLink.match(fileExtRegex)?.[1]?.toLowerCase();
isVideo = ext === 'webm' || ext === 'mp4';
isImage = ext === 'jpg' || ext === 'png' || ext === 'gif';
soundLink = fileName.match(/\[sound=(.+?)\]/);
mediaLinkFlag = true;
}
hasEmbeddedMediaLink = matches.length > 0;
}
}
// replace the "#pcXXXXXXX" or "#pXXXXXXX" with an empty string to get the actual thread url
if (thread.includes("#")) {
postURL = thread.replace(/#p\d+/, "");
postURL = postURL.replace(/#pc\d+/, "");
} else {
postURL = thread;
}
// post info (constant)
board = thread.match(/\/\/[^\/]+\/([^\/]+)/)[1];
if (soundLink) {
encodedSoundPostLink = `https://4chan.mahdeensky.top/${board}/thread/${threadID}/${postID}`;
}
if (mediaLinkFlag) {
// Check if the post should be loaded based on the mode
if (
mode === "all" ||
(mode === "webm" && (isVideo || (isImage && soundLink)))
) {
// Insert a button/link to open media in new tab for videos
const cell = document.createElement("div");
setStyles(cell, {
border: "1px solid #d9d9d9",
position: "relative",
});
// Make the cell draggable
cell.draggable = true;
cell.addEventListener("dragstart", (e) => {
e.dataTransfer.setData("text/plain", [...gridContainer.children].indexOf(cell));
e.dataTransfer.dropEffect = "move";
});
// Allow drops on this cell
cell.addEventListener("dragover", (e) => {
e.preventDefault();
e.dataTransfer.dropEffect = "move";
});
cell.addEventListener("drop", (e) => {
e.preventDefault();
const draggedIndex = e.dataTransfer.getData("text/plain");
const containerChildren = [...gridContainer.children];
const draggedCell = containerChildren[draggedIndex];
if (draggedCell !== cell) {
const dropIndex = containerChildren.indexOf(cell);
if (draggedIndex < dropIndex) {
gridContainer.insertBefore(draggedCell, containerChildren[dropIndex].nextSibling);
} else {
gridContainer.insertBefore(draggedCell, containerChildren[dropIndex]);
}
}
});
const buttonDiv = document.createElement("div");
setStyles(buttonDiv, {
display: "flex",
justifyContent: "space-between",
alignItems: "center",
padding: "5px",
});
if (isVideo) {
const videoContainer = document.createElement("div");
setStyles(videoContainer, {
position: "relative",
display: "flex",
justifyContent: "center",
});
// if medialink is catbox.moe or pixstash.moe, then video thumbnail is a video element with no controls
let videoThumbnail;
if (mediaLink.match(/catbox.moe|pixstash.moe/)) {
videoThumbnail = document.createElement("video");
} else {
videoThumbnail = document.createElement("img");
}
videoThumbnail.src = thumbnailUrl;
videoThumbnail.alt = "Video Thumbnail";
setStyles(videoThumbnail, {
width: "100%",
maxHeight: `${settings.Grid_Cell_Max_Height.value}px`,
objectFit: "contain",
cursor: "pointer",
});
videoThumbnail.loading = "lazy";
const video = document.createElement("video");
video.src = mediaLink;
video.controls = true;
video.title = comment.innerText;
video.videothumbnailDisplayed = "true";
video.setAttribute("fileName", fileName);
video.setAttribute("board", board);
video.setAttribute("threadID", threadID);
video.setAttribute("postID", postID);
setStyles(video, {
maxWidth: "100%",
maxHeight: `${settings.Grid_Cell_Max_Height.value}px`,
objectFit: "contain",
cursor: "pointer",
display: "none",
});
// videoJS stuff (not working for some reason)
// video.className = "video-js";
// video.setAttribute("data-setup", "{}");
// const source = document.createElement("source");
// source.src = mediaLink;
// source.type = "video/webm";
// video.appendChild(source);
videoThumbnail.addEventListener("click", () => {
videoThumbnail.style.display = "none";
video.style.display = "block";
video.videothumbnailDisplayed = "false";
// video.load();
});
// hide the video thumbnail and show the video when hovered
videoThumbnail.addEventListener("mouseenter", () => {
videoThumbnail.style.display = "none";
video.style.display = "block";
video.videothumbnailDisplayed = "false";
// video.load();
});
// Play webms without sound automatically on hover or if autoPlayWebms is true
if (!soundLink) {
if (autoPlayWebms) {
video.addEventListener("canplaythrough", () => {
video.play();
video.loop = true; // Loop webms when autoPlayWebms is true
});
} else {
if (settings.Play_Webms_On_Hover.value) {
video.addEventListener("mouseenter", () => {
video.play();
});
video.addEventListener("mouseleave", () => {
video.pause();
});
}
}
}
videoContainer.appendChild(videoThumbnail);
videoContainer.appendChild(video);
if (soundLink) {
// video.preload = "none"; // Disable video preload for better performance
const audio = document.createElement("audio");
audio.src = decodeURIComponent(
soundLink[1].startsWith("http")
? soundLink[1]
: `https://${soundLink[1]}`
);
// if switch catbox to pixstash is enabled, replace catbox.moe with pixstash.moe
if (settings.Switch_Catbox_To_Pixstash_For_Soundposts.value) {
audio.src = audio.src.replace("catbox.moe", "pixstash.moe");
}
// add attribute to the audio element with the encoded soundpost link
audio.setAttribute(
"encodedSoundPostLink",
encodedSoundPostLink
);
videoContainer.appendChild(audio);
const resetButton = document.createElement("button");
resetButton.textContent = "Reset";
setStyles(resetButton, {
backgroundColor: "#1c1c1c",
color: "#d9d9d9",
padding: "5px 10px",
borderRadius: "3px",
border: "none",
cursor: "pointer",
boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
});
resetButton.addEventListener("click", () => {
video.currentTime = 0;
audio.currentTime = 0;
});
buttonDiv.appendChild(resetButton);
// html5 video play
video.onplay = (event) => {
audio.play();
};
video.onpause = (event) => {
audio.pause();
};
let lastVideoTime = 0;
// Sync audio with video on timeupdate event only if the difference is 2 seconds or more
video.addEventListener("timeupdate", () => {
if (Math.abs(video.currentTime - lastVideoTime) >= 2) {
audio.currentTime = video.currentTime;
lastVideoTime = video.currentTime;
}
lastVideoTime = video.currentTime;
});
}
cell.appendChild(videoContainer);
} else if (isImage) {
const imageContainer = document.createElement("div");
setStyles(imageContainer, {
position: "relative",
display: "flex",
justifyContent: "center",
alignItems: "center",
});
const image = document.createElement("img");
image.src = thumbnailUrl;
if (settings.Load_High_Res_Images_By_Default.value) {
image.src = mediaLink;
}
if (mediaLink.includes(".gif")) {
image.src = mediaLink;
if (
settings.Strictly_Load_GIFs_As_Thumbnails_On_Hover.value
) {
mediaLink = thumbnailUrl;
image.src = thumbnailUrl;
}
}
image.setAttribute("fileName", fileName);
image.setAttribute("actualSrc", mediaLink);
image.setAttribute("thumbnailUrl", thumbnailUrl);
image.setAttribute("board", board);
image.setAttribute("threadID", threadID);
image.setAttribute("postID", postID);
setStyles(image, {
maxWidth: "100%",
maxHeight: `${settings.Grid_Cell_Max_Height.value}px`,
objectFit: "contain",
cursor: "pointer",
});
let createDarkenBackground = () => {
const background = document.createElement("div");
background.id = "darkenBackground";
setStyles(background, {
position: "fixed",
top: "0",
left: "0",
width: "100%",
height: "100%",
backgroundColor: "rgba(0, 0, 0, 0.3)",
backdropFilter: "blur(5px)",
zIndex: "9999",
});
return background;
};
let zoomImage = () => {
// have the image pop up centered in front of the screen so that it fills about 80% of the screen
image.style = "";
image.src = mediaLink;
setStyles(image, {
position: "fixed",
top: "50%",
left: "50%",
transform: "translate(-50%, -50%)",
zIndex: "10000",
height: "80%",
width: "80%",
objectFit: "contain",
cursor: "pointer",
});
// darken and blur the background behind the image without affecting the image
const background = createDarkenBackground();
background.appendChild(createArrowButton('left'));
background.appendChild(createArrowButton('right'));
gallery.appendChild(background);
// create a container for the buttons, number, and download buttons (even space between them)
// position: fixed; bottom: 10px; display: flex; flex-direction: row; justify-content: space-around; z-index: 10000; width: 100%; margin:auto;
const bottomContainer = document.createElement("div");
setStyles(bottomContainer, {
position: "fixed",
bottom: "10px",
display: "flex",
flexDirection: "row",
justifyContent: "space-around",
zIndex: "10000",
width: "100%",
margin: "auto",
});
background.appendChild(bottomContainer);
// buttons on the bottom left of the screen for reverse image search (SauceNAO, Google Lens, Yandex)
const buttonContainer = document.createElement("div");
setStyles(buttonContainer, {
display: "flex",
gap: "10px",
});
buttonContainer.setAttribute("mediaLink", mediaLink);
const sauceNAOButton = document.createElement("button");
sauceNAOButton.textContent = "SauceNAO";
setStyles(sauceNAOButton, {
backgroundColor: "#1c1c1c",
color: "#d9d9d9",
padding: "5px 10px",
borderRadius: "3px",
border: "none",
cursor: "pointer",
});
sauceNAOButton.addEventListener("click", () => {
window.open(
`https://saucenao.com/search.php?url=${encodeURIComponent(
buttonContainer.getAttribute("mediaLink")
)}`
);
});
buttonContainer.appendChild(sauceNAOButton);
const googleLensButton = document.createElement("button");
googleLensButton.textContent = "Google Lens";
setStyles(googleLensButton, {
backgroundColor: "#1c1c1c",
color: "#d9d9d9",
padding: "5px 10px",
borderRadius: "3px",
border: "none",
cursor: "pointer",
boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
});
googleLensButton.addEventListener("click", () => {
window.open(
`https://lens.google.com/uploadbyurl?url=${encodeURIComponent(
buttonContainer.getAttribute("mediaLink")
)}`
);
});
buttonContainer.appendChild(googleLensButton);
const yandexButton = document.createElement("button");
yandexButton.textContent = "Yandex";
setStyles(yandexButton, {
backgroundColor: "#1c1c1c",
color: "#d9d9d9",
padding: "5px 10px",
borderRadius: "3px",
border: "none",
cursor: "pointer",
boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
});
yandexButton.addEventListener("click", () => {
window.open(
`https://yandex.com/images/search?rpt=imageview&url=${encodeURIComponent(
buttonContainer.getAttribute("mediaLink")
)}`
);
});
buttonContainer.appendChild(yandexButton);
bottomContainer.appendChild(buttonContainer);
// download container for video/img and audio
const downloadButtonContainer =
document.createElement("div");
setStyles(downloadButtonContainer, {
display: "flex",
gap: "10px",
});
bottomContainer.appendChild(downloadButtonContainer);
const viewPostButton = document.createElement("a");
viewPostButton.textContent = "View Post";
viewPostButton.href = `https://boards.4chan.org/${board}/thread/${threadID}#p${postID}`;
setStyles(viewPostButton, {
backgroundColor: "#1c1c1c",
color: "#d9d9d9",
padding: "5px 10px",
borderRadius: "3px",
border: "none",
cursor: "pointer",
boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
});
downloadButtonContainer.appendChild(viewPostButton);
const downloadButton = document.createElement("a");
downloadButton.textContent = "Download Video/Image";
downloadButton.href = mediaLink;
downloadButton.download = fileName;
downloadButton.target = "_blank";
setStyles(downloadButton, {
backgroundColor: "#1c1c1c",
color: "#d9d9d9",
padding: "5px 10px",
borderRadius: "3px",
border: "none",
cursor: "pointer",
boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
});
downloadButtonContainer.appendChild(downloadButton);
const audioDownloadButton = document.createElement("a");
audioDownloadButton.textContent = "Download Audio";
audioDownloadButton.target = "_blank";
setStyles(audioDownloadButton, {
backgroundColor: "#1c1c1c",
color: "#d9d9d9",
padding: "5px 10px",
borderRadius: "3px",
border: "none",
cursor: "pointer",
boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
});
if (soundLink) {
audioDownloadButton.href = decodeURIComponent(
soundLink[1].startsWith("http")
? soundLink[1]
: `https://${soundLink[1]}`
);
// if switch catbox to pixstash is enabled, replace catbox.moe with pixstash.moe
if (settings.Switch_Catbox_To_Pixstash_For_Soundposts.value) {
audioDownloadButton.href = audioDownloadButton.href.replace(
"catbox.moe",
"pixstash.moe"
);
}
audioDownloadButton.download = soundLink[1]
.split("/")
.pop();
} else {
audioDownloadButton.style.display = "none";
}
downloadButtonContainer.appendChild(audioDownloadButton);
// a button beside the download video and download audio button that says download encoded soundpost which links to the following url in a new tab "https://4chan.mahdeensky.top/<board>/thread/<thread>/<post>" where things between the <>, are variables to be replaced
const encodedSoundPostButton =
document.createElement("a");
encodedSoundPostButton.textContent =
"Download Encoded Soundpost";
encodedSoundPostButton.target = "_blank";
setStyles(encodedSoundPostButton, {
backgroundColor: "#1c1c1c",
color: "#d9d9d9",
padding: "5px 10px",
borderRadius: "3px",
border: "none",
cursor: "pointer",
boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
});
if (soundLink) {
encodedSoundPostButton.href = `https://4chan.mahdeensky.top/${board}/thread/${threadID}/${postID}`;
} else {
encodedSoundPostButton.style.display = "none";
}
downloadButtonContainer.appendChild(
encodedSoundPostButton
);
// number on the bottom right of the screen to show which image is currently being viewed
const imageNumber = document.createElement("div");
let currentImageNumber =
Array.from(cell.parentNode.children).indexOf(cell) + 1;
let imageTotal = cell.parentNode.children.length;
imageNumber.textContent = `${currentImageNumber}/${imageTotal}`;
setStyles(imageNumber, {
backgroundColor: "#1c1c1c",
color: "#d9d9d9",
padding: "5px 10px",
borderRadius: "3px",
border: "none",
cursor: "pointer",
boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
position: "fixed",
top: "10px",
left: "10px",
});
background.appendChild(imageNumber);
// title of the image/video on the top left of the screen
const imageTitle = document.createElement("div");
imageTitle.textContent = fileName;
setStyles(imageTitle, {
position: "fixed",
top: "10px",
right: "10px",
backgroundColor: "#1c1c1c",
color: "#d9d9d9",
padding: "5px 10px",
borderRadius: "3px",
border: "none",
cursor: "pointer",
boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
zIndex: "10000",
});
background.appendChild(imageTitle);
let currentCell = cell;
function navigateImage(direction) {
const targetCell = direction === 'left' ? currentCell.previousElementSibling : currentCell.nextElementSibling;
if (!targetCell) return;
// ...existing navigation code using targetCell instead of previousCell/nextCell...
if (gallery.querySelector("#zoomedVideo")) {
if (
gallery
.querySelector("#zoomedVideo")
.querySelector("audio")
) {
gallery
.querySelector("#zoomedVideo")
.querySelector("audio")
.pause();
}
gallery.removeChild(
gallery.querySelector("#zoomedVideo")
);
} else if (gallery.querySelector("#zoomedImage")) {
gallery.removeChild(
gallery.querySelector("#zoomedImage")
);
} else {
image.style = "";
// image.src = thumbnailUrl;
setStyles(image, {
maxWidth: "100%",
maxHeight: `${settings.Grid_Cell_Max_Height.value}px`,
objectFit: "contain",
});
}
// check if it has a video
const video = targetCell?.querySelector("video");
if (video) {
const video = targetCell
.querySelector("video")
.cloneNode(true);
video.id = "zoomedVideo";
video.style = "";
setStyles(video, {
position: "fixed",
top: "50%",
left: "50%",
transform: "translate(-50%, -50%)",
zIndex: "10000",
height: "80%",
width: "80%",
objectFit: "contain",
cursor: "pointer",
preload: "auto",
});
gallery.appendChild(video);
// check if there is an audio element
let audio = targetCell.querySelector("audio");
if (audio) {
audio = audio.cloneNode(true);
// same event listeners as the video
video.onplay = (event) => {
audio.play();
};
video.onpause = (event) => {
audio.pause();
};
let lastVideoTime = 0;
video.addEventListener("timeupdate", () => {
if (
Math.abs(
video.currentTime - lastVideoTime
) >= 2
) {
audio.currentTime = video.currentTime;
lastVideoTime = video.currentTime;
}
lastVideoTime = video.currentTime;
});
video.appendChild(audio);
}
} else {
// if it doesn't have a video, it must have an image
const originalImage =
targetCell.querySelector("img");
const currentImage =
originalImage.cloneNode(true);
currentImage.id = "zoomedImage";
currentImage.style = "";
currentImage.src =
currentImage.getAttribute("actualSrc");
originalImage.src =
originalImage.getAttribute("actualSrc");
setStyles(currentImage, {
position: "fixed",
top: "50%",
left: "50%",
transform: "translate(-50%, -50%)",
zIndex: "10000",
height: "80%",
width: "80%",
objectFit: "contain",
cursor: "pointer",
});
gallery.appendChild(currentImage);
currentImage.addEventListener("click", () => {
gallery.removeChild(currentImage);
gallery.removeChild(background);
document.removeEventListener(
"keydown",
keybindHandler
);
});
let audio = targetCell.querySelector("audio");
if (audio) {
audio = audio.cloneNode(true);
currentImage.appendChild(audio);
// event listeners when hovering over the image
currentImage.addEventListener(
"mouseenter",
() => {
audio.play();
}
);
currentImage.addEventListener(
"mouseleave",
() => {
audio.pause();
}
);
}
}
if (targetCell) {
currentCell = targetCell;
buttonContainer.setAttribute(
"mediaLink",
targetCell.querySelector("img").src
);
currentImageNumber += direction === 'left' ? -1 : 1;
imageNumber.textContent = `${currentImageNumber}/${imageTotal}`;
// filename of the video if it has one, otherwise the filename of the image
imageTitle.textContent = video
? video.getAttribute("fileName")
: targetCell
.querySelector("img")
.getAttribute("fileName");
// update view post button link
let targetMedia = video || targetCell.querySelector("img");
let targetBoard = targetMedia.getAttribute("board");
let targetThreadID = targetMedia.getAttribute("threadID");
let targetPostID = targetMedia.getAttribute("postID");
viewPostButton.href = `https://boards.4chan.org/${targetBoard}/thread/${targetThreadID}#p${targetPostID}`;
// update the download button links
downloadButton.href = targetMedia.src;
if (targetCell.querySelector("audio")) {
// updating audio button download link
audioDownloadButton.href =
targetCell.querySelector("audio").src;
audioDownloadButton.download = targetCell
.querySelector("audio")
.src.split("/")
.pop();
audioDownloadButton.style.display = "block";
// updating encoded soundpost button link
encodedSoundPostButton.href = targetCell.querySelector("audio")
.getAttribute("encodedSoundPostLink");
encodedSoundPostButton.style.display = "block";
} else {
audioDownloadButton.style.display = "none";
encodedSoundPostButton.style.display = "none";
}
}
}
const keybindHandler = (event) => {
if (event.key === "ArrowLeft") {
navigateImage('left');
} else if (event.key === "ArrowRight") {
navigateImage('right');
}
};
document.addEventListener("keydown", keybindHandler);
image.addEventListener(
"click",
() => {
image.style = "";
// image.src = thumbnailUrl;
setStyles(image, {
maxWidth: "99%",
maxHeight: `${settings.Grid_Cell_Max_Height.value}px`,
objectFit: "contain",
});
if (gallery.querySelector("#darkenBackground")) {
gallery.removeChild(background);
}
document.removeEventListener(
"keydown",
keybindHandler
);
image.addEventListener("click", zoomImage, {
once: true,
});
},
{ once: true }
);
};
image.addEventListener("click", zoomImage, { once: true });
image.title = comment.innerText;
image.loading = "lazy";
if (soundLink) {
const audio = document.createElement("audio");
audio.src = decodeURIComponent(
soundLink[1].startsWith("http")
? soundLink[1]
: `https://${soundLink[1]}`
);
// if switch catbox to pixstash is enabled, replace catbox.moe with pixstash.moe
if (settings.Switch_Catbox_To_Pixstash_For_Soundposts.value) {
audio.src = audio.src.replace("catbox.moe", "pixstash.moe");
}
audio.loop = true;
// set the attribute to the audio element with the encoded soundpost link
audio.setAttribute(
"encodedSoundPostLink",
encodedSoundPostLink
);
imageContainer.appendChild(audio);
image.addEventListener("mouseenter", () => {
audio.play();
});
image.addEventListener("mouseleave", () => {
audio.pause();
});
const playPauseButton = document.createElement("button");
playPauseButton.textContent = "Play/Pause";
setStyles(playPauseButton, {
backgroundColor: "#1c1c1c",
color: "#d9d9d9",
padding: "5px 10px",
borderRadius: "3px",
border: "none",
cursor: "pointer",
boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
});
playPauseButton.addEventListener("click", () => {
if (audio.paused) {
audio.play();
} else {
audio.pause();
}
});
buttonDiv.appendChild(playPauseButton);
}
imageContainer.appendChild(image);
cell.appendChild(imageContainer);
} else {
return; // Skip non-video and non-image posts
}
// Add button that scrolls to the post in the thread
const viewPostButton = document.createElement("button");
viewPostButton.textContent = "View Post";
setStyles(viewPostButton, {
backgroundColor: "#1c1c1c",
color: "#d9d9d9",
padding: "5px 10px",
borderRadius: "3px",
border: "none",
cursor: "pointer",
boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
});
viewPostButton.addEventListener("click", () => {
gallerySize = {
width: gridContainer.offsetWidth,
height: gridContainer.offsetHeight,
};
lastScrollPosition = gridContainer.scrollTop;
window.location.href = postURL + "#" + post.id;
// post id example: "pc77515440"
gallery.style.display = "none"; // hide instead of removing
});
buttonDiv.appendChild(viewPostButton);
// Add button that opens the media in a new tab if the media
const openInNewTabButton = document.createElement("button");
openInNewTabButton.textContent = "Open";
setStyles(openInNewTabButton, {
backgroundColor: "#1c1c1c",
color: "#d9d9d9",
padding: "5px 10px",
borderRadius: "3px",
border: "none",
cursor: "pointer",
boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
});
openInNewTabButton.addEventListener("click", () => {
window.open(mediaLink, "_blank");
});
buttonDiv.appendChild(openInNewTabButton);
cell.appendChild(buttonDiv);
gridContainer.appendChild(cell);
}
}
// In the loadPosts function, update the embedded links section:
if (hasEmbeddedMediaLink) {
// Create a proper post link that includes the thread ID and post ID
const fullPostLink = postURL + "#" + post.id;
matches.forEach(url => {
createMediaCell(url, comment.innerText, mode, fullPostLink, board, threadID, postID); // Pass the current post's URL
});
}
});
});
};
// only load the fake image in the first thread
loadPostsFromThread(checkedThreads[0], addFakeImage);
// load the rest of the threads with no fake image
checkedThreads.slice(1).forEach((thread) => {
loadPostsFromThread(thread, false);
});
};
loadPosts(mode, addFakeImage);
gallery.appendChild(gridContainer);
const closeButton = document.createElement("button");
closeButton.textContent = "Close";
closeButton.id = "closeGallery";
setStyles(closeButton, {
position: "absolute",
bottom: "10px",
right: "10px",
zIndex: "10000",
backgroundColor: "#1c1c1c",
color: "#d9d9d9",
padding: "10px 20px",
borderRadius: "5px",
border: "none",
cursor: "pointer",
boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
});
closeButton.addEventListener("click", () => {
gallerySize = {
width: gridContainer.offsetWidth,
height: gridContainer.offsetHeight,
};
gallery.style.display = "none"; // hide instead of removing
});
gallery.appendChild(closeButton);
// Add scroll to bottom button
const scrollBottomButton = document.createElement("button");
scrollBottomButton.textContent = "Scroll to Last";
setStyles(scrollBottomButton, {
position: "fixed",
bottom: "20px",
left: "20px",
backgroundColor: "#1c1c1c",
color: "#d9d9d9",
padding: "10px 20px",
borderRadius: "5px",
border: "none",
cursor: "pointer",
zIndex: "10000",
});
scrollBottomButton.addEventListener("click", () => {
const lastCell = gridContainer.lastElementChild;
if (lastCell) {
lastCell.scrollIntoView({ behavior: "smooth" });
}
});
gallery.appendChild(scrollBottomButton);
// Add zoom mode arrow buttons
const background = document.createElement('div');
background.appendChild(createArrowButton('left'));
background.appendChild(createArrowButton('right'));
document.body.appendChild(gallery);
// Store the current scroll position and grid container size when closing the gallery
// (`Last scroll position: ${lastScrollPosition} px`);
gridContainer.addEventListener("scroll", () => {
lastScrollPosition = gridContainer.scrollTop;
// (`Current scroll position: ${lastScrollPosition} px`);
});
// Restore the last scroll position and grid container size when opening the gallery after a timeout if the url is the same
if (window.location.href.includes(threadURL.replace(/#.*$/, ""))) {
setTimeout(() => {
if (gallerySize.width > 0 && gallerySize.height > 0) {
gridContainer.style.width = `${gallerySize.width}px`;
gridContainer.style.height = `${gallerySize.height}px`;
}
// (`Restored scroll position: ${lastScrollPosition} px`);
gridContainer.scrollTop = lastScrollPosition;
}, 100);
} else {
// Reset the last scroll position and grid container size if the url is different
threadURL = window.location.href;
lastScrollPosition = 0;
gallerySize = { width: 0, height: 0 };
}
gallery.addEventListener("click", (event) => {
if (event.target === gallery) {
closeButton.click();
}
});
};
button.addEventListener("click", openImageGallery);
// Append the button to the body
document.body.appendChild(button);
if (isArchivePage) {
// adds the category to thead
const thead = document.querySelector(".flashListing thead tr");
const checkboxCell = document.createElement("td");
checkboxCell.className = "postblock";
checkboxCell.textContent = "Selected";
thead.insertBefore(checkboxCell, thead.firstChild);
// Add checkboxes to each thread row
const threadRows = document.querySelectorAll(".flashListing tbody tr");
threadRows.forEach((row) => {
const checkbox = document.createElement("input");
checkbox.type = "checkbox";
const checkboxCell = document.createElement("td");
checkboxCell.appendChild(checkbox);
row.insertBefore(checkboxCell, row.firstChild);
});
}
};
// Use the "i" key to open and close the gallery/grid
document.addEventListener("keydown", (event) => {
if (event.target.tagName === "INPUT" || event.target.tagName === "TEXTAREA") {
return;
}
if (event.key === settings.Open_Close_Gallery_Key.value) {
if (!document.querySelector("#imageGallery")) {
document.querySelector("#openImageGallery").click();
return;
}
if (document.querySelector("#imageGallery").style.display === "none") {
document.querySelector("#openImageGallery").click();
} else {
document.querySelector("#closeGallery").click();
}
}
});
loadButton();
console.log("4chan Gallery loaded successfully!");
})();