Skip to content

Instantly share code, notes, and snippets.

@xmoforf
Last active November 4, 2025 02:32
Show Gist options
  • Save xmoforf/cfb3caf58ddb07f37b650df2b655739d to your computer and use it in GitHub Desktop.
Save xmoforf/cfb3caf58ddb07f37b650df2b655739d to your computer and use it in GitHub Desktop.
WedgeMarker-alpha.user.js
// ==UserScript==
// @name MAM - WedgeWaster (Alpha Test) (Bookmark Edition)
// @author WirlyWirly
// @namespace https://github.com/xmoforf
// @version 2.8
// @match https://www.myanonamouse.net/
// @match https://www.myanonamouse.net/index.php
// @match https://www.myanonamouse.net/preferences/index.php*
// @match https://www.myanonamouse.net/t/*
// @match https://www.myanonamouse.net/tor/browse.php*
// @icon https://www.myanonamouse.net/favicon.ico
// @homepage https://gist.github.com/xmoforf/cfb3caf58ddb07f37b650df2b655739d
// @updateURL https://gist.github.com/xmoforf/cfb3caf58ddb07f37b650df2b655739d/raw/WedgeMarker-alpha.user.js
// @downloadURL https://gist.github.com/xmoforf/cfb3caf58ddb07f37b650df2b655739d/raw/WedgeMarker-alpha.user.js
// @description Waste a wedge and not your time when bookmarking!
// @grant GM.getValue
// @grant GM.setValue
// @grant GM.deleteValue
// @run-at document-end
// ==/UserScript==
// ----------------------------------- CODE --------------------------------------
(async () => {
// Grabbing action is different based on configuration
const GRAB_ACTIONS = {
bookmark: (id, url) => {
// eslint-disable-next-line no-undef
bookmarkClick("add", id); // bookmark
},
download: (id, url) => {
window.location = url; // download
},
};
// Base64 Encoded Icons
const IMG_BUTTON_TO_CLICK = ``;
const IMG_ICON_TO_CLICK_MAP = {
bookmark: ``,
download: ``,
};
// User-configurable actions
const PURCHASE_FAIL_ACTION = await getFreeleechFailAction();
const WASTE_ACTION = await getWasteAction();
const grabTorrent = GRAB_ACTIONS[WASTE_ACTION] ?? ((id, url) => {});
const IMG_ICON_TO_CLICK = IMG_ICON_TO_CLICK_MAP[WASTE_ACTION];
const IMG_ICON_SUCCESS = ``;
async function clearSettings() {
await GM.deleteValue("wasteAction");
await GM.deleteValue("freeleechFailAction");
window.location.reload();
}
function promptDropdown(label, options, defaultValue) {
return new Promise((resolve) => {
// Create modal elements
const modal = document.createElement("div");
modal.style = `
position: fixed; left: 0; top: 0; width: 100vw; height: 100vh;
background: rgba(0,0,0,0.5); display: flex; align-items: center; justify-content: center; z-index: 10000;
`;
const box = document.createElement("div");
box.style = `padding: 16px; border-radius: 6px; min-width: 220px; border: 1px solid white`;
box.innerHTML = `<label>${label}</label><br><br>`;
box.style.backgroundColor = getComputedStyle(
document.body,
).backgroundColor;
const select = document.createElement("select");
for (const opt of options) {
let option = document.createElement("option");
option.value = opt;
option.textContent = opt;
if (opt === defaultValue) option.selected = true;
select.appendChild(option);
}
const okButton = document.createElement("button");
okButton.textContent = "OK";
okButton.style = "margin-left:10px;";
box.appendChild(select);
box.appendChild(okButton);
modal.appendChild(box);
document.body.appendChild(modal);
okButton.onclick = () => {
resolve(select.value);
document.body.removeChild(modal);
};
});
}
async function getWasteAction() {
const storageKey = "wasteAction";
const grabChoices = Object.keys(GRAB_ACTIONS);
let choice = await GM.getValue(storageKey, null);
if (grabChoices.includes(choice)) {
return choice;
} else {
choice = await promptDropdown(
"What action do you want to take when Wedge Wasting?",
grabChoices,
grabChoices[0],
);
await GM.setValue(storageKey, choice);
return choice;
}
}
async function getFreeleechFailAction() {
const storageKey = "freeleechFailAction";
let choice = await GM.getValue(storageKey, null);
if (choice === null) {
let rawChoice = await promptDropdown(
"Do you want to take the action if the freeleech purchase fails?",
["Yes", "No"],
["Yes"],
);
choice = { Yes: "act", No: "dontAct" }[rawChoice] ?? null;
await GM.setValue(storageKey, choice);
}
return choice;
}
// Returns whether we are on the details page
function onDetailsPage() {
return window.location.pathname.match(/\/t\/\d+/);
}
// Returns whether we are on the home page
function onHomePage() {
return (
window.location.pathname === "/" ||
window.location.pathname === "/index.php"
);
}
// Returns whether we're on the browse page
function onBrowsePage() {
return window.location.pathname.match(/\/tor\/browse\.php/);
}
function onPrefsPage() {
return window.location.pathname.match(/preferences\/index\.php.*/);
}
// Create the large WedgeWaster button
function makeWWButton() {
let button = document.createElement("div");
button.style = "margin-top: 1em";
button.innerHTML = `<img style="max-width: 130px" src="${IMG_BUTTON_TO_CLICK}">`;
return button;
}
// Create the small WedgeWaster icon
function makeWWIcon(source) {
const computedStyle = window.getComputedStyle(source);
const copiedProps = [
"display",
"padding",
"width",
"height",
"user-select",
"cursor",
];
let button = document.createElement("a");
copiedProps.forEach((prop) => {
button.style[prop] = computedStyle.getPropertyValue(prop);
});
button.style.backgroundImage = `url(${IMG_ICON_TO_CLICK})`;
button.style.backgroundRepeat = "no-repeat";
return button;
}
// When the purchase button is clicked
function purchaseTorrent(torrentID, torrentFileURL, pageType) {
let freeleechURL = `https://www.myanonamouse.net/json/bonusBuy.php/?spendtype=personalFL&torrentid=${torrentID}`;
function purchaseSuccessAction(message) {
// Update Ratio box
if (pageType === "details") {
let ratioBox = document.getElementById("ratio");
ratioBox.getElementsByClassName(
"torDetInnerBottomSpan",
)[0].innerText = "This torrent is a Personal freeleech!";
}
grabTorrent(torrentID, torrentFileURL);
console.log(`WedgeMarker | Success: ${message}`);
}
const PURCHASE_ACTIONS = {
act: (message) => {
console.log(`WedgeMarker | Warn: ${message}`);
grabTorrent(torrentID, torrentFileURL);
},
dontAct: (message) => {
console.log(`WedgeMarker | Error: ${message}`);
alert(
`WedgeMarker\n\nError: Torrent Not Grabbed.\n\n${message}`,
);
},
};
const purchaseFailAction =
PURCHASE_ACTIONS[PURCHASE_FAIL_ACTION] ?? ((id, url) => {});
var xhr = new XMLHttpRequest();
xhr.open("GET", freeleechURL, true);
xhr.onreadystatechange = () => {
if (this.readyState == 4 && this.status == 200) {
let mamResponse = JSON.parse(this.response);
let successfulPurchase = mamResponse.success;
successfulPurchase
? purchaseSuccessAction(mamResponse.type)
: purchaseFailAction(mamResponse.error);
}
};
xhr.send();
}
// (Browe page) Look for changes and add buttons
function findThenAddButtons(mutations) {
// Function to Process each row that is a torrent
function processTorrentRow(torrentRow) {
function alreadyFreeleech() {
let vipFreeleech =
torrentRow.children[1].querySelector('img[alt="VIP"]');
let siteFreeleech = torrentRow.children[1].querySelector(
'img[alt="freeleech"]',
);
let personalFreeleech = torrentRow.children[1].querySelector(
'span[title="personal freeleech"]',
);
return vipFreeleech || siteFreeleech || personalFreeleech;
}
// Get torrent ID and File URL from the default download button
let downloadButton =
torrentRow.getElementsByClassName("directDownload")[0];
let titleLink = torrentRow.getElementsByClassName("torTitle");
// let torrentID = downloadButton.id.match(/dlLink(\d+)/)[1]; // other way doesnt work on homepage
let torrentID = titleLink[0].href.match(/\/t\/(.*)/)[1];
let torrentFileURL = downloadButton.href;
// Add the WedgeMarker button above/before the normal download button
let wedgeWasterButton = makeWWIcon(downloadButton);
wedgeWasterButton.addEventListener("click", () => {
if (alreadyFreeleech()) {
console.log(
"WedgeMarker | Info: Save a wedge, this torrent is already freeleech!",
);
grabTorrent(torrentID, torrentFileURL);
wedgeWasterButton.src = IMG_ICON_SUCCESS;
} else {
purchaseTorrent(torrentID, torrentFileURL, "browse");
wedgeWasterButton.src = IMG_ICON_SUCCESS;
}
});
downloadButton.parentNode.insertBefore(
wedgeWasterButton,
downloadButton,
);
}
try {
// For all torrent elements in the table
let torrentRows = document
.getElementsByTagName("tbody")[1]
.getElementsByTagName("tr");
for (let torrentRow of torrentRows) {
let isActualTorrentRow =
torrentRow.getElementsByClassName("directDownload")
.length == 1;
if (isActualTorrentRow) {
processTorrentRow(torrentRow);
}
}
} catch (e) {
if (e instanceof TypeError) {
// No changes or not ready
} else {
console.log(`WedgeMarker | Error: Unexpected Error\n\n${e}`);
}
}
}
//
// When page loads
//
console.log(
`WedgeMarker | Info: PURCHASE_FAIL_ACTION = ${PURCHASE_FAIL_ACTION}`,
);
console.log(`WedgeMarker | Info: WASTE_ACTION = ${WASTE_ACTION}`);
var searchResultsUpdated = new MutationObserver(findThenAddButtons);
if (onDetailsPage()) {
let torrentFileURL = document.getElementById("tddl").href;
let torrentID = document.URL.match(/\/t\/(\d+)/)[1];
let pageType = "details";
let downloadButtons = document
.getElementById("download")
.getElementsByClassName("torDetInnerBottom")[0];
// Add WedgeMarker button
let button = makeWWButton();
button.addEventListener("click", () => {
let freeLeechStatusText = document
.getElementById("ratio")
.getElementsByClassName("torDetInnerBottomSpan")[0].innerText;
let freeLeechStatus = freeLeechStatusText.match(/freeleech/i);
if (freeLeechStatus) {
console.log(
"WedgeMarker | Info: Save a wedge, this torrent is already freeleech!",
);
grabTorrent(torrentID);
} else {
purchaseTorrent(torrentID, torrentFileURL);
}
});
// ...below normal download button
downloadButtons.appendChild(button);
//
//
} else if (onBrowsePage() || onHomePage()) {
let target = document.getElementById("ssr");
let config = { childList: true };
// Monitor page for any changes
searchResultsUpdated.observe(target, config);
//
//
} else if (onPrefsPage()) {
function getSettingsMenu() {
return document
.getElementById("mainBody")
.getElementsByTagName("tr")[0];
}
function makeWWResetSettingsButton(menu) {
let newButton = [
menu.childNodes[2].cloneNode(true), // spacer
menu.childNodes[3].cloneNode(true), // anchor tag
];
let anchor = newButton[1].children[0];
anchor.innerHTML = "WedgeWaster<br/>Reset Settings";
anchor.href = "#";
anchor.onclick = (event) => {
event.preventDefault();
clearSettings();
};
return newButton;
}
function insertWWResetSettingsButton(menu, newSettingsButton) {
newSettingsButton.forEach((el) => {
menu.insertBefore(el, menu.children[menu.children.length - 1]);
});
}
let menu = getSettingsMenu();
let newButton = makeWWResetSettingsButton(menu);
insertWWResetSettingsButton(menu, newButton);
}
})();
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment