MyAnonamouse Requests Page ➜ Audible + Amazon Kindle Search Links.

Adds Audible and Amazon search links to the requests page on myanonamouse. The links added are Audible for Audiobooks and Amazon for E-Books.

// ==UserScript==
// @name         MyAnonamouse Requests Page ➜ Audible + Amazon Kindle Search Links.
// @namespace    https://www.myanonamouse.net/u/253587
// @version      1.1
// @description  Adds Audible and Amazon search links to the requests page on myanonamouse. The links added are Audible for Audiobooks and Amazon for E-Books.
// @match        https://www.myanonamouse.net/tor/requests*
// @grant        none
// @author       Gorgonian
// ==/UserScript==

(function () {
    'use strict';

    const audiobookCategories = new Set([
        39, 49, 50, 83, 51, 97, 40, 41, 106, 42, 52, 98, 54, 55, 43, 99, 84, 44,
        56, 45, 57, 85, 87, 119, 88, 58, 59, 46, 47, 53, 89, 100, 108, 48, 111,
    ]);

    const bookCategories = new Set([
        60, 71, 72, 90, 61, 73, 101, 62, 63, 107, 64, 74, 102, 76, 77, 65, 103, 115,
        91, 66, 78, 67, 79, 80, 92, 118, 94, 120, 95, 81, 82, 68, 69, 75, 96, 104, 109, 70, 112,
    ]);

    const base64AudibleIcon = ''; // your Audible icon base64 here
    const base64AmazonIcon = ''; // your Amazon icon base64 here

    const processedRows = new WeakSet();

    function addLinkAfterLastAuthor(row, href, title, base64Icon, altText) {
        const authorLinks = row.querySelectorAll('a.author');
        if (authorLinks.length === 0) return;

        const searchLink = document.createElement('a');
        searchLink.href = href;
        searchLink.target = '_blank';
        searchLink.title = title;
        searchLink.style.cssText = `
      display: inline-block;
      margin-top: 4px;
      margin-right: 6px;
      float: right;
    `;
        searchLink.innerHTML = `<img src="${base64Icon}" alt="${altText}" style="width:14px;height:14px;">`;

        const lastAuthor = authorLinks[authorLinks.length - 1];
        lastAuthor.insertAdjacentElement('afterend', searchLink);
    }

    function processRow(row) {
        if (processedRows.has(row)) return;
        processedRows.add(row);

        const catDiv = row.querySelector('a.catLink > div');
        if (!catDiv || !catDiv.className.startsWith('cat')) return;

        const catNum = parseInt(catDiv.className.replace('cat', ''), 10);

        const titleLink = row.querySelector('a.torTitle');
        const authorLinks = row.querySelectorAll('a.author');
        if (!titleLink || authorLinks.length === 0) return;

        const title = titleLink.textContent.trim();
        const firstAuthor = authorLinks[0].textContent.trim();

        if (audiobookCategories.has(catNum)) {
            const audibleSearchQuery = encodeURIComponent(`${firstAuthor} ${title}`);
            const audibleHref = `https://www.audible.com/search?keywords=${audibleSearchQuery}`;
            addLinkAfterLastAuthor(row, audibleHref, 'Search Audible', base64AudibleIcon, 'Audible');
        }

        if (bookCategories.has(catNum)) {
            const amazonSearchQuery = encodeURIComponent(`${firstAuthor} ${title}`);
            const amazonHref = `https://www.amazon.com/s?k=${amazonSearchQuery}&i=stripbooks&rh=n%3A283155%2Cp_n_feature_browse-bin%3A618073011`;
            addLinkAfterLastAuthor(row, amazonHref, 'Search Amazon', base64AmazonIcon, 'Amazon');
        }
    }

    function processAllRows() {
        document.querySelectorAll('.torRow').forEach(processRow);
    }

    function waitForTorRows() {
        const rows = document.querySelectorAll('.torRow');
        if (rows.length === 0) {
            requestAnimationFrame(waitForTorRows);
        } else {
            processAllRows();

            const observer = new MutationObserver(mutations => {
                for (const mutation of mutations) {
                    mutation.addedNodes.forEach(node => {
                        if (!(node instanceof HTMLElement)) return;
                        if (node.classList.contains('torRow')) {
                            processRow(node);
                        } else {
                            node.querySelectorAll?.('.torRow')?.forEach(processRow);
                        }
                    });
                }
            });

            observer.observe(document.body, { childList: true, subtree: true });
        }
    }

    waitForTorRows();
})();