Last active
November 4, 2025 03:13
-
-
Save xmoforf/226ec18bbff8b639d289841c96e0b6ee to your computer and use it in GitHub Desktop.
Book Stuff?
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| // ==UserScript== | |
| // @name MAM - WedgeWaster (Bookmark Edition) | |
| // @author WirlyWirly | |
| // @namespace https://github.com/xmoforf | |
| // @version 2.9.2 | |
| // @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/226ec18bbff8b639d289841c96e0b6ee | |
| // @updateURL https://gist.github.com/xmoforf/226ec18bbff8b639d289841c96e0b6ee/raw/WedgeMarker.user.js | |
| // @downloadURL https://gist.github.com/xmoforf/226ec18bbff8b639d289841c96e0b6ee/raw/WedgeMarker.user.js | |
| // @description Waste a wedge and not your time when bookmarking! | |
| // @grant GM.getValue | |
| // @grant GM.setValue | |
| // @run-at document-end | |
| // ==/UserScript== | |
| // ----------------------------------- CODE -------------------------------------- | |
| (async () => { | |
| // needed to initialize memory in the correct order | |
| const GRAB_ACTIONS = generateGrabActions(); | |
| ///////////////////////////////////////////////////////////////////////////////////////////// | |
| // USER SETTINGS | |
| ///////////////////////////////////////////////////////////////////////////////////////////// | |
| // ICONS | |
| const IMG_BUTTON_TO_CLICK = ``; | |
| const IMG_ICON_SUCCESS = ``; | |
| const IMG_ICON_TO_CLICK_MAP = { | |
| bookmark: ``, | |
| download: ``, | |
| }; | |
| ///////////////////////////////////////////////////////////////////////////////////////////// | |
| // SETTINGS | |
| async function clearSettings() { | |
| await GM.deleteValue("wasteAction"); | |
| await GM.deleteValue("freeleechFailAction"); | |
| window.location.reload(); | |
| } | |
| // Basic user interface window to select settings | |
| 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); | |
| }; | |
| }); | |
| } | |
| ///////////////////////////////////////////////////////////////////////////////////////////// | |
| // GRABBING | |
| async function getGrabAction() { | |
| 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; | |
| } | |
| } | |
| function generateGrabActions() { | |
| return { | |
| bookmark: (id, url) => { | |
| // eslint-disable-next-line no-undef | |
| bookmarkClick("add", id); // bookmark | |
| }, | |
| download: (id, url) => { | |
| window.location = url; // download | |
| }, | |
| }; | |
| } | |
| const GRAB_ACTION = await getGrabAction(); | |
| const grabTorrent = GRAB_ACTIONS[GRAB_ACTION] ?? ((id, url) => {}); | |
| const IMG_ICON_TO_CLICK = IMG_ICON_TO_CLICK_MAP[GRAB_ACTION]; | |
| ///////////////////////////////////////////////////////////////////////////////////////////// | |
| // PURCHASE FAILURE | |
| async function getPurchaseFailAction() { | |
| const storageKey = "freeleechFailAction"; | |
| let choice = await GM.getValue(storageKey, null); | |
| if (choice === null) { | |
| let rawChoice = await promptDropdown( | |
| "Do you want to grab the torrent if the freeleech purchase fails?", | |
| ["Yes", "No"], | |
| ["Yes"], | |
| ); | |
| choice = { Yes: "act", No: "dontAct" }[rawChoice] ?? null; | |
| await GM.setValue(storageKey, choice); | |
| } | |
| return choice; | |
| } | |
| const PURCHASE_FAIL_ACTION = await getPurchaseFailAction(); | |
| ///////////////////////////////////////////////////////////////////////////////////////////// | |
| // PAGE DETECTION | |
| function onDetailsPage() { | |
| return window.location.pathname.match(/\/t\/\d+/); | |
| } | |
| function onHomePage() { | |
| return ( | |
| window.location.pathname === "/" || | |
| window.location.pathname === "/index.php" | |
| ); | |
| } | |
| function onBrowsePage() { | |
| return window.location.pathname.match(/\/tor\/browse\.php/); | |
| } | |
| function onPrefsPage() { | |
| return window.location.pathname.match(/preferences\/index\.php.*/); | |
| } | |
| ///////////////////////////////////////////////////////////////////////////////////////////// | |
| // UI ELEMENT GENERATION | |
| 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; | |
| } | |
| ///////////////////////////////////////////////////////////////////////////////////////////// | |
| 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(); | |
| } | |
| ///////////////////////////////////////////////////////////////////////////////////////////// | |
| // (Browse 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}`); | |
| } | |
| } | |
| } | |
| ///////////////////////////////////////////////////////////////////////////////////////////// | |
| ///////////////////////////////////////////////////////////////////////////////////////////// | |
| ///////////////////////////////////////////////////////////////////////////////////////////// | |
| // PAGE LOAD | |
| console.log( | |
| `WedgeMarker | Info: PURCHASE_FAIL_ACTION = ${PURCHASE_FAIL_ACTION}`, | |
| ); | |
| console.log(`WedgeMarker | Info: GRAB_ACTION = ${GRAB_ACTION}`); | |
| // Customize the different pages | |
| 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