// ==UserScript== // @name Zibzab's GameDox/Rom Upload Helper // @namespace http://tampermonkey.net/ // @version 1.7 // @description try to take over the world :) // @author BestGrapeLeaves // @match https://gazellegames.net/upload.php?groupid=* // @match https://gazellegames.net/torrents.php?id=* // @icon https://i.imgur.com/UFOk0Iu.png // @grant GM_xmlhttpRequest // @grant GM_listValues // @grant GM_deleteValue // @grant GM_setValue // @grant GM_getValue // @connect datomatic.no-intro.org // @license MIT // @downloadURL none // ==/UserScript== // cache class based on GM_setValue, GM_getValue, GM_deleteValue, GM_listValues global functions. With expiration. class GMCache { constructor(name) { this.cache = {}; this.name = name; } getKeyName(key) { return `zibzabhelper.cache${this.name}.${key}`; } get(key, fallback) { const whenNotFound = () => typeof fallback === "function" ? fallback() : fallback; const res = GM_getValue(this.getKeyName(key)); if (res === undefined) { return whenNotFound(); } const { value, expires } = res; if (expires && expires < Date.now()) { this.delete(key); return whenNotFound(); } return value; } set(key, value, ttl) { const expires = Date.now() + ttl; GM_setValue(this.getKeyName(key), { value, expires }); } delete(key) { GM_deleteValue(this.getKeyName(key)); } cleanUp() { const keys = GM_listValues(); keys.forEach((key) => { if (key.startsWith(this.getKeyName(""))) { const { expires } = GM_getValue(key); if (expires < Date.now()) { GM_deleteValue(key); } } }); } } // Code is a spaghetti mess, don't read it. Do something else with your time. (function () { "use strict"; const noIntroCache = new GMCache("no-intro"); const PARENS_TAGS_REGEX = /\(.*?\)/g; const NO_INTRO_TAGS_REGEX = /\((Unl|Proto|Sample|Aftermarket|Homebrew)\)|\(Rev \d+\)|\(v[\d\.]+\)|\(Beta(?: \d+)?\)/; const GAME_DOX_INSERT = `[align=center] pdf pages [/align] `; const genRomInsert = ( url = "xxx", filename = "xxx" ) => `[align=center]${filename} matches [url=${url}]No-Intro checksum[/url] Compressed with [url=https://sourceforge.net/projects/trrntzip/]torrentzip.[/url][/align] `; const regionToLanguage = { USA: "English", Europe: "English", Japan: "Japanese", World: "English", "USA, Europe": "English", Other: "English", Korea: "Korean", Taiwan: "Chinese", }; const twoLetterLanguageCodeToGGn = { en: "English", de: "German", fr: "French", cz: "Czech", zh: "Chinese", it: "Italian", ja: "Japanese", ko: "Korean", pl: "Polish", pt: "Portuguese", ru: "Russian", es: "Spanish", }; const parseLanguage = (region, possiblyLanguages) => { if (possiblyLanguages === undefined) { return regionToLanguage[region] || "Other"; } const twoLetterCodes = possiblyLanguages .split(",") .map((l) => l.trim().toLowerCase()); const isLanguages = twoLetterCodes.every((l) => l.length === 2); if (!isLanguages || twoLetterCodes.length === 0) { return regionToLanguage[region] || "Other"; } if (twoLetterCodes.length > 1) { return "Multi-Language"; } return twoLetterLanguageCodeToGGn[twoLetterCodes[0]] || "Other"; }; function noIntroLinkForTorrentId(torrentId) { const links = $(`#torrent_${torrentId} #description a`); return links .map(function () { return $(this).attr("href"); }) .get() .map((link) => { const url = new URL(link); url.protocol = "https:"; // Rarely descriptions have the http protocol return url.toString(); }) .find(link => link.startsWith("https://datomatic.no-intro.org/")) } function fetchNoIntro(url) { return new Promise((resolve, reject) => { const cached = noIntroCache.get(url); if (cached) { console.log("Using cached no-intro data", url, cached); resolve({ ...cached, cached: true }); return; } GM_xmlhttpRequest({ method: "GET", url, timeout: 5000, onload: ({ responseText }) => { try { const parser = new DOMParser(); const scraped = parser.parseFromString(responseText, "text/html"); // HTML is great const dumpsTitle = [ ...scraped.querySelectorAll("td.TableTitle"), ].find((td) => td.innerText.trim() === "Dump(s)"); if (!dumpsTitle) { window.GMPARSER = scraped; console.err('zibzab dumps title not found, set parser as global: GMPARSER', responseText); new Error("No dump's title found"); } const filename = dumpsTitle.parentElement.parentElement.parentElement.nextElementSibling .querySelector( "table > tbody > tr:nth-child(2) > td:last-child" ) .innerText.trim(); const title = scraped .querySelector("tr.romname_section > td") .innerText.trim(); const parenMatches = title .match(/\(.+?\)/g) .map((p) => p.slice(1, -1)); const [region, possiblyLanguages] = parenMatches; const matchedGGnRegion = [ "USA", "Europe", "Japan", "Asia", "Australia", "France", "Germany", "Spain", "Italy", "UK", "Netherlands", "Sweden", "Russia", "China", "Korea", "Hong Kong", "Taiwan", "Brazil", "Canada", "Japan, USA", "Japan, Europe", "USA, Europe", "Europe, Australia", "Japan, Asia", "UK, Australia", "World", "Region-Free", "Other", ].find((r) => r === region) || "Other"; const matchedGGnLanguage = parseLanguage( matchedGGnRegion, possiblyLanguages ); const res = { filename, matchedGGnRegion, matchedGGnLanguage }; // One hour seems appropriate noIntroCache.set(url, res, 1000 * 60 * 60); resolve({ ...res, cached: false }); } catch (err) { console.error("zibzab helper failed to parse no-intro:", err); reject( new Error( "Failed to parse no-intro :/\nPlease report to BestGrapeLeaves,\nthe error was logged to the browser console" ) ); } }, ontimeout: () => { reject(new Error("Request to no-intro timed out after 5 seconds")); }, }); }); } // We are fetching files for checking, might as well reduce load and save to dom function fetchTorrentFilesWithoutShowing(torrentId) { const fromDOM = () => $( `#files_${torrentId} > table > tbody > tr:not(.colhead_dark) > td:first-child` ) .map(function () { return $(this).text(); }) .get(); return new Promise((resolve) => { if ($("#files_" + torrentId).raw().innerHTML === "") { // $('#files_' + torrentId).gshow().raw().innerHTML = '
${text}`;
insert(
`The filename in the torrent is: ${pre(
res.actualFilename,
"lightcoral"
)} but the desired filename, based on No-Intro is: ${pre(
res.expectedFilename,
"lightgreen"
)}`
);
prevCached = res.cached;
}
if (trumps === 0) {
checkForTrumpsButton.val("No Trumps Found");
} else if (trumps === 1) {
checkForTrumpsButton.val("1 Trump Found");
} else {
checkForTrumpsButton.val(`${trumps} Trumps Found`);
}
}
function insertTrumpButtonAndMaybeCheck() {
const torrents = getNoIntroTorrentsOnPage();
if (torrents.length === 0) {
return;
}
const checkForTrumpsButton = $(
``
);
$(".torrent_table > tbody > tr:first-child > td:first-child")
.first()
.append(checkForTrumpsButton);
if (torrents.length <= 4) {
insertTrumpSuggestions(torrents);
}
checkForTrumpsButton.click((e) => {
e.stopImmediatePropagation();
insertTrumpSuggestions(torrents);
});
}
// No Intro Button
function makeNoIntro(filename) {
const tags = filename
? filename
.match(PARENS_TAGS_REGEX)
.filter((p) => NO_INTRO_TAGS_REGEX.test(p))
.join(" ")
: "";
// Release type = ROM
$("select#miscellaneous").val("ROM").change();
// It is a special edition
if (!$("input#remaster").prop("checked")) {
$("input#remaster").prop("checked", true);
Remaster();
}
// Not a scene release
$("#ripsrc_home").prop("checked", true);
// Update title
updateReleaseTitle($("#title").raw().value + " " + tags);
// Get url params
const params = new URLSearchParams(window.location.search);
// Set correct edition (fallback to guessing)
const setEdition = (edition) => {
try {
$("#groupremasters").val(edition).change();
GroupRemaster();
} catch {
// group remaster always throws (regardless of the userscript)
}
};
const editionInfo = params.get("edition");
$("#groupremasters > option").each(function () {
const title = $(this).text().toLowerCase();
console.log("checking", title);
if (editionInfo && title === editionInfo.toLowerCase()) {
setEdition($(this).val());
return false; // This breaks out of the jquery loop
} else {
if (title.includes("no-intro") || title.includes("nointro")) {
setEdition($(this).val());
}
}
});
// Trigger no-intro link scraper
const noIntroLink = params.get("no-intro");
if (noIntroLink) {
$("#no-intro-url-input").val(noIntroLink).change();
}
}
function noIntroUI() {
// elements
const noIntroContainer = $(
`Loading...
' ).hide(); // structure const td = $("