Last active
November 4, 2025 02:32
-
-
Save xmoforf/cfb3caf58ddb07f37b650df2b655739d to your computer and use it in GitHub Desktop.
WedgeMarker-alpha.user.js
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 (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