您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
自動更新、引用元に引用アンカー追加、マウスオーバーで画像拡大表示などお役立ち機能を追加するスクリプト
// ==UserScript== // @name FutabaX // @namespace http://tampermonkey.net/ // @version 1.1 // @description 自動更新、引用元に引用アンカー追加、マウスオーバーで画像拡大表示などお役立ち機能を追加するスクリプト // @author としあき // @match *://*.2chan.net/* // @match https://www.2chan.net/index2.html // @grant GM_xmlhttpRequest // @grant GM_addStyle // @connect 2chan.net // @connect futakuro.com // @connect futabaforest.net // @connect ftbucket.info // @license MIT // ==/UserScript== (function() { 'use strict'; GM_addStyle(` td.rtd { white-space: normal !important; word-wrap: break-word !important; overflow-wrap: break-word !important; text-align: left !important; } a[data-quote-post] { display: inline-block !important; white-space: nowrap; margin: 2px 5px 2px 0 !important; } `); let resolvedThreadCache = {}; if (window.top === window.self && document.getElementsByTagName("frameset").length > 0) { window.location.replace("https://www.2chan.net/index2.html"); } if (window.location.href.indexOf("index2.html") !== -1) { function modifyFrontPageLinks() { document.querySelectorAll('a[href="https://dec.2chan.net/dec/futaba.htm"]').forEach(a => { a.textContent = "二次元裏dec"; }); document.querySelectorAll('a[href="https://jun.2chan.net/jun/futaba.htm"]').forEach(a => { a.textContent = "二次元裏jun"; }); document.querySelectorAll('a[href="https://may.2chan.net/b/futaba.htm"]').forEach(a => { a.textContent = "二次元裏may"; }); const mayLink = document.querySelector('a[href="https://may.2chan.net/b/futaba.htm"]'); if (mayLink) { mayLink.insertAdjacentHTML("afterend", '<br><a href="https://img.2chan.net/b/futaba.htm">二次元裏img</a><br>' + '<a href="https://cgi.2chan.net/b/futaba.htm">二次元裏cgi</a><br>' + '<a href="https://dat.2chan.net/b/futaba.htm">二次元裏dat</a>' ); } document.querySelectorAll('a[href="https://may.2chan.net/b/futaba.htm"]').forEach(a => { a.insertAdjacentHTML("afterend", '<span style="color:red;font-size:80%">人気</span>'); }); document.querySelectorAll('a[href="https://img.2chan.net/b/futaba.htm"]').forEach(a => { a.insertAdjacentHTML("afterend", '<span style="color:red;font-size:80%">人気</span>'); }); document.querySelectorAll('a[href="https://dec.2chan.net/84/futaba.htm"]').forEach(a => { a.insertAdjacentHTML("afterend", '<span style="color:red;font-size:80%">人気</span>'); }); document.querySelectorAll('a[href="https://dec.2chan.net/60/futaba.htm"]').forEach(a => { a.insertAdjacentHTML("afterend", '<span style="color:red;font-size:80%">人気</span>'); }); document.querySelectorAll('a[href="https://dec.2chan.net/55/futaba.htm"]').forEach(a => { a.insertAdjacentHTML("afterend", '<span style="color:red;font-size:80%">人気</span>'); }); document.querySelectorAll('a[href="https://dec.2chan.net/73/futaba.htm"]').forEach(a => { a.insertAdjacentHTML("afterend", '<span style="color:red;font-size:80%">人気</span>'); }); } modifyFrontPageLinks(); } const originalFavicon = "https://2chan.net/favicon.ico"; const thresholdFaviconBase64 = ""; const newReplyFaviconBase64 = ""; let thresholdMode = false; const extraStyle = document.createElement("style"); extraStyle.innerHTML = ` .highlighted { background-color: #EDD0BD !important; transition: background-color 2s ease-out; } .quote-preview { background-color: #F0E0D6; padding: 8px; border: 1px solid #F0E0D6; max-width: 90vw; overflow: auto; box-shadow: 2px 2px 10px rgba(0,0,0,0.5); } #replyContainer { position: fixed; bottom: 20px; right: 20px; background: #fff; border: 1px solid #ccc; padding: 5px; z-index: 15000; box-shadow: 2px 2px 10px rgba(0,0,0,0.5); } #replyDragHandle { background: #ccc; padding: 4px; cursor: move; font-size: 0.9em; text-align: center; user-select: none; } `; document.head.appendChild(extraStyle); function highlightPost(post) { post.classList.add("highlighted"); setTimeout(() => { post.classList.remove("highlighted"); }, 2000); } let currentQuotePreview = null; function handleQuoteLinkMouseEnter(e) { const link = this; const quoteNum = link.dataset.quotePost; const target = document.getElementById("post-" + quoteNum); if (!target) return; currentQuotePreview = target.cloneNode(true); currentQuotePreview.classList.add("quote-preview"); currentQuotePreview.style.position = "fixed"; currentQuotePreview.style.zIndex = "20000"; currentQuotePreview.style.left = (e.clientX + 10) + "px"; currentQuotePreview.style.top = (e.clientY - 10) + "px"; document.body.appendChild(currentQuotePreview); } function handleQuoteLinkMouseMove(e) { if (currentQuotePreview) { currentQuotePreview.style.left = (e.clientX + 10) + "px"; currentQuotePreview.style.top = (e.clientY - 10) + "px"; } } function handleQuoteLinkMouseLeave(e) { if (currentQuotePreview) { currentQuotePreview.remove(); currentQuotePreview = null; } } function removeUnwantedElements(root = document) { root.querySelectorAll('iframe[src^="https://dec.2chan.net/bin/sphead.htm"], iframe[src^="https://dec.2chan.net/bin/spfoot_a.htm"]').forEach(iframe => { if (iframe.parentElement) { iframe.parentElement.remove(); } }); root.querySelectorAll('div').forEach(div => { if (div.innerHTML.includes("ヘッダ広告ここから")) { div.remove(); } }); root.querySelectorAll('.tue2').forEach(el => el.remove()); root.querySelectorAll('iframe[src^="https://dec.2chan.net/bin/overlay.htm"]').forEach(iframe => { if (iframe.parentElement) { let parent = iframe.parentElement; parent.remove(); let prev = parent.previousElementSibling; if (prev && prev.tagName === 'DIV' && prev.getAttribute('style') && prev.getAttribute('style').includes('height:68px')) { prev.remove(); } } }); root.querySelectorAll('style').forEach(styleEl => { if (styleEl.innerHTML.includes('.footfix')) { styleEl.remove(); } }); root.querySelectorAll('iframe[src^="https://dec.2chan.net/bin/foot2_a.htm"]').forEach(iframe => { if (iframe.parentElement) { iframe.parentElement.remove(); } }); root.querySelectorAll('iframe[src^="https://dec.2chan.net/bin/hsi1.htm"], iframe[src^="https://dec.2chan.net/bin/foot4_ab.htm"]').forEach(iframe => { if (iframe.parentElement) { iframe.parentElement.remove(); } }); root.querySelectorAll('iframe[src^="https://dec.2chan.net/bin/hsif.htm"]').forEach(iframe => { if (iframe.parentElement) { iframe.parentElement.remove(); } }); root.querySelectorAll('iframe[src^="/bin/catp.htm"]').forEach(iframe => { let container = iframe.closest('div[style*="width:610px"]'); if (container) { container.remove(); } else if (iframe.parentElement) { iframe.parentElement.remove(); } }); root.querySelectorAll('.footfix').forEach(el => el.remove()); ["imobile_adspotdiv1", "imobile_adspotdiv2"].forEach(id => { const el = document.getElementById(id); if (el) { el.style.display = "none"; } }); root.querySelectorAll('div[class^="adWrapper"]').forEach(el => el.remove()); root.querySelectorAll('#rightadc').forEach(el => el.remove()); root.querySelectorAll('div#rightadfloat').forEach(el => el.remove()); root.querySelectorAll('div.heaven-728-90.ninja-slider.mode-b.dark').forEach(el => el.remove()); root.querySelectorAll('iframe[src^="https://dec.2chan.net/bin/foot1_n.htm"]').forEach(iframe => { if (iframe.parentElement) { iframe.parentElement.remove(); } }); root.querySelectorAll('div[id*="ad" i]').forEach(el => el.remove()); root.querySelectorAll('div[style*="width:680px"][style*="margin: 0 auto"]').forEach(el => el.remove()); } function cleanseJumpLinks(root = document) { const jumpLinks = root.querySelectorAll('a[href^="/bin/jump.php?"]'); jumpLinks.forEach(link => { let href = link.getAttribute('href'); let cleaned = href.replace(/^\/bin\/jump\.php\?/, ''); link.setAttribute('href', cleaned); }); } function isSrcLink(anchor) { return /\.(png|jpe?g|gif|mp4|webm)$/i.test(anchor.href); } function addDownloadButtons(root = document) { const allAnchors = Array.from( root.querySelectorAll('a[href*="/src/"], a[href*="fu"]') ).filter(isSrcLink); const filenameAnchors = allAnchors.filter(anchor => !anchor.querySelector('img')); filenameAnchors.forEach(anchor => { if (anchor.nextElementSibling && anchor.nextElementSibling.classList?.contains('download-btn')) { return; } const downloadLink = document.createElement('a'); downloadLink.href = anchor.href; downloadLink.download = ''; downloadLink.className = 'download-btn'; downloadLink.style.display = 'inline'; downloadLink.style.verticalAlign = 'middle'; downloadLink.style.marginLeft = '5px'; if (/\.(mp4|webm)$/i.test(downloadLink.href)) { downloadLink.addEventListener('click', function(e) { e.preventDefault(); fetch(downloadLink.href) .then(resp => resp.blob()) .then(blob => { const blobUrl = window.URL.createObjectURL(blob); const a = document.createElement('a'); a.href = blobUrl; a.download = downloadLink.href.split('/').pop(); document.body.appendChild(a); a.click(); a.remove(); window.URL.revokeObjectURL(blobUrl); }) .catch(err => console.error("Download failed", err)); }); } const svgNS = "http://www.w3.org/2000/svg"; const svg = document.createElementNS(svgNS, "svg"); svg.setAttribute("xmlns", svgNS); svg.setAttribute("viewBox", "0 0 512 512"); svg.setAttribute("aria-hidden", "true"); svg.setAttribute("focusable", "false"); svg.style.width = "1em"; svg.style.height = "1em"; svg.style.verticalAlign = "middle"; svg.style.fill = "currentColor"; const path = document.createElementNS(svgNS, "path"); path.setAttribute("d", "M288 32c0-17.7-14.3-32-32-32s-32 14.3-32 32l0 242.7-73.4-73.4c-12.5-12.5-32.8-12.5-45.3 0s-12.5 32.8 0 45.3l128 128c12.5 12.5 32.8 12.5 45.3 0l128-128c12.5-12.5 12.5-32.8 0-45.3s-32.8-12.5-45.3 0L288 274.7 288 32zM64 352c-35.3 0-64 28.7-64 64l0 32c0 35.3 28.7 64 64 64l384 0c35.3 0 64-28.7 64-64l0-32c0-35.3-28.7-64-64-64l-101.5 0-45.3 45.3c-25 25-65.5 25-90.5 0L165.5 352 64 352zm368 56a24 24 0 1 1 0 48 24 24 0 1 1 0-48z"); svg.appendChild(path); downloadLink.appendChild(svg); anchor.insertAdjacentElement('afterend', downloadLink); }); } function processPost(td) { const bq = td.querySelector("blockquote"); if (!bq) return; const originalHTML = bq.innerHTML; const lines = originalHTML.split(/<br\s*\/?>/i); const imageFiles = new Set(); const videoFiles = new Set(); lines.forEach(line => { const div = document.createElement("div"); div.innerHTML = line; const text = div.textContent.trim(); if (text.startsWith(">") || text.startsWith(">")) return; const matches = text.match(/\b(fu?\d+\.\w+)\b/gi); if (!matches) return; matches.forEach(fullname => { const ext = (fullname.split(".").pop() || "").toLowerCase(); if (ext === "mp4" || ext === "webm") { videoFiles.add(fullname); } else { imageFiles.add(fullname); } }); }); if (imageFiles.size > 0) { td.querySelectorAll("a[href*='/up/src/'], a[href*='/up2/src/']").forEach(a => { if (!bq.contains(a)) { a.remove(); } }); const container = document.createElement("div"); container.style.margin = "0"; container.style.padding = "0"; container.style.display = "block"; Array.from(imageFiles).forEach((filename, index, arr) => { const baseUrl = filename.toLowerCase().startsWith("fu") ? "https://dec.2chan.net/up2/src/" : "https://dec.2chan.net/up/src/"; const imageUrl = baseUrl + filename; const thumbLink = document.createElement("a"); thumbLink.href = imageUrl; thumbLink.target = "_blank"; const img = document.createElement("img"); img.src = imageUrl; img.setAttribute("border", "0"); img.setAttribute("align", "left"); img.setAttribute("hspace", "20"); img.setAttribute("alt", "外部画像"); img.setAttribute("loading", "lazy"); img.style.maxWidth = "200px"; img.style.maxHeight = "200px"; thumbLink.appendChild(img); container.appendChild(thumbLink); if (index < arr.length - 1) { container.appendChild(document.createElement("br")); } }); td.insertBefore(container, bq); bq.style.marginLeft = "241px"; } let replacedHTML = originalHTML; replacedHTML = replacedHTML.replace( /\b(fu?\d+\.(mp4|webm))\b/gi, (match, filename) => { const baseUrl = filename.toLowerCase().startsWith("fu") ? "https://dec.2chan.net/up2/src/" : "https://dec.2chan.net/up/src/"; const videoUrl = baseUrl + filename; return `<a href="${videoUrl}" target="_blank">${filename}</a>` + ` <span class="embed-video-link" data-url="${videoUrl}" style="cursor:pointer;">(動画)</span>`; } ); replacedHTML = replacedHTML.replace( /\b(fu?\d+\.(?!mp4|webm)\w+)\b/gi, (match, filename) => { const baseUrl = filename.toLowerCase().startsWith("fu") ? "https://dec.2chan.net/up2/src/" : "https://dec.2chan.net/up/src/"; const imageUrl = baseUrl + filename; return `<a href="${imageUrl}" target="_blank">${filename}</a>`; } ); bq.innerHTML = replacedHTML; bq.querySelectorAll(".embed-video-link").forEach(el => { el.addEventListener("click", function() { if (this.nextSibling && this.nextSibling.className === "video-iframe-wrap") { this.nextSibling.remove(); return; } const iframeWrap = document.createElement("div"); iframeWrap.className = "video-iframe-wrap"; iframeWrap.style.position = "relative"; iframeWrap.style.marginTop = "5px"; const closeBtn = document.createElement("div"); closeBtn.textContent = "✖"; closeBtn.style.position = "absolute"; closeBtn.style.top = "3px"; closeBtn.style.right = "5px"; closeBtn.style.cursor = "pointer"; closeBtn.style.backgroundColor = "rgba(0,0,0,0.6)"; closeBtn.style.color = "#fff"; closeBtn.style.padding = "2px 6px"; closeBtn.style.borderRadius = "3px"; closeBtn.style.zIndex = "10"; closeBtn.onclick = () => iframeWrap.remove(); const iframe = document.createElement("iframe"); iframe.src = this.dataset.url; iframe.style.width = "400px"; iframe.style.height = "300px"; iframe.style.border = "1px solid #ccc"; iframe.style.borderRadius = "3px"; iframeWrap.appendChild(closeBtn); iframeWrap.appendChild(iframe); this.parentNode.insertBefore(iframeWrap, this.nextSibling); }); }); } function updateFileSizes(root = document) { const elements = root.querySelectorAll('.thre, .rtd'); elements.forEach(el => { el.innerHTML = el.innerHTML.replace(/(-\()(\d+)\s*B\)/g, (match, prefix, num) => { let kb = (parseInt(num, 10) / 1024).toFixed(1); return prefix + kb + " KB)"; }); }); } function attachFullSizeHover(root = document) { const allAnchors = Array.from(root.querySelectorAll('a[href*="/src/"], a[href*="2chan.net/up2/src/"]')).filter(anchor => { return /\.(jpg|jpeg|png|gif)$/i.test(anchor.href); }); const thumbnails = allAnchors .filter(anchor => anchor.querySelector('img')) .map(anchor => anchor.querySelector('img')); thumbnails.forEach(img => { if (img.dataset.fullsizeHoverAttached) return; img.addEventListener('mouseenter', e => { showFullSizePreview(e.currentTarget); }); img.addEventListener('mouseleave', e => { hideFullSizePreview(); }); img.dataset.fullsizeHoverAttached = "true"; }); } function showFullSizePreview(img) { hideFullSizePreview(); const fullSrc = img.parentElement.href; const preview = document.createElement('img'); preview.id = 'fullsize-preview'; preview.src = fullSrc; preview.style.position = 'fixed'; preview.style.top = '50%'; preview.style.left = '50%'; preview.style.transform = 'translate(-50%, -50%)'; preview.style.maxWidth = '90vw'; preview.style.maxHeight = '90vh'; preview.style.zIndex = '2147483647'; preview.style.border = '1px solid #ccc'; preview.style.pointerEvents = 'none'; document.body.appendChild(preview); } function hideFullSizePreview() { const preview = document.getElementById('fullsize-preview'); if (preview) preview.remove(); } function autoReloadThread() { if (document.title.includes("404 File Not Found")) return; const reloadAnchor = document.querySelector('#contres a'); if (reloadAnchor) { const onClickAttr = reloadAnchor.getAttribute('onClick'); const match = onClickAttr && onClickAttr.match(/scrlf\((\d+)\)/); if (match) { const threadId = match[1]; if (typeof unsafeWindow.scrlf === 'function') { unsafeWindow.scrlf(threadId); } } } } setInterval(autoReloadThread, 5000); function changeFavicon(newIconData) { let link = document.querySelector("link[rel*='icon']"); if (!link) { link = document.createElement("link"); link.rel = "icon"; document.head.appendChild(link); } link.href = newIconData; } function changeFaviconIfCondition() { if (thresholdMode) return; const maxresElement = document.querySelector('span.maxres'); const contdispElement = document.querySelector('#contdisp'); if ((maxresElement && maxresElement.textContent.includes("上限1000レスに達しました")) || (contdispElement && contdispElement.textContent.includes("スレッドがありません"))) { changeFavicon(thresholdFaviconBase64); thresholdMode = true; } } setInterval(changeFaviconIfCondition, 1000); function checkSodCondition() { if (document.title.includes("404 File Not Found")) return; const sodElements = document.querySelectorAll('.sod'); sodElements.forEach(el => { if (el.textContent.trim() === "そうだねx0" && !el.dataset.sodAlerted) { alert("お使いのipアドレスからそうだねできません"); el.dataset.sodAlerted = "true"; } }); } setInterval(checkSodCondition, 1000); function addQuoteLinks() { if (document.title.includes("404 File Not Found")) return; const allPosts = Array.from(document.querySelectorAll('.thre, .rtd')) .filter(post => post.querySelector('span.cno')); let postDict = {}; allPosts.forEach(post => { const cnoEl = post.querySelector('span.cno'); if (cnoEl) { const num = cnoEl.textContent.trim().replace(/^No\./, ''); if (!post.id) { post.id = "post-" + num; } post.dataset.postNumber = num; const isOP = !post.closest('table') && post.textContent.includes('画像ファイル名'); if (isOP) { post.dataset.isOP = "true"; } postDict[num] = post; } }); const replyPosts = allPosts.filter(post => post.closest("table") !== null); let quotes = []; replyPosts.forEach(post => { const quotePostNum = post.dataset.postNumber; const fonts = Array.from(post.querySelectorAll('blockquote font[color="#789922"]')); fonts.forEach(font => { let text = font.textContent.trim(); text = text.replace(/^>+/, '').replace(/^(>)+/, '').trim(); if (text.indexOf('\n') !== -1) { text = text.split('\n')[0].trim(); } if (!text) return; let type = "text"; let value = text; const numMatch = text.match(/^(?:No\.?\s*)?(\d{8,})$/); if (numMatch) { type = "number"; value = numMatch[1]; } else if (text.match(/\.(jpg|jpeg|png|gif|webp|mp4|webm)$/i)) { type = "filename"; value = text; } quotes.push({ quotePostNum, type, value, sourcePost: post }); }); }); quotes.sort((a, b) => parseInt(a.quotePostNum) - parseInt(b.quotePostNum)); allPosts.forEach(originalPost => { const origPostNum = originalPost.dataset.postNumber; const isOP = originalPost.dataset.isOP === "true"; let lines = []; let anchors = []; let originalFilename = ''; if (isOP) { const filenameAnchor = originalPost.querySelector('a[href*="/src/"]'); if (filenameAnchor) { originalFilename = filenameAnchor.textContent.trim(); } const block = originalPost.querySelector('blockquote'); if (block) { lines = block.innerText.split('\n').map(l => l.trim()).filter(l => l.length > 0); } } else { const block = originalPost.querySelector('blockquote'); if (block) { lines = block.innerText.split('\n').map(l => l.trim()).filter(l => l.length > 0); } else { lines = originalPost.innerText.split('\n').map(l => l.trim()).filter(l => l.length > 0); } anchors = Array.from(originalPost.querySelectorAll('a[href*="/src/"]')); } let insertAfterElement; if (isOP) { insertAfterElement = originalPost.querySelector('.cntd'); } else { insertAfterElement = originalPost.querySelector('.sod'); } if (!insertAfterElement) return; for (let i = quotes.length - 1; i >= 0; i--) { let quote = quotes[i]; if (quote.sourcePost.id === originalPost.id) continue; let matchFound = false; if (quote.type === "number") { matchFound = (origPostNum === quote.value); } else if (quote.type === "filename") { if (isOP) { matchFound = (originalFilename === quote.value); } else { matchFound = anchors.some(a => a.getAttribute('href').includes(quote.value) || a.textContent.trim() === quote.value ); } } else { matchFound = lines.includes(quote.value); } if (matchFound) { if (!originalPost.querySelector(`a[data-quote-post="${quote.quotePostNum}"]`)) { const link = document.createElement('a'); link.href = "javascript:void(0);"; link.textContent = ">>" + quote.quotePostNum; link.style.marginLeft = "5px"; link.dataset.quotePost = quote.quotePostNum; link.addEventListener('click', () => { const target = document.getElementById("post-" + quote.quotePostNum); if (target) { target.scrollIntoView({ behavior: "smooth" }); highlightPost(target); } }); link.addEventListener('mouseenter', handleQuoteLinkMouseEnter); link.addEventListener('mousemove', handleQuoteLinkMouseMove); link.addEventListener('mouseleave', handleQuoteLinkMouseLeave); insertAfterElement.parentNode.insertBefore(link, insertAfterElement.nextSibling); } } } }); } setInterval(addQuoteLinks, 1000); function insertQuoteBreaks() { var posts = document.querySelectorAll('td.rtd'); posts.forEach(function(post) { var quotes = post.querySelectorAll('a[data-quote-post]'); quotes.forEach(function(quote, index) { if ((index + 1) % 10 === 0) { if (!quote.nextSibling || quote.nextSibling.nodeName !== 'BR') { quote.insertAdjacentHTML('afterend', '<br>'); } } }); }); } function makeDraggable(element, handle) { handle = handle || element; handle.style.cursor = "move"; let offsetX, offsetY; handle.addEventListener("mousedown", function(e) { offsetX = e.clientX - element.getBoundingClientRect().left; offsetY = e.clientY - element.getBoundingClientRect().top; document.addEventListener("mousemove", moveHandler); document.addEventListener("mouseup", upHandler); }); function moveHandler(e) { element.style.left = (e.clientX - offsetX) + "px"; element.style.top = (e.clientY - offsetY) + "px"; element.style.bottom = "auto"; element.style.right = "auto"; } function upHandler(e) { document.removeEventListener("mousemove", moveHandler); document.removeEventListener("mouseup", upHandler); } } function setupReplyForm() { const replyForm = document.getElementById("fm"); if (replyForm) { const reszbElements = replyForm.querySelectorAll("span#reszb"); reszbElements.forEach(el => el.remove()); const ftb2 = replyForm.querySelector("table.ftb2"); if (ftb2) { ftb2.parentNode.removeChild(ftb2); const hrElement = document.querySelector("hr"); if (hrElement) { hrElement.insertAdjacentElement("afterend", ftb2); } else { document.body.insertBefore(ftb2, document.body.firstChild); } } const container = document.createElement("div"); container.id = "replyContainer"; container.style.position = "fixed"; container.style.bottom = "20px"; container.style.right = "20px"; container.style.background = "#fff"; container.style.border = "1px solid #ccc"; container.style.padding = "5px"; container.style.zIndex = "15000"; container.style.boxShadow = "2px 2px 10px rgba(0,0,0,0.5)"; const handle = document.createElement("div"); handle.id = "replyDragHandle"; handle.textContent = "返信フォーム(ドラッグで移動)"; handle.style.background = "#ccc"; handle.style.padding = "4px"; handle.style.cursor = "move"; handle.style.fontSize = "0.9em"; handle.style.textAlign = "center"; handle.style.userSelect = "none"; container.appendChild(handle); container.appendChild(replyForm); document.body.appendChild(container); makeDraggable(container, handle); } } if (window.location.href.indexOf("/res/") !== -1) { if (document.readyState === "loading") { document.addEventListener("DOMContentLoaded", setupReplyForm); } else { setupReplyForm(); } } function addArchiveLinksTo404() { if (!document.title.includes("404 File Not Found")) return; let h1s = document.querySelectorAll("h1"); let targetH1 = null; h1s.forEach(h => { if (h.textContent.includes("掲示板に戻る")) { targetH1 = h; } }); if (!targetH1) return; let hr = targetH1.nextElementSibling; while (hr && hr.tagName !== "HR") { hr = hr.nextElementSibling; } if (!hr) return; let archiveHeader = document.createElement("h1"); archiveHeader.textContent = "過去ログを取得"; hr.insertAdjacentElement("afterend", archiveHeader); let linkContainer = document.createElement("div"); linkContainer.style.marginTop = "10px"; let path = window.location.pathname; let regex = /\/([^\/]+)\/res\/(\d+)\.htm/i; let match = path.match(regex); if (!match) return; let board = match[1]; let threadId = match[2]; let host = window.location.host; let links = []; if (host.includes("may.2chan.net") && board === "b") { links.push({ text: "futakuro", url: `https://kako.futakuro.com/futa/may_b/${threadId}/` }); links.push({ text: "ftbucket (may1)", url: `https://may1.ftbucket.info/may/cont/may.2chan.net_b_res_${threadId}/index.htm` }); links.push({ text: "ftbucket (may2)", url: `https://may2.ftbucket.info/may/cont/may.2chan.net_b_res_${threadId}/index.htm` }); links.push({ text: "ftbucket (may3)", url: `https://may3.ftbucket.info/may/cont/may.2chan.net_b_res_${threadId}/index.htm` }); links.push({ text: "futabaforest", url: `https://futabaforest.net/b/res/${threadId}.htm` }); } else if (host.includes("img.2chan.net") && board === "b") { links.push({ text: "ftbucket (c3)", url: `https://c3.ftbucket.info/img/cont/img.2chan.net_b_res_${threadId}/index.htm` }); links.push({ text: "futakuro", url: `https://kako.futakuro.com/futa/img_b/${threadId}/` }); } else if (host.includes("jun.2chan.net") && board === "jun") { links.push({ text: "ftbucket (c3)", url: `https://c3.ftbucket.info/jun/cont/jun.2chan.net_jun_res_${threadId}/index.htm` }); } else if (host.includes("dec.2chan.net") && board === "55") { links.push({ text: "ftbucket (c3)", url: `https://c3.ftbucket.info/dec55/cont/dec.2chan.net_55_res_${threadId}/index.htm` }); } else if (host.includes("dec.2chan.net") && board === "60") { links.push({ text: "ftbucket (c3)", url: `https://c3.ftbucket.info/dec60/cont/dec.2chan.net_60_res_${threadId}/index.htm` }); } else { links.push({ text: "ftbucket (c3)", url: `https://c3.ftbucket.info/other/cont/${host}_${board}_res_${threadId}/index.htm` }); } links.forEach(linkInfo => { let a = document.createElement("a"); a.href = linkInfo.url; a.textContent = linkInfo.text; a.style.marginRight = "10px"; a.target = "_blank"; linkContainer.appendChild(a); }); archiveHeader.insertAdjacentElement("afterend", linkContainer); } addArchiveLinksTo404(); const originalTitle = document.title; let maxSeenIndex = -1; let prevUnread = 0; let initialUnread = 0; let initialUnreadSet = false; let wasZeroOnBlur = false; function isInViewport(el) { const rect = el.getBoundingClientRect(); return rect.bottom > 0 && rect.top < window.innerHeight; } function updateUnreadCountNew() { if (window.location.href.indexOf("/res/") === -1) return; if (document.title.includes("404 File Not Found")) { changeFavicon(thresholdFaviconBase64); return; } const contdispElement = document.querySelector('#contdisp'); if (contdispElement && contdispElement.textContent.includes("スレッドがありません")) { document.title = `【落ち】 ${originalTitle}`; return; } const replies = Array.from(document.querySelectorAll(".thre, .rtd")) .filter(post => post.closest("table") !== null); const totalReplies = replies.length; let visibleIndex = -1; for (let i = 0; i < replies.length; i++) { if (isInViewport(replies[i])) { visibleIndex = i; } } if (document.hasFocus()) { if (visibleIndex > maxSeenIndex) { maxSeenIndex = visibleIndex; } if (!initialUnreadSet) { initialUnread = totalReplies - (maxSeenIndex + 1); initialUnreadSet = true; } } else { if (!initialUnreadSet) { maxSeenIndex = totalReplies - 1; initialUnread = 0; initialUnreadSet = true; } } const unread = totalReplies - (maxSeenIndex + 1); document.title = `(${unread}) ${originalTitle}`; if (!thresholdMode && !document.hasFocus() && wasZeroOnBlur && prevUnread === 0 && unread > 0) { changeFavicon(newReplyFaviconBase64); } if (document.hasFocus() && unread === 0 && !thresholdMode) { changeFavicon(originalFavicon); } prevUnread = unread; } window.addEventListener("scroll", updateUnreadCountNew); setInterval(updateUnreadCountNew, 1000); window.addEventListener("blur", () => { if (prevUnread === 0) { wasZeroOnBlur = true; } }); window.addEventListener("focus", () => { wasZeroOnBlur = false; if (prevUnread === 0 && !thresholdMode) { changeFavicon(originalFavicon); } }); function addYouTubeEmbeds(root = document) { const ytRegex = /(?:https?:\/\/)?(?:www\.)?(?:youtube\.com\/(?:watch\?v=|live\/)|youtu\.be\/)([\w-]{11})/; const processedLinks = new Set(); function createFloatingWindow(videoId) { const floatDiv = document.createElement('div'); floatDiv.style.cssText = ` position: fixed; width: 480px; height: 270px; z-index: 9999; background: white; box-shadow: 0 0 10px rgba(0,0,0,0.5); overflow: hidden; cursor: move; `; const handle = document.createElement('div'); handle.style.cssText = ` position: absolute; top: 0; left: 0; right: 0; height: 20px; background: #eee0d7; cursor: move; user-select: none; `; const closeButton = document.createElement('button'); closeButton.textContent = 'X'; closeButton.style.cssText = ` position: absolute; top: 0; right: 0; padding: 2px 6px; background: #781208; color: white; border: none; cursor: pointer; z-index: 10000; `; closeButton.addEventListener('click', () => floatDiv.remove()); const iframe = document.createElement('iframe'); iframe.style.cssText = ` width: 100%; height: calc(100% - 20px); border: none; margin-top: 20px; `; iframe.src = `https://www.youtube.com/embed/${videoId}?rel=0&wmode=opaque`; iframe.setAttribute('allow', 'accelerometer; autoplay; encrypted-media; gyroscope; picture-in-picture'); iframe.setAttribute('allowfullscreen', ''); handle.appendChild(closeButton); floatDiv.appendChild(handle); floatDiv.appendChild(iframe); const leftHandle = document.createElement('div'); leftHandle.style.cssText = ` position: absolute; top: 20px; left: 0; width: 8px; height: calc(100% - 20px); cursor: ew-resize; z-index: 10001; `; const rightHandle = document.createElement('div'); rightHandle.style.cssText = ` position: absolute; top: 20px; right: 0; width: 8px; height: calc(100% - 20px); cursor: ew-resize; z-index: 10001; `; const bottomHandle = document.createElement('div'); bottomHandle.style.cssText = ` position: absolute; bottom: 0; left: 0; height: 8px; width: 100%; cursor: ns-resize; z-index: 10001; `; floatDiv.appendChild(leftHandle); floatDiv.appendChild(rightHandle); floatDiv.appendChild(bottomHandle); document.body.appendChild(floatDiv); const startX = (window.innerWidth - 480) / 2; const startY = (window.innerHeight - 270) / 2; floatDiv.style.left = `${startX}px`; floatDiv.style.top = `${startY}px`; let isDragging = false; let startXPos = 0; let startYPos = 0; let initialMouseX = 0; let initialMouseY = 0; handle.addEventListener('mousedown', startDrag); document.addEventListener('mousemove', drag); document.addEventListener('mouseup', stopDrag); function startDrag(e) { isDragging = true; initialMouseX = e.clientX; initialMouseY = e.clientY; startXPos = floatDiv.offsetLeft; startYPos = floatDiv.offsetTop; floatDiv.style.cursor = 'grabbing'; } function drag(e) { if (isDragging) { const deltaX = e.clientX - initialMouseX; const deltaY = e.clientY - initialMouseY; floatDiv.style.left = `${startXPos + deltaX}px`; floatDiv.style.top = `${startYPos + deltaY}px`; } } function stopDrag() { isDragging = false; floatDiv.style.cursor = 'move'; } leftHandle.addEventListener('mousedown', function(e) { e.preventDefault(); e.stopPropagation(); let startX = e.clientX; let startLeft = floatDiv.offsetLeft; let startWidth = floatDiv.offsetWidth; function doResize(ev) { let deltaX = ev.clientX - startX; let newWidth = startWidth - deltaX; let newLeft = startLeft + deltaX; if(newWidth < 200) { newWidth = 200; newLeft = startLeft + (startWidth - 200); } floatDiv.style.width = newWidth + 'px'; floatDiv.style.left = newLeft + 'px'; } function stopResize() { document.removeEventListener('mousemove', doResize); document.removeEventListener('mouseup', stopResize); } document.addEventListener('mousemove', doResize); document.addEventListener('mouseup', stopResize); }); rightHandle.addEventListener('mousedown', function(e) { e.preventDefault(); e.stopPropagation(); let startX = e.clientX; let startWidth = floatDiv.offsetWidth; function doResize(ev) { let deltaX = ev.clientX - startX; let newWidth = startWidth + deltaX; if(newWidth < 200) { newWidth = 200; } floatDiv.style.width = newWidth + 'px'; } function stopResize() { document.removeEventListener('mousemove', doResize); document.removeEventListener('mouseup', stopResize); } document.addEventListener('mousemove', doResize); document.addEventListener('mouseup', stopResize); }); bottomHandle.addEventListener('mousedown', function(e) { e.preventDefault(); e.stopPropagation(); let startY = e.clientY; let startHeight = floatDiv.offsetHeight; function doResize(ev) { let deltaY = ev.clientY - startY; let newHeight = startHeight + deltaY; if(newHeight < 150) { newHeight = 150; } floatDiv.style.height = newHeight + 'px'; } function stopResize() { document.removeEventListener('mousemove', doResize); document.removeEventListener('mouseup', stopResize); } document.addEventListener('mousemove', doResize); document.addEventListener('mouseup', stopResize); }); } function processLink(link) { if (processedLinks.has(link)) return; const match = link.href.match(ytRegex); if (!match) return; const videoId = match[1]; const containerId = `yt-${videoId}-${Date.now()}`; const toggle = document.createElement('a'); toggle.href = "javascript:void(0);"; toggle.textContent = "(埋め込み)"; toggle.style.marginLeft = "5px"; const floatButton = document.createElement('a'); floatButton.href = "javascript:void(0);"; floatButton.textContent = "(フロート表示)"; floatButton.style.marginLeft = "5px"; const container = document.createElement('div'); container.id = containerId; container.style.cssText = ` margin: 10px 0; position: relative; width: 100%; max-width: 640px; display: none; `; const iframe = document.createElement('iframe'); iframe.style.cssText = ` width: 100%; height: 360px; border: none; border-radius: 4px; `; iframe.src = `https://www.youtube.com/embed/${videoId}?rel=0&wmode=opaque`; iframe.setAttribute('allow', 'accelerometer; autoplay; encrypted-media; gyroscope; picture-in-picture'); iframe.setAttribute('allowfullscreen', ''); toggle.addEventListener('click', (e) => { e.preventDefault(); container.style.display = container.style.display === 'none' ? 'block' : 'none'; iframe.src = iframe.src; }); floatButton.addEventListener('click', (e) => { e.preventDefault(); createFloatingWindow(videoId); }); container.appendChild(iframe); link.parentNode.insertBefore(toggle, link.nextSibling); link.parentNode.insertBefore(floatButton, toggle.nextSibling); link.parentNode.insertBefore(container, floatButton.nextSibling); processedLinks.add(link); } root.querySelectorAll('a[href*="youtube"], a[href*="youtu.be"]').forEach(processLink); const observer = new MutationObserver(mutations => { mutations.forEach(mutation => { mutation.addedNodes.forEach(node => { if (node.nodeType === 1) { node.querySelectorAll('a[href*="youtube"], a[href*="youtu.be"]').forEach(processLink); } }); }); }); observer.observe(root, { childList: true, subtree: true }); } function resolveOtherThreadLinks(root = document) { if (window.location.href.indexOf("/res/") === -1) return; const candidateLinks = root.querySelectorAll('a[href$=".htm"]'); candidateLinks.forEach(link => { if (link.closest('#contres')) return; let url = link.href.replace(/^http:\/\//, "https://"); link.href = url; if (link.dataset.resolvedThread) return; let hostname; try { hostname = new URL(url).hostname; } catch (e) { return; } if (!hostname.endsWith(".2chan.net") || !url.includes("/res/")) return; const idMatch = url.match(/\/res\/(\d+)\.htm$/); if (!idMatch) return; const threadId = idMatch[1]; if (resolvedThreadCache[url]) { const cache = resolvedThreadCache[url]; if (cache.type === 'original') { link.textContent = `>>${threadId} 「${cache.title}」 (別スレ)`; } else if (cache.type === 'archive') { link.href = cache.archiveUrl; link.textContent = `>>${threadId} 「${cache.title}」 (過去ログ)`; } else { link.textContent = `>>${threadId} (別スレ) (過去ログが見つかりません)`; } link.dataset.resolvedThread = "true"; return; } GM_xmlhttpRequest({ method: "GET", url: url + (url.includes('?') ? '&' : '?') + '_=' + Date.now(), onload: function(response) { const parser = new DOMParser(); const doc = parser.parseFromString(response.responseText, "text/html"); const titleEl = doc.querySelector("title"); let title = titleEl ? titleEl.textContent.trim() : "タイトル取得失敗"; if (response.status === 200 && !title.includes("404 File Not Found")) { resolvedThreadCache[url] = { type: 'original', title: title }; updateLink(link, threadId, title, url, 'original'); } else { const archiveUrls = generateArchiveUrls(new URL(url).hostname, new URL(url).pathname.split('/')[1], threadId); checkArchiveUrls(archiveUrls, 0, function(archiveUrl, archiveTitle) { if (archiveUrl) { resolvedThreadCache[url] = { type: 'archive', title: archiveTitle, archiveUrl: archiveUrl }; updateLink(link, threadId, archiveTitle, archiveUrl, 'archive'); } else { resolvedThreadCache[url] = { type: 'not_found' }; updateLink(link, threadId, null, url, 'not_found'); } }); } }, onerror: function() { resolvedThreadCache[url] = { type: 'not_found' }; updateLink(link, threadId, null, url, 'not_found'); } }); }); } function generateArchiveUrls(host, board, threadId) { if (host.includes("may.2chan.net") && board === "b") { return [ `https://may1.ftbucket.info/may/cont/may.2chan.net_b_res_${threadId}/index.htm`, `https://may2.ftbucket.info/may/cont/may.2chan.net_b_res_${threadId}/index.htm`, `https://may3.ftbucket.info/may/cont/may.2chan.net_b_res_${threadId}/index.htm`, `https://kako.futakuro.com/futa/may_b/${threadId}/`, `https://futabaforest.net/b/res/${threadId}.htm` ]; } if (host.includes("img.2chan.net") && board === "b") { return [ `https://c3.ftbucket.info/img/cont/img.2chan.net_b_res_${threadId}/index.htm`, `https://kako.futakuro.com/futa/img_b/${threadId}/` ]; } if (host.includes("jun.2chan.net") && board === "jun") { return [ `https://c3.ftbucket.info/jun/cont/jun.2chan.net_jun_res_${threadId}/index.htm` ]; } if (host.includes("dec.2chan.net")) { return [ `https://c3.ftbucket.info/other/cont/${host.replace(/\./g, '_')}_${board}_res_${threadId}/index.htm` ]; } return []; } function checkArchiveUrls(urls, index, callback) { if (index >= urls.length) return callback(null, null); let options = { method: "HEAD", url: urls[index], onload: function(res) { if (res.status >= 200 && res.status < 400) { if (urls[index].includes("ftbucket.info")) { GM_xmlhttpRequest({ method: "GET", url: urls[index], responseType: "arraybuffer", onload: function(getRes) { let decoder = new TextDecoder("shift_jis"); let text = decoder.decode(getRes.response); const parser = new DOMParser(); const doc = parser.parseFromString(text, "text/html"); const titleEl = doc.querySelector("title"); let title = titleEl ? titleEl.textContent.trim() : ""; if (!title || title === "タイトル不明") { checkArchiveUrls(urls, index + 1, callback); } else { callback(urls[index], title); } }, onerror: function() { checkArchiveUrls(urls, index + 1, callback); } }); } else if (urls[index].includes("futakuro.com")) { GM_xmlhttpRequest({ method: "GET", url: urls[index], onload: function(getRes) { const parser = new DOMParser(); const doc = parser.parseFromString(getRes.responseText, "text/html"); const titleEl = doc.querySelector("title"); let title = titleEl ? titleEl.textContent.trim() : ""; if (!title || title === "タイトル不明") { checkArchiveUrls(urls, index + 1, callback); } else { callback(urls[index], title); } }, onerror: function() { checkArchiveUrls(urls, index + 1, callback); } }); } else if (urls[index].includes("futabaforest.net")) { options.timeout = 10000; options.ontimeout = function() { checkArchiveUrls(urls, index + 1, callback); }; GM_xmlhttpRequest({ method: "GET", url: urls[index], responseType: "text", onload: function(getRes) { const parser = new DOMParser(); const doc = parser.parseFromString(getRes.responseText, "text/html"); const titleEl = doc.querySelector("title"); let title = titleEl ? titleEl.textContent.trim() : ""; if (!title || title === "タイトル不明") { checkArchiveUrls(urls, index + 1, callback); } else { callback(urls[index], title); } }, onerror: function() { checkArchiveUrls(urls, index + 1, callback); } }); } else { GM_xmlhttpRequest({ method: "GET", url: urls[index], onload: function(getRes) { const parser = new DOMParser(); const doc = parser.parseFromString(getRes.responseText, "text/html"); const titleEl = doc.querySelector("title"); let title = titleEl ? titleEl.textContent.trim() : ""; if (!title || title === "タイトル不明") { checkArchiveUrls(urls, index + 1, callback); } else { callback(urls[index], title); } }, onerror: function() { checkArchiveUrls(urls, index + 1, callback); } }); } } else { checkArchiveUrls(urls, index + 1, callback); } }, onerror: function() { checkArchiveUrls(urls, index + 1, callback); } }; GM_xmlhttpRequest(options); } function updateLink(link, threadId, title, newUrl, type) { if (type === 'original') { link.textContent = `>>${threadId} 「${title}」 (別スレ)`; } else if (type === 'archive') { link.href = newUrl; link.textContent = `>>${threadId} 「${title}」 (過去ログ)`; } else { link.textContent = `>>${threadId} (別スレ) (過去ログが見つかりません)`; } link.dataset.resolvedThread = "true"; } function processNodes(root = document) { removeUnwantedElements(root); addDownloadButtons(root); updateFileSizes(root); attachFullSizeHover(); changeFaviconIfCondition(); addYouTubeEmbeds(root); cleanseJumpLinks(root); insertQuoteBreaks(root); root.querySelectorAll("td.rtd").forEach(processPost); if (window.location.href.indexOf("/res/") !== -1) { resolveOtherThreadLinks(root); } } processNodes(); const observer = new MutationObserver(mutations => { mutations.forEach(mutation => { mutation.addedNodes.forEach(node => { if (node.nodeType === 1) { processNodes(node); } }); }); }); observer.observe(document.body, { childList: true, subtree: true }); setInterval(autoReloadThread, 5000); setInterval(checkSodCondition, 1000); if (window.location.href.indexOf("/res/") !== -1) { const scrollKey = "2chan_lastScroll_" + window.location.pathname; window.addEventListener("load", () => { const lastScroll = localStorage.getItem(scrollKey); if (lastScroll) { window.scrollTo(0, parseInt(lastScroll, 10)); } }); window.addEventListener("scroll", () => { localStorage.setItem(scrollKey, window.pageYOffset); }); } if (window.location.hostname === "www.2chan.net" && window.location.pathname === "/index2.html") { const boardMapping = { "img_b": "二次元裏img", "dec_dec": "二次元裏dec", "jun_jun": "二次元裏jun", "may_b": "二次元裏may", "dat_b": "二次元裏dat", "dec_58": "転載不可", "dec_59": "転載可", "dec_img2": "二次元表", "may_id": "二次元ID", "dat_43": "二次元業界", "dat_20": "甘味", "dat_21": "ラーメン", "dat_23": "スピグラ", "dat_l": "壁紙二", "may_25": "麻雀", "may_26": "うま", "may_27": "ねこ", "zip_12": "サッカー", "zip_14": "自作絵裏", "zip_5": "えろげ", "zip_1": "野球", "zip_11": "自作絵", "zip_2": "ろぼ", "dec_63": "映画", "zip_3": "自作PC", "zip_32": "女装", "zip_15": "ばら", "zip_7": "ゆり", "zip_8": "やおい", "dec_65": "刀剣乱舞", "dec_64": "占い", "dec_66": "ファッション", "dec_67": "旅行", "dec_68": "子育て", "may_webm": "webm", "dec_71": "そうだね", "zip_p": "お絵かき", "nov_q": "落書き", "zip_z": "しょくぶつ", "dat_d": "どうぶつ", "dat_e": "のりもの", "dat_j": "二輪", "nov_37": "自転車", "dat_45": "カメラ", "dat_48": "家電", "dat_r": "鉄道", "dat_t": "料理", "dat_44": "おもちゃ", "dat_v": "模型", "nov_y": "模型裏", "jun_47": "模型裏裏", "dat_w": "虫", "dat_49": "アクア", "dec_62": "アウトドア", "dec_73": "VTuber", "dec_84": "ホロライブ", "dec_81": "合成音声", "dat_x": "3DCG", "dec_85": "人工知能", "nov_35": "政治", "nov_36": "経済", "dec_79": "宗教", "dat_38": "尹錫悦", "dec_80": "岸田文雄", "dec_50": "三次実況", "cgi_f": "軍", "may_39": "軍裏", "dec_74": "FGO", "dec_75": "アイマス", "dec_86": "ZOIDS", "dec_78": "ウメハラ総合", "jun_31": "ゲーム", "nov_28": "ネトゲ", "dec_56": "ソシャゲ", "dec_60": "艦これ", "dec_82": "任天堂", "dec_69": "モアイ", "dec_61": "ソニー", "dat_10": "ネットキャラ", "nov_34": "なりきり", "cgi_g": "特撮", "cgi_i": "flash", "may_40": "東方", "dec_55": "東方裏", "cgi_k": "壁紙", "cgi_m": "数学", "zip_6": "ニュース表", "dec_76": "昭和", "dec_77": "平成", "dec_53": "発電", "dec_52": "自然災害", "dec_83": "コロナ", "img_9": "雑談", "dec_70": "新板提案", "cgi_o": "二次元グロ", "jun_51": "二次元グロ裏", "cgi_u": "落書き裏", "jun_oe": "お絵sql", "jun_72": "お絵sqlip", "www_hinan": "ふたば避難所", "jun_junbi": "準備" }; const RSS_URL = 'https://futapo.futakuro.com/ranking/index.rdf'; function fetchRSS() { GM_xmlhttpRequest({ method: 'GET', url: RSS_URL, onload: function(response) { if (response.status === 200) { const parser = new DOMParser(); const xmlDoc = parser.parseFromString(response.responseText, "text/xml"); const items = xmlDoc.getElementsByTagName('item'); let threads = []; for (let i = 0; i < Math.min(9, items.length); i++) { const item = items[i]; const titleRaw = item.getElementsByTagName('title')[0].textContent; const title = titleRaw.replace(/^\d+位[::]?/, '').trim(); let link = item.getElementsByTagName('link')[0].textContent; if(link.indexOf('//') === 0){ link = 'https:' + link; } let match = link.match(/https?:\/\/([^\.]+)\.2chan\.net\/([^\/]+)\/res\//); let subdomain = '', identifier = '', boardName = 'unknown'; if(match) { subdomain = match[1]; identifier = match[2]; let boardKey = subdomain + "_" + identifier; if(boardMapping[boardKey]){ boardName = boardMapping[boardKey]; } else { boardName = identifier; } } const descCDATA = item.getElementsByTagName('description')[0].textContent; let imgSrc = ''; const imgMatch = descCDATA.match(/<img\s+src=["']([^"']+)["']/); if (imgMatch) { imgSrc = imgMatch[1]; } threads.push({ subdomain, identifier, boardName, imgSrc, title, link }); } insertGrid(threads); } } }); } function insertGrid(threads) { const gridTable = document.createElement('table'); gridTable.style.margin = '20px auto'; gridTable.style.borderCollapse = 'collapse'; gridTable.style.fontFamily = 'sans-serif'; gridTable.style.fontSize = '10px'; gridTable.style.width = '70%'; gridTable.style.maxWidth = '800px'; gridTable.style.boxShadow = '0 0 10px rgba(0,0,0,0.1)'; gridTable.style.border = '1px solid #000000'; gridTable.style.tableLayout = 'fixed'; const thead = document.createElement('thead'); const headerRow = document.createElement('tr'); const headerCell = document.createElement('th'); headerCell.colSpan = 3; headerCell.textContent = '勢い上昇中のスレ'; headerCell.style.fontSize = '14px'; headerCell.style.fontWeight = 'bold'; headerCell.style.color = '#fff'; headerCell.style.backgroundColor = '#781225'; headerCell.style.padding = '2px 5px'; headerCell.style.textAlign = 'center'; headerRow.appendChild(headerCell); thead.appendChild(headerRow); gridTable.appendChild(thead); const tbody = document.createElement('tbody'); for (let row = 0; row < 3; row++) { const tr = document.createElement('tr'); for (let col = 0; col < 3; col++) { const td = document.createElement('td'); td.style.border = '1px solid #000000'; td.style.padding = '4px'; td.style.textAlign = 'center'; td.style.verticalAlign = 'top'; td.style.width = '33%'; td.style.boxSizing = 'border-box'; td.style.height = '200px'; td.style.overflow = 'hidden'; const index = row * 3 + col; if (threads[index]) { const { subdomain, identifier, boardName, imgSrc, title, link } = threads[index]; const boardLink = document.createElement('a'); boardLink.href = 'https://' + subdomain + '.2chan.net/' + identifier + '/'; boardLink.target = '_blank'; boardLink.textContent = boardName; boardLink.style.fontSize = '14px'; boardLink.style.fontWeight = 'bold'; boardLink.style.marginBottom = '2px'; boardLink.style.display = 'block'; td.appendChild(boardLink); const anchor = document.createElement('a'); anchor.href = link; anchor.target = '_blank'; anchor.style.textDecoration = 'none'; anchor.style.color = 'inherit'; if (imgSrc) { const imgElem = document.createElement('img'); imgElem.src = imgSrc; imgElem.style.maxWidth = '90%'; imgElem.style.maxHeight = '100px'; imgElem.style.height = 'auto'; imgElem.style.marginBottom = '2px'; anchor.appendChild(imgElem); } const titleElem = document.createElement('div'); titleElem.textContent = title; titleElem.style.marginTop = '2px'; titleElem.style.fontSize = '16px'; anchor.appendChild(titleElem); td.appendChild(anchor); } tr.appendChild(td); } tbody.appendChild(tr); } gridTable.appendChild(tbody); const existingTables = document.getElementsByTagName('table'); if (existingTables.length > 0) { const targetTable = existingTables[0]; const spacer = document.createElement('div'); spacer.style.height = '20px'; targetTable.parentNode.insertBefore(spacer, targetTable.nextSibling); targetTable.parentNode.insertBefore(gridTable, spacer.nextSibling); } } window.addEventListener('load', fetchRSS, false); } GM_addStyle(` .lds-ring, .lds-ring div { box-sizing: border-box; } .lds-ring { display: inline-block; position: relative; width: 14px; height: 14px; } .lds-ring div { box-sizing: border-box; display: block; position: absolute; width: 15px; height: 15px; border: 2px solid currentColor; border-radius: 50%; animation: lds-ring 1.2s cubic-bezier(0.5, 0, 0.5, 1) infinite; border-color: currentColor transparent transparent transparent; } .lds-ring div:nth-child(1) { animation-delay: -0.45s; } .lds-ring div:nth-child(2) { animation-delay: -0.3s; } .lds-ring div:nth-child(3) { animation-delay: -0.15s; } @keyframes lds-ring { 0% { transform: rotate(0deg); } 100% { transform: rotate(360deg); } } `); if (!window.location.href.includes('futaba.php?mode=cat')) return; if (window.location.href.includes('mode=catset')) return; const urlParams = new URLSearchParams(window.location.search); const sortParam = urlParams.get('sort'); let existingThreads = new Set(); let fetchedThreads = []; let board = ''; let subdomain = ''; let useCatVersion = false; const catalogTable = document.querySelector('#cattable tbody'); if (!catalogTable) return; function recordExistingThreads() { catalogTable.querySelectorAll('td').forEach(td => { const link = td.querySelector('a[href*="res/"]'); if (link) { existingThreads.add(link.getAttribute('href')); } }); } recordExistingThreads(); let maxColumns = 1; const firstRow = catalogTable.querySelector('tr'); if (firstRow) { maxColumns = firstRow.children.length; } const firstImg = catalogTable.querySelector('td img'); if (firstImg && firstImg.src.includes('/cat/')) { useCatVersion = true; } const preExistingSmall = catalogTable.querySelector('td small'); const includeTitle = !!(preExistingSmall && preExistingSmall.textContent.trim().length > 0); let maxTitleLength = 100; if (preExistingSmall) { maxTitleLength = preExistingSmall.textContent.trim().length; } const searchBox = document.createElement('input'); searchBox.type = 'text'; searchBox.placeholder = '絞り込み....'; searchBox.style.margin = '5px'; searchBox.style.padding = '3px'; searchBox.style.fontSize = '90%'; const targetLink = document.querySelector('a[href*="futaba.php?mode=cat"][href*="sort=9"][href*="guid=on"]'); if (targetLink) { targetLink.parentNode.insertBefore(searchBox, targetLink.nextSibling); } else { document.body.insertBefore(searchBox, document.body.firstChild); } searchBox.addEventListener('input', () => { const query = searchBox.value.trim().toLowerCase(); const tds = catalogTable.querySelectorAll('td'); tds.forEach(td => { if (query === '') { td.style.display = ''; } else { const small = td.querySelector('small'); const text = small ? small.textContent.toLowerCase() : ''; const href = td.querySelector('a') ? td.querySelector('a').getAttribute('href').toLowerCase() : ''; td.style.display = (text.includes(query) || href.includes(query)) ? '' : 'none'; } }); }); const loadingDialog = document.createElement('div'); loadingDialog.style.position = 'fixed'; loadingDialog.style.top = '46px'; loadingDialog.style.right = '20px'; loadingDialog.style.background = 'rgba(0,0,0,0.7)'; loadingDialog.style.color = '#fff'; loadingDialog.style.padding = '10px 15px'; loadingDialog.style.borderRadius = '5px'; loadingDialog.style.zIndex = '9999'; loadingDialog.style.fontSize = '14px'; loadingDialog.style.display = 'flex'; loadingDialog.style.alignItems = 'center'; loadingDialog.innerHTML = ` <div class="lds-ring"><div></div><div></div><div></div><div></div></div> <span id="loadingText" style="margin-left: 10px;">カタログを取得しています...</span> `; document.body.appendChild(loadingDialog); (function extractBoard() { const parts = window.location.hostname.split('.'); subdomain = parts[0]; const pathParts = window.location.pathname.split('/'); board = pathParts[1]; })(); function isInsideExcludedElement(el) { let p = el.parentElement; while (p) { if (['table', 'tr', 'td'].includes(p.tagName.toLowerCase())) return true; p = p.parentElement; } return false; } function createNewRow() { const tr = document.createElement('tr'); catalogTable.appendChild(tr); return tr; } let currentRow = catalogTable.lastElementChild || createNewRow(); function addThreadToDOM(thread) { if (existingThreads.has(thread.href)) return; existingThreads.add(thread.href); let titleHtml = ""; if (includeTitle && typeof thread.title === 'string' && thread.title.trim().length > 0) { let t = thread.title.trim(); if (t.length > maxTitleLength) { t = t.substring(0, maxTitleLength) + '...'; } titleHtml = `<br><small>${t}</small>`; } const td = document.createElement('td'); td.innerHTML = ` <a href="${thread.href}" target="_blank"> <img src="${thread.imgSrc}" border="0" width="${thread.width}" height="${thread.height}" alt="" loading="lazy"> </a> ${titleHtml} <br><font size="2">${thread.replyCount}</font><div class="pdmc" data-no="${thread.id}"></div> `; if (currentRow.children.length >= maxColumns) { currentRow = createNewRow(); } currentRow.appendChild(td); } function addThreadToCatalog(thread) { if (existingThreads.has(thread.href)) return; if (sortParam && !['6','7','8','9'].includes(sortParam)) { fetchedThreads.push(thread); return; } addThreadToDOM(thread); } function finalizeSortedThreads() { if (!sortParam || ['6','7','8','9'].includes(sortParam)) return; if (sortParam === '1') { fetchedThreads.sort((a, b) => new Date(b.date) - new Date(a.date)); fetchedThreads.reverse(); } else if (sortParam === '2') { fetchedThreads.sort((a, b) => new Date(a.date) - new Date(b.date)); fetchedThreads.reverse(); } else if (sortParam === '3') { fetchedThreads.sort((a, b) => b.replyCount - a.replyCount); } else if (sortParam === '4') { fetchedThreads.sort((a, b) => a.replyCount - b.replyCount); } fetchedThreads.forEach(thread => addThreadToDOM(thread)); } function gmFetch(url, attempts = 0) { const maxAttempts = 10; return new Promise(resolve => { function attempt() { GM_xmlhttpRequest({ method: 'GET', url: url, onload: resp => resolve(resp), onerror: err => { if (attempts < maxAttempts) { attempts++; console.warn(`Error fetching ${url}, attempt ${attempts}`, err); attempt(); } else { console.error(`Max attempts reached for ${url}. Returning empty`); resolve({ status: 408, responseText: "" }); } } }); } attempt(); }); } function fetchThreadPromise(threadURL, relativeHref) { return new Promise(resolve => { gmFetch(threadURL).then(response => { const parser = new DOMParser(); const doc = parser.parseFromString(response.responseText, "text/html"); const threadIDMatch = relativeHref.match(/res\/(\d+)\.htm/); const threadID = threadIDMatch ? threadIDMatch[1] : ""; let imgElement = null; const imgs = doc.querySelectorAll("img"); imgs.forEach(img => { if (!img.getAttribute("src").includes(`/${board}/thumb/`)) return; if (isInsideExcludedElement(img)) return; if (!imgElement) imgElement = img; }); if (!imgElement) { resolve(); return; } const thumbSrc = imgElement.getAttribute("src"); const catSrc = thumbSrc.replace(`/${board}/thumb/`, `/${board}/cat/`); const thumbWidth = imgElement.getAttribute("width") || "50"; const thumbHeight = imgElement.getAttribute("height") || "50"; const finalSrc = useCatVersion ? catSrc : thumbSrc; function processThread(finalW, finalH) { let blockquotes = doc.querySelectorAll("blockquote"); let threadText = ""; for (const bq of blockquotes) { if (isInsideExcludedElement(bq)) continue; threadText = bq.textContent.trim(); if (threadText) break; } let dateSpans = doc.querySelectorAll("span.cnw"); let threadDate = ""; for (const sp of dateSpans) { if (isInsideExcludedElement(sp)) continue; threadDate = sp.textContent.trim(); if (threadDate) break; } let replyCount = doc.querySelectorAll("table").length; let title = threadText.split("\n")[0] || ""; const threadInfo = { href: threadURL, id: threadID, imgSrc: finalSrc, width: finalW, height: finalH, title: title, date: threadDate, replyCount: replyCount }; addThreadToCatalog(threadInfo); resolve(); } if (useCatVersion) { let timedOut = false; const tId = setTimeout(() => { timedOut = true; console.warn(`Image load timeout for ${catSrc}, fallback size`); processThread(50, 50); }, 500); let tempImg = new Image(); tempImg.onload = () => { if (!timedOut) { clearTimeout(tId); processThread(tempImg.naturalWidth, tempImg.naturalHeight); } }; tempImg.onerror = () => { if (!timedOut) { clearTimeout(tId); processThread(50, 50); } }; tempImg.src = catSrc; } else { processThread(thumbWidth, thumbHeight); } }).catch(() => { resolve(); }); }); } function fetchIndexPagePromise(pageUrl) { console.log("Fetching index page:", pageUrl); return new Promise(resolve => { gmFetch(pageUrl).then(response => { if (response.status === 404 || response.responseText.includes("404 File Not Found")) { console.log(`Index page ${pageUrl} not found, skipping.`); resolve(); return; } const parser = new DOMParser(); const doc = parser.parseFromString(response.responseText, "text/html"); let threadPromises = []; const links = doc.querySelectorAll("a.hsbn"); links.forEach(link => { const relHref = link.getAttribute("href"); console.log("Found thread URL:", relHref, "on page:", pageUrl); const threadURL = `${window.location.protocol}//${subdomain}.2chan.net/${board}/` + relHref; if (!existingThreads.has(relHref)) { existingThreads.add(relHref); threadPromises.push(fetchThreadPromise(threadURL, relHref)); } }); Promise.all(threadPromises).then(resolve).catch(resolve); }).catch(resolve); }); } const baseURL = `${window.location.protocol}//${subdomain}.2chan.net/${board}/`; const indexPages = []; indexPages.push(baseURL + "futaba.htm"); for (let i = 1; i <= 20; i++) { indexPages.push(baseURL + i + ".htm"); } if (['6','7','8','9'].includes(sortParam)) { loadingDialog.style.display = "none"; setTimeout(() => { loadingDialog.style.display = "none"; }, 1500); return; } const indexPromises = indexPages.map(url => fetchIndexPagePromise(url)); Promise.all(indexPromises).then(() => { if (sortParam) { finalizeSortedThreads(); } loadingDialog.innerHTML = ` <div style="font-size: 14px; margin-right: 6px;">✅</div> <div id="loadingText" style="color: #fff; font-weight: bold;">カタログを取得しました!</div> `; loadingDialog.style.background = "green"; loadingDialog.style.opacity = "1"; setTimeout(() => { loadingDialog.style.display = "none"; }, 1500); }); })();
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址