Greasy Fork

osu! download redirector

redirect download request from ppy.sh to bloodcat.com

当前为 2019-07-16 提交的版本,查看 最新版本

// ==UserScript==
// @name         osu! download redirector
// @namespace    https://github.com/lywbh/osu-download-redirector
// @version      0.1.5
// @description  redirect download request from ppy.sh to bloodcat.com
// @author       kamimi
// @match        https://osu.ppy.sh/*
// @grant        none
// ==/UserScript==

(function () {
    'use strict';

    /**
     * when page loaded/refreshed.
     */
    window.onload = () => {
        handleNewSite();
        handleOldSite();
    };

    /**
     * The new osu site is a single page app, modify the xhr here for listening page redirect.
     */
    (function () {
        function ajaxEventTrigger(event) {
            let ajaxEvent = new CustomEvent(event, {detail: this});
            window.dispatchEvent(ajaxEvent);
        }

        let oldXHR = window.XMLHttpRequest;

        function newXHR() {
            let realXHR = new oldXHR();
            realXHR.addEventListener('readystatechange', function () {
                ajaxEventTrigger.call(this, 'ajaxReadyStateChange');
            }, false);
            return realXHR;
        }

        window.XMLHttpRequest = newXHR;
    })();

    /**
     * listen page redirect.
     */
    window.addEventListener('ajaxReadyStateChange', function (e) {
        if (e.detail.readyState === 4) {
            console.log(e.detail); // XMLHttpRequest Object
            if (/^https:\/\/osu.ppy.sh\/beatmapsets(\/)?[0-9]*$/.test(e.detail.responseURL)) {
                // wait a while for dom rendering
                setTimeout(handleNewSite, 500);
            }
        }
    });

    /**
     * add icons for new site.
     */
    function handleNewSite() {
        if (window.location.pathname === "/beatmapsets") {
            // beatmap list
            let beatmapList = document.getElementsByClassName("beatmapsets__items")[0];
            if (beatmapList) {
                let iconBoxes = beatmapList.getElementsByClassName("beatmapset-panel__icons-box");
                addIconList(iconBoxes);
                beatmapList.addEventListener("DOMNodeInserted", e => {
                    let iconBoxes = e.target.getElementsByClassName("beatmapset-panel__icons-box");
                    addIconList(iconBoxes);
                });
            }
        } else if (/^\/beatmapsets\/[0-9]*.*$/.test(window.location.pathname)) {
            // beatmap details
            let iconBoxes = document.getElementsByClassName("beatmapset-header__buttons");
            addIconDetail(iconBoxes);
        }
    }

    /**
     * add icons for old site.
     */
    function handleOldSite() {
        if (/^\/(s)|(b)\/[0-9]*.*$/.test(window.location.pathname)) {
            // beatmap details only at old site
            let iconBoxes = document.getElementsByClassName("paddingboth");
            addIconDetailOld(iconBoxes);
        }
    }

    function addIconList(iconBoxes) {
        for (let i = 0; i < iconBoxes.length; i++) {
            let downloadLink = iconBoxes[i].getElementsByClassName("js-beatmapset-download-link")[0];
            if (downloadLink) {
                let newNode = deepCloneNode(downloadLink);
                replaceUrl(newNode, -2);
                newNode.style.color = "#4ad";
                iconBoxes[i].appendChild(newNode);
            }
        }
    }

    function addIconDetail(iconBoxes) {
        for (let i = 0; i < iconBoxes.length; i++) {
            let downloadLink = iconBoxes[i].getElementsByClassName("js-beatmapset-download-link")[0];
            if (downloadLink) {
                let newNode = deepCloneNode(downloadLink);
                replaceUrl(newNode, -2);
                newNode.style.backgroundColor = "#4ad";
                iconBoxes[i].appendChild(newNode);
            }
        }
    }

    function addIconDetailOld(iconBoxes) {
        for (let i = 0; i < iconBoxes.length; i++) {
            let downloadButtons = iconBoxes[i].getElementsByClassName("beatmapDownloadButton");
            let downloadButton = downloadButtons[downloadButtons.length - 1];
            if (downloadButton) {
                let downloadLink = downloadButton.getElementsByClassName("beatmap_download_link")[0];
                if (downloadLink) {
                    let newButton = downloadButton.cloneNode();
                    let newNode = downloadLink.cloneNode();
                    let img = document.createElement("img");
                    img.src = "";
                    newNode.appendChild(img);
                    replaceUrl(newNode, -1);
                    newButton.appendChild(newNode);
                    iconBoxes[i].insertBefore(newButton, downloadButton);
                }
            }
        }
    }

    /**
     * replace the origin url with bloodcat.com in download button element
     */
    function replaceUrl(linkNode, pathIndex) {
        const bloodUrl = "https://bloodcat.com/osu/s/";
        if (linkNode.href && linkNode.href.indexOf(bloodUrl) === -1) {
            let splitUrl = linkNode.href.split("/");
            let index;
            if (pathIndex >= 0) {
                index = pathIndex;
            } else {
                index = splitUrl.length + pathIndex;
            }
            let setId = splitUrl[index];
            linkNode.href = bloodUrl + setId;
        }
    }

    function deepCloneNode(linkNode) {
        let objE = document.createElement("div");
        objE.innerHTML = linkNode.outerHTML;
        return objE.childNodes[0];
    }
})();