[OFFICIAL] Steam Workshop Downloader :: IO

Quickly download files from the steam workshop- using steamworkshopdownloader.io.

// ==UserScript==
// @name [OFFICIAL] Steam Workshop Downloader :: IO
// @description Quickly download files from the steam workshop- using steamworkshopdownloader.io.
// @namespace http://steamworkshopdownloader.io/extension/
// @match *://*.steamcommunity.com/sharedfiles/filedetails/?id=*
// @match *://*.steamcommunity.com/workshop/filedetails/?id=*
// @run-at document-end
// @version 1.6
// @icon https://steamworkshopdownloader.io/logo.png
// ==/UserScript==

const arrow = "";

function getIframeAnchor()
{
    return getPageType() === 'item' ? jQuery("div.workshopItemPreviewArea") : jQuery("div.collectionControls");
}

function getButtonAnchor() {
    return getPageType() === 'item' ? jQuery("#SubscribeItemBtn") : (
        jQuery("div.subscribeCollection:contains('Subscribe')").find('> span').last().length
            ? jQuery("div.subscribeCollection:contains('Subscribe')").find('> span').last() // Has a subscribe collection box, use that.
            : jQuery("div.workshopItemDescriptionTitle:contains('Items')").find("> span") // Use the defailt items count title.
    );
}

/**
 * Determine the type of page based on the elements displayed on it.
 */
function getPageType() {
    if (jQuery("div.game_area_purchase_game").length && jQuery("span.subscribeText").length) {
        return 'item';
    } else if (jQuery("#mainContentsCollection").length) {
        return 'collection';
    }

    return null;
}


/**
 * When the download button is pressed, insert an iframe with the download page into the site.
 */
function onDownloadWorkshopItem() {
    // Hide the iframe in case it already exists- a toggle
    const downloadButton = jQuery('#downloadWorkshopItemButton');
    const existingItemPanel = jQuery('#downloadWorkshopItemIframe');
    if (existingItemPanel.length) {
        existingItemPanel.remove();
        downloadButton.find('img').first().css('transform', '');
        return;
    }

    // Button was pressed, invert its icon, then insert the iframe.
    downloadButton.find('img').first().css('transform', 'rotate(180deg)');

    getIframeAnchor().after(`<iframe
            allowtransparency='true'
            onload="window.parent.postMessage(['steamdownloaderFrameLoaded', true], '*')"
            style="height: 0; margin: 0; padding: 0; border: 0; width: 100%;"
            scrolling="no"
            id="downloadWorkshopItemIframe"
            src="//steamworkshopdownloader.io/extension/embedded/${new URLSearchParams(window.location.search).get('id')}"/>`);
}

/**
 * Add window events.
 */
function addWindowEvents() {
    /**
     * The plugin loads an iframe with the download site on it.
     * We are able to make the iframe access its outside frame using messages.
     * We use this functionality to push the scroll height to the parent window, resize the iframe to fit its content using a script and to close the iframe.
     */
    window.addEventListener('message', function (e) {
        const frame = jQuery("#downloadWorkshopItemIframe");
        const button = jQuery('#downloadWorkshopItemButton');

        //
        const eventName = e.data[0];
        const data = e.data[1];

        switch (eventName) {
            case 'steamdownloaderButtonPressed':
                onDownloadWorkshopItem();
                break;
            case 'steamdownloaderFrameLoaded':
                getIframeAnchor().css("margin-bottom", "8px");
                break;
            case 'steamdownloaderResizeIframe':
                frame.css('height', `${data}px`);
                break;
            case 'steamdownloaderScrollToFrame':
                jQuery([document.documentElement, document.body]).animate({
                    scrollTop: frame.offset().top - 100
                }, 100);
                break;
            case 'steamdownloaderClose':
                button.trigger('click');
                break;
        }
    }, false);
}

/**
 * Add a download button into the page on the specific anchor element.
 */
function addButton() {
    const downloadButton = getPageType() === 'item' ?
        `<a id="downloadWorkshopItemButton" class="btn_darkblue_white_innerfade btn_medium" style="margin: 2px; float: right;" onclick="window.parent.postMessage(['steamdownloaderButtonPressed', true], '*')"> 
        <img style="position: absolute; margin-left: 6px; margin-top: 6px;" width="16px" height="16px" src="${arrow}"/> 
        <span style="padding-left: 32px; padding-right: 32px;"> 
            Download 
        </span> 
    </a>` : `<a id="downloadWorkshopItemButton" class="btn_darkblue_white_innerfade btn_medium" style="height: 26px; float: right;" onclick="window.parent.postMessage(['steamdownloaderButtonPressed', true], '*')"> 
        <img style="position: absolute; margin-left: 6px; margin-top: 4px;" width="14px" height="14px" src="${arrow}"/> 
        <span style="padding-left: 32px; padding-right: 32px; line-height: 22px;"> 
            Download all
        </span> 
    </a>`;

    if (getPageType() === 'item') {
        getButtonAnchor().css('float', 'right');
    }

    getButtonAnchor().after(downloadButton);
}

(() => {
    const load = () => {
        // Only neccesary when ran as a userscript: keep retrying until jQuery is available from the page headers.
        if(typeof jQuery === 'undefined') {
            setTimeout(load, 50);
        }

        if (getPageType() == null) {
            return;
        }

        addWindowEvents();
        addButton();
    };

    load();
})();