// ==UserScript==
// @name         NexDirect v2
// @namespace    http://nexerq.cf/
// @homepage     https://github.com/NexerqDev/NexDirect
// @version      0.3.0
// @icon         https://raw.githubusercontent.com/NexerqDev/NexDirect/master/Designs/logo.png
// @description  Adds download button to page to use NexDirect & replaces the heart button on the listings -- You must visit the Settings panel (the logo in the bottom right) and register the URI scheme before you are able to use this script.
// @author       Nicholas Tay (Nexerq / @n2468txd) <nexerq@gmail.com>
// @license      https://github.com/NexerqDev/NexDirect/blob/master/LICENSE
// @match        http*://osu.ppy.sh/*
// @grant        none
// @updateURL    https://raw.githubusercontent.com/NexerqDev/NexDirect/master/nexdirect.user.js
// ==/UserScript==

(function() {
    'use strict';
    var nexDirectImage = ""; 
    var miniIcon = "";
    
    function welcome() {
        console.log("%cWelcome to NexDirect.", "font-size: 35px; line-height: 38px; font-weight: bold; color: white; text-shadow: 0 0 5px green;");
    }

    function log(t) {
        console.log("%c[NexDirect] %c" + t, "font-weight: bold;", "");
    }
    
    function injectListingPage() {
        // Replace the <3 icons with nexdirect
        log("Injecting download links into beatmap listing page...");
        var $heartIcons = $("div.beatmapListing .beatmap .bmlist-options i.icon-heart");
        if (!$heartIcons || ($heartIcons.length < 1))
            return log("Could not find any heart buttons to replace."); // rip
        
        log("Found " + $heartIcons.length + " heart icons. Replacing...");
        $heartIcons.each(function() {
            var $this = $(this);
            var $closestBeatmap = $this.closest(".beatmap");
            if (!$closestBeatmap || $closestBeatmap.length < 1)
                return log("Could not find closest beatmap for element");

            var beatmapSetId = $closestBeatmap.attr("id");
            if (!beatmapSetId)
                return log("Could not find beatmap set ID for element");
            
            $this.parent().attr("href", "nexdirect://" + beatmapSetId);
            $this.replaceWith("<div class=\"icon-nexdirect\"></div>");
        });
        log("Injected download buttons.");
        
        $('head').append("<style type=\"text/css\"> .icon-nexdirect { cursor: default; background-image: url(" + miniIcon + "); opacity: 0.5; width: 20px; height: 20px; display: block !important; margin-left: 2px; } .icon-nexdirect:hover { opacity: 1 } </style>");
        log("Injected additional CSS.");
    }
    
    function injectDownloadPage() {
        log("Injecting download button into beatmap download page...");
        // Get the beatmap *SET* id
        var $imgSrc = $("img.bmt").attr("src"); // jack from image preview src elem
        if (!$imgSrc)
            return log("Could not find image preview src for ID."); // rip

        log("Found image preview src element.");
        
        var beatmapSetId = $imgSrc.match(/\/(\d+)l\.jpg/);
        if (!beatmapSetId || !beatmapSetId[1])
            return log("Could not identify the beatmap set ID."); // rip
        beatmapSetId = beatmapSetId[1]; // its match 1

        var $downloadButton = $(".beatmapDownloadButton");
        if (!$downloadButton || $downloadButton.length < 1) {
            log("Could not find the beatmap download button. Probably not logged in, going to try use description box.");

            $downloadButton = $(".posttext");
            if (!$downloadButton || $downloadButton.length < 1) // it really shouldnt get here
                return log("Still cannot find element to inject.");
        }

        if ($downloadButton.length > 1) {
            // rotate 180 for lined up, usually dont because it stands out - https://stackoverflow.com/questions/14233341/how-can-i-rotate-an-html-div-90-degrees
            log("Found a no video/osu!supporter button, injecting after that one and rotating for conformity.");
            $downloadButton = $($downloadButton[0]);
            $('head').append("<style type=\"text/css\"> .nexdirectDownload { -webkit-transform: rotate(180deg); -moz-transform: rotate(180deg); -o-transform: rotate(180deg); -ms-transform: rotate(180deg); transform: rotate(180deg); } </style>");
        }
        $downloadButton.before("<div class=\"nexdirectDownload beatmapDownloadButton\"><a href=\"nexdirect://" + beatmapSetId + "\"><img src=\"" + nexDirectImage + "\" alt=\"Download with NexDirect\"/></a></div>");
        log("Injected download button.");
    }

    function injectNewListingPage(firstLoad) {
        log("[new] Finding new download button(s) to load into beatmap listing page...");
        var $dlIconBoxes = $("div.beatmapset-panel__icons-box:not([nexdirect-loaded])");
        if (!$dlIconBoxes || ($dlIconBoxes.length < 1))
            return log("Could not find any new download icons to add."); // rip
        
        log("Found " + $dlIconBoxes.length + " new download icons. Injecting NexDirect icons...");
        $dlIconBoxes.each(function() {
            var $this = $(this);
            var $idElem = $this.parents("div.beatmapset-panel").children("a.js-audio--play");
            if (!$idElem || ($idElem.length < 1))
                return log("Could not find ID elem."); // rip

            var beatmapSetId = $idElem.attr("data-audio-url").match(/\/(\d+)\.mp3$/);
            if (!beatmapSetId || !beatmapSetId[1])
                return log("Could not identify the beatmap set ID."); // rip
            beatmapSetId = beatmapSetId[1]; // its match 1

            $this.attr("nexdirect-loaded", "true"); // give it an attr to mark beatmap set as loaded
            $this.prepend("<a href=\"nexdirect://" + beatmapSetId + "\" class=\"beatmapset-panel__icon\"><img src=\"" + miniIcon + "\" alt=\"Download with NexDirect\" style=\"filter: brightness(0.35); -webkit-filter: brightness(0.35); opacity: 0.9; margin-bottom: 5px;\"/></a>");
        });
        log("Injected download icons to beatmap listing page.");
    }
    
    function injectNewDownloadPage() {
        log("[new] Injecting download button into beatmap download page...");
        // Get set id
        var $audioElem = $(".js-audio--play");
        if (!$audioElem || ($audioElem.length < 1))
            return log("Could not find link element."); // rip
        log("Found link element.");

        var previewLinkMatch = $audioElem.attr("data-audio-url").match(/\/preview\/(\d+)\.mp3/);
        if (!previewLinkMatch || !previewLinkMatch[1])
            return log("Could not retrieve beatmap set ID."); // rip
        var beatmapSetId = previewLinkMatch[1]; // set the group
        log("Found beatmap set ID.");
        
        var nexDirectInject = `
<a href="${"nexdirect://" + beatmapSetId}" data-turbolinks="false" class="btn-osu-big btn-osu-big--beatmapset-header" style="filter: hue-rotate(284deg); -webkit-filter: hue-rotate(284deg);">
    <div class="btn-osu-big__content">
        <div class="btn-osu-big__left">
            <span class="btn-osu-big__text-top">NexDirect</span>
        </div>
        <div class="btn-osu-big__icon">
            <img src="${miniIcon}" alt="NexDirect icon" style="filter: brightness(0) invert(1); -webkit-filter: brightness(0) invert(1);">
        </div>
    </div>
</a>
`;

        var $buttons = $(".beatmapset-header__buttons");
        if (!$buttons || ($buttons.length < 1))
            return log("Couldn't find buttons elem to inject.");
        var $buttonAs = $buttons.find("a");

        log("Injecting download button.");
        if (!$buttonAs || ($buttonAs.length < 1)) {
            // nicer to inject here if no buttons
            log("No buttons, injecting to mapper portion instead.");

            // needs wrapping when injecting in this spot
            $(".beatmapset-mapping__content").after(`<div class="nexdirect--dl-wrapper" style="margin-left: 25px;">` + nexDirectInject + `</div>`);
        } else {
            $buttons.append(nexDirectInject);
        }

        var $osuDirectButton = $buttons.find("a:contains('osu!direct')");
        if ($osuDirectButton.legnth > 0 && $osuDirectButton.attr("href").indexOf("osu://") < 0) { // no supporter, just get rid of that obsolete button
            log("Found osu!supporter button, but user has no osu!supporter, deleting the obsolete element...");
            $osuDirectButton.remove();
        }

        log("[new] Injected download button.");
    }

    
    function detectInjection(firstLoad) {
        var newSite = typeof osu !== "undefined";
        var path = window.location.pathname;

        if (newSite) {
            if (/\/beatmapsets\/(\d+)/.test(path)) {
                if (!firstLoad)
                    log("----- PAGE CHANGE - beatmap page -----");
                injectNewDownloadPage();
            }
            else if (path.indexOf("/beatmapsets") === 0) {
                if (!firstLoad)
                    log("----- PAGE CHANGE - beatmap sets -----");
                injectNewListingPage(true);
            }

            if (firstLoad) { // new site uses turbolinks, so use this to check page changes
                log("[new] Attached turbolink cross-page handlers.");
                var $document = $(document);
                $document.on("turbolinks:load", function() { setTimeout(function() { detectInjection(); }, 650); }); // delay for page load change
                $document.on("beatmap:search:done", function() { log("--- BEATMAP SEARCHED ---"); injectNewListingPage(); }); // beatmap search page load complete bind
                $document.on("beatmap:load_more", function() {
                    if (NexDirect.loading) // need a loading var as for some reason the event gets fired 3 times each scroll as well
                        return;

                    NexDirect.loading = true;
                    log("--- VALID MORE BEATMAP LOADS REQUESTED ---");
                    setTimeout(function() {
                        injectNewListingPage();
                        setTimeout(function() { NexDirect.loading = false; }, 250); // allow loading again
                    }, 650); // delay to wait for load more to load and then inject again
                });
            }
        } else {
            if (path.indexOf("/p/beatmaplist") === 0)
                injectListingPage();
            else
                injectDownloadPage();
        }
    }

    window.NexDirect = {};
    welcome();
    
    setTimeout(function() {
        detectInjection(true);
    }, 650);
})();