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.1.0
// @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 {URL} url
* @returns {Promise<[string[], URL | 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(
/** @type {HTMLAnchorElement[]} */ (
Array.from(doc.querySelectorAll(".watch-list 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 pageInput = /** @type {HTMLInputElement | null} */ (
nextForm.querySelector('input[name="page"]')
);
const nextButton = nextForm.querySelector("button");
const nextUrl = new URL(nextForm.action);
if (pageInput) nextUrl.searchParams.set("page", pageInput.value);
return [userLinks, nextButton?.disabled ? 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 {URL | null} */
let nextUrl = new URL(
`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;
})();