Audible Buttons ➜ MyAnonamouse Search

Replace All orange buttons with MyAnonamouse search.

// ==UserScript==
// @name         Audible Buttons ➜ MyAnonamouse Search
// @namespace    https://www.myanonamouse.net/u/253587
// @version      1.7
// @description  Replace All orange buttons with MyAnonamouse search.
// @include      https://www.audible.*
// @grant        none
// @run-at       document-idle
// @author       Gorgonian
// ==/UserScript==

(function () {
    'use strict';

    const icons = {
        icon1: '',
    };

    const cleanAuthor = (name = '') =>
    name.replace(/^(Sir|Dame|Dr\.?|Prof\.?|Mr\.?|Mrs\.?|Ms\.?)\s+/i, '')
    .replace(/,?\s*(Jr\.?|Sr\.?|III|IV|Ph\.?D\.?|M\.?D\.?|DPhil|EdD|Esq\.?|OBE|MBE|CBE|by)$/i, '')
    .trim();

    const cleanTitle = (title = '') =>
    title.replace(/\b(the complete|box\s*set|boxset|books|series|volume|audiobook|editor|by|1st\s*Edition)\b/gi, '')
    .replace(/[^\w\s]/g, '')
    .replace(/\s+/g, ' ')
    .trim();

    function getSearchUrl(author = '', title = '') {
        const query = `${title} ${author}`.trim();
        return `https://www.myanonamouse.net/tor/browse.php?tor[text]=${encodeURIComponent(query)}&tor[srchIn][title]=true&tor[srchIn][tags]=true&tor[srchIn][author]=true&tor[srchIn][narrator]=true&tor[srchIn][series]=true&tor[srchIn][fileTypes]=true&tor[srchIn][filenames]=true&tor[searchType]=all&tor[searchIn]=torrents&tor[cat][]=0&tor[main_cat]=13`;
    }

    function getTitleAndAuthor() {
        const ldJsonScripts = document.querySelectorAll('script[type="application/ld+json"]');
        for (const script of ldJsonScripts) {
            try {
                const json = JSON.parse(script.textContent);
                const items = Array.isArray(json) ? json : [json];
                for (const item of items) {
                    if (item['@type'] === 'Audiobook') {
                        return {
                            title: item.name,
                            author: cleanAuthor(item.author?.[0]?.name || ''),
                        };
                    }
                }
            } catch (_) {}
        }
        return null;
    }

    const patchAdblButton = () => {
        const buttonSelectors = ['adbl-button', 'span.bc-button-primary'];
        const buttons = document.querySelectorAll(buttonSelectors.join(', '));
        buttons.forEach(buttonEl => {
            const text = buttonEl.textContent?.trim();
            if (!text || !/try|credit|confirm|play/i.test(text)) return;
            if (buttonEl.dataset.mamPatched) return;
            buttonEl.dataset.mamPatched = 'true';
            let title = document.title.replace(/[-|–] Audible.*$/, '').trim();
            const ldData = document.querySelector('script[type="application/ld+json"]');
            if (ldData) {
                try {
                    const json = JSON.parse(ldData.textContent);
                    if (json && json['@type'] === 'Audiobook' && json.name) {
                        title = json.name;
                    }
                } catch (_) {}
            }
            let author = '';
            const authorLabel = document.querySelector('.authorLabel');
            if (authorLabel) {
                const authorLink = authorLabel.querySelector('a');
                if (authorLink) {
                    author = authorLink.textContent.trim();
                }
            }
            author = cleanAuthor(author);
            title = cleanTitle(title);
            const searchUrl = getSearchUrl(author, title);
            if (/play/i.test(text)) {
                if (buttonEl.dataset.mamPlayHandled) return;
                buttonEl.dataset.mamPlayHandled = 'true';
                const MAMlink = document.createElement('adbl-button');
                MAMlink.href = searchUrl;
                MAMlink.target = '_blank';
                MAMlink.variant = 'primary';
                MAMlink.style.display = 'flex';
                MAMlink.style.alignItems = 'center';
                MAMlink.style.justifyContent = 'center';
                MAMlink.style.fontWeight = 'bold';
                MAMlink.addEventListener('click', (e) => {
                    e.preventDefault();
                    e.stopImmediatePropagation();
                    window.open(searchUrl, '_blank');
                });
                const icon = document.createElement('img');
                icon.src = icons.icon1;
                icon.alt = '';
                icon.style.cssText = 'margin-right: 6px; height: 24px; width: auto; vertical-align: middle;';
                const textNode = document.createTextNode('Search MyAnonamouse');
                MAMlink.appendChild(icon);
                MAMlink.appendChild(textNode);
                buttonEl.insertAdjacentElement('afterend', MAMlink);
                return;
            }
            const MAMlink = document.createElement('adbl-button');
            MAMlink.href = searchUrl;
            MAMlink.target = '_blank';
            MAMlink.className = buttonEl.className;
            MAMlink.variant = 'primary';
            const icon = document.createElement('img');
            icon.src = icons.icon1;
            icon.alt = '';
            icon.style.cssText = 'margin-right: 8px; height: 24px; width: auto; vertical-align: middle;';
            const spanText = document.createElement('span');
            spanText.textContent = 'Search MyAnonamouse';
            spanText.style.cssText = 'vertical-align: middle;';
            MAMlink.appendChild(icon);
            MAMlink.appendChild(spanText);
            buttonEl.replaceWith(MAMlink);
        });
    };

    function updateListLinks() {
        if (/\/(pd|ac)\//.test(location.pathname)) return;
        const buttons = document.querySelectorAll('a.bc-button-text[href*="/pd/"], a.bc-button-text[href*="/ac/"], button.bc-button-text[title*=" to cart"]');
        buttons.forEach(button => {
            if (button.dataset.mamPatched) return;
            button.dataset.mamPatched = 'true';
            let titleMatch = button.title?.match(/Add (.*?) to cart/);
            let rawTitle = '';
            if (titleMatch) {
                rawTitle = titleMatch[1];
            } else {
                const hrefMatch = button.href?.match(/\/(pd|ac)\/([^/?#]+)/);
                if (hrefMatch) {
                    rawTitle = hrefMatch[2].replace(/-+/g, ' ').replace(/\bAudiobook\b/i, '').trim();
                } else {
                    return;
                }
            }
            const title = cleanTitle(rawTitle);
            let author = '';
            let node = button.closest('li.bc-list-item');
            while (node && !author) {
                const authorLabel = node.querySelector('.authorLabel');
                if (authorLabel) {
                    const authorLink = authorLabel.querySelector('a');
                    if (authorLink) {
                        author = authorLink.textContent.trim();
                        break;
                    }
                }
                node = node.parentElement?.closest('li.bc-list-item');
            }
            author = cleanAuthor(author);
            const searchUrl = getSearchUrl(author, title);
            const MAMlink = document.createElement('a');
            MAMlink.href = searchUrl;
            MAMlink.target = '_blank';
            MAMlink.className = button.className;
            MAMlink.style.display = 'flex';
            MAMlink.style.alignItems = 'center';
            MAMlink.style.justifyContent = 'center';
            MAMlink.style.fontWeight = 'bold';
            MAMlink.dataset.mamPatched = 'true';
            MAMlink.addEventListener('click', (e) => {
                e.preventDefault();
                e.stopImmediatePropagation();
                window.open(searchUrl, '_blank');
            });
            const icon = document.createElement('img');
            icon.src = icons.icon1;
            icon.alt = '';
            icon.style.cssText = 'margin-right: 6px; height: 24px; width: auto; vertical-align: middle;';
            const text = document.createTextNode('Search MyAnonamouse');
            MAMlink.appendChild(icon);
            MAMlink.appendChild(text);
            button.replaceWith(MAMlink);
            const observer = new MutationObserver(() => {
                if (!MAMlink.textContent.includes('Search MyAnonamouse')) {
                    MAMlink.innerHTML = '';
                    MAMlink.appendChild(icon);
                    MAMlink.appendChild(text);
                }
            });
            observer.observe(MAMlink, {
                childList: true,
                subtree: true,
            });
        });

        const playSpans = document.querySelectorAll('span[id^="adbl-buy-box-play-now-button"]');
        playSpans.forEach(span => {
            if (span.dataset.mamPatched) return;
            const playButton = span.querySelector('button.bc-button-text');
            const label = playButton?.innerText?.trim();
            if (label && /play/i.test(label)) {
                let title = '';
                let author = '';
                let container = span.closest('li.bc-list-item') || span.closest('[class*="productList"]') || document;
                const titleNode = container.querySelector('.bc-list-item-title a, h3 a');
                const authorNode = container.querySelector('.authorLabel a');
                if (titleNode) title = cleanTitle(titleNode.textContent.trim());
                if (authorNode) author = cleanAuthor(authorNode.textContent.trim());
                const searchUrl = getSearchUrl(author, title);
                const MAMlink = document.createElement('a');
                MAMlink.href = searchUrl;
                MAMlink.className = 'bc-button-text';
                MAMlink.target = '_blank';
                MAMlink.dataset.mamPatched = 'true';
                MAMlink.style.alignItems = 'center';
                MAMlink.style.justifyContent = 'center';
                MAMlink.style.fontWeight = 'bold';
                MAMlink.addEventListener('click', (e) => {
                    e.preventDefault();
                    e.stopImmediatePropagation();
                    window.open(searchUrl, '_blank');
                });
                const icon = document.createElement('img');
                icon.src = icons.icon1;
                icon.alt = '';
                icon.style.cssText = 'margin-right: 6px; height: 24px; width: auto; vertical-align: middle;';
                const text = document.createTextNode('Search MyAnonamouse');
                MAMlink.appendChild(icon);
                MAMlink.appendChild(text);
                const wrapper = document.createElement('span');
                wrapper.className = 'bc-button bc-button-primary bc-button-small';
                wrapper.id = 'searchmambutton';
                wrapper.style.marginTop = '8px';
                wrapper.appendChild(MAMlink);
                span.dataset.mamPatched = 'true';
                span.insertAdjacentElement('afterend', wrapper);
            }
        });

        document.querySelectorAll('span.adbl-library-listen-now-button').forEach(span => {
            if (span.dataset.mamPatched) return;

            const playButton = span.querySelector('button.bc-button-text');
            const label = playButton?.innerText?.trim();

            if (label && /listen now|play/i.test(label)) {
                let title = '';
                let author = '';
                let container = span.closest('.adbl-library-item') || span.parentElement;
                let current = container;
                while (current && (!title || !author)) {
                    const titleNode = current.querySelector('.bc-size-headline3');
                    const authorNode = current.querySelector('.authorLabel a');
                    if (!title && titleNode) title = cleanTitle(titleNode.textContent.trim());
                    if (!author && authorNode) author = cleanAuthor(authorNode.textContent.trim());
                    current = current.parentElement;
                }

                console.log('TITLE:', title);
                console.log('AUTHOR:', author);

                const searchUrl = getSearchUrl(author, title);
                const MAMlink = document.createElement('a');
                MAMlink.href = searchUrl;
                MAMlink.className = 'bc-button-text';
                MAMlink.target = '_blank';
                MAMlink.dataset.mamPatched = 'true';
                MAMlink.style.alignItems = 'center';
                MAMlink.style.justifyContent = 'center';
                MAMlink.style.fontWeight = 'bold';
                MAMlink.addEventListener('click', (e) => {
                    e.preventDefault();
                    e.stopImmediatePropagation();
                    window.open(searchUrl, '_blank');
                });

                const icon = document.createElement('img');
                icon.src = icons.icon1;
                icon.alt = '';
                icon.style.cssText = 'margin-right: 6px; height: 16px; width: auto; vertical-align: middle;';
                const text = document.createTextNode('MyAnonamouse');
                MAMlink.appendChild(icon);
                MAMlink.appendChild(text);

                const wrapper = document.createElement('span');
                wrapper.className = 'bc-button bc-button-primary bc-button-small';
                wrapper.style.marginTop = '8px';
                wrapper.appendChild(MAMlink);

                span.dataset.mamPatched = 'true';
                span.insertAdjacentElement('afterend', wrapper);
            }
        });
    }

    const runPatch = () => {
        const meta = getTitleAndAuthor();
        if (!meta) return;
        patchAdblButton();
    };

    function observeListArea() {
        const listContainer = document.querySelector('[data-test="productList"]') || document.body;
        const observer = new MutationObserver(() => {
            updateListLinks();
        });
        observer.observe(listContainer, {
            childList: true,
            subtree: true,
        });
    }

    function fullyReady(callback) {
        if (document.readyState === 'complete') {
            setTimeout(callback, 100);
        } else {
            window.addEventListener('load', () => setTimeout(callback, 100));
        }
    }

    fullyReady(() => {
        updateListLinks();
        runPatch();
        observeListArea();
    });
})();