您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
This automatically selects submission notifications from unwatched users on the current FurAffinity notifications page.
// ==UserScript== // @name Auto-select unwatched // @namespace https://github.com/f1r3w4rr10r/fa-utils // @version 1.0.1 // @description This automatically selects submission notifications from unwatched users on the current FurAffinity notifications page. // @author f1r3w4rr10r // @match https://www.furaffinity.net/msg/submissions/* // @icon  // @license MIT // @grant none // ==/UserScript== (async function () { "use strict"; /** * @param {string} url * @returns {Promise<[string[], string | null]>} */ async function getWatchedFromPage(url) { const result = await fetch(url); if (!result.ok) { console.error("Could not get watchlist.", url, result); throw new Error("Could not get watchlist."); } const doc = new DOMParser().parseFromString( await result.text(), "text/html", ); const userLinks = Array.from(doc.querySelectorAll("a")).map((a) => a.href); const nextForm = doc.querySelector(".floatright form"); if (!(nextForm instanceof HTMLFormElement)) throw new Error(`Could not find the next page form on: ${url}`); const nextUrl = nextForm.action; return [userLinks, nextUrl === url ? null : nextUrl]; } /** * @returns {Promise<string[]>} */ async function getWatched() { const watchList = []; const userAnchor = document.querySelector("article.mobile-menu a"); if (!(userAnchor instanceof HTMLAnchorElement)) throw new Error("Could not get the user anchor."); const urlMatch = userAnchor.href.match(/user\/(.+?)$/); if (!urlMatch) throw new Error("The user profile URL did not match."); const userName = urlMatch[1]; if (!userName) throw new Error("Could not extract a user name."); /** @type {string | null} */ let nextUrl = `https://www.furaffinity.net/watchlist/by/${userName}/`; while (nextUrl !== null) { let pageWatchList = []; [pageWatchList, nextUrl] = await getWatchedFromPage(nextUrl); watchList.push(...pageWatchList); } return watchList; } /** * @param {string[]} watched * @returns {Promise<number>} */ async function iterateLabels(watched) { const figures = Array.from( document.querySelectorAll("section.gallery figure"), ); let selected = 0; for (const figure of figures) { const checkbox = figure.querySelector("input"); if (!(checkbox instanceof HTMLInputElement)) throw new Error("Could not find a checkbox."); const userAnchor = figure.querySelector( "figcaption label p:last-child a", ); if (!(userAnchor instanceof HTMLAnchorElement)) throw new Error("Could not find a user anchor."); const userLink = userAnchor.href; if (watched.includes(userLink)) continue; figure.classList.add("unwatched"); checkbox.checked = true; selected += 1; } return selected; } const sheet = new CSSStyleSheet(); sheet.replaceSync("figure.unwatched { outline: red 3px solid; }"); document.adoptedStyleSheets.push(sheet); const sectionHeader = document.querySelector(".section-header"); if (!sectionHeader) throw new Error("Could not find the section header."); const unwatchedSelectMessage = document.createElement("p"); unwatchedSelectMessage.textContent = "Checking for submissions from unwatched users…"; sectionHeader.appendChild(unwatchedSelectMessage); const watched = await getWatched(); const selected = await iterateLabels(watched); const message = `Selected ${selected} unwatched submissions.`; unwatchedSelectMessage.textContent = message; })();
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址