[TS] deviantArt Download Link

Toggle ability to redirect to image file. Adds "Download" button on illustration page if missing. Show's if available download image is max-size. Adds copy button for fav.me and other meta-data. Removes open in new tab.

Устаревшая версия на 16.07.2015. Перейти к последней версии.

// ==UserScript==
// @name            [TS] deviantArt Download Link
// @namespace       TimidScript
// @version         1.1.13 beta
// @description     Toggle ability to redirect to image file. Adds "Download" button on illustration page if missing. Show's if available download image is max-size. Adds copy button for fav.me and other meta-data. Removes open in new tab.
// @author          TimidScript
// @homepageURL     https://openuserjs.org/users/TimidScript
// @copyright       © 2014 TimidScript, All Rights Reserved.
// @license         Creative Commons BY-NC-SA + Read Copyright inside the script
// @include         *//*.deviantart.com/*
// @require         https://openuserjs.org/src/libs/TimidScript/TSL_-_GM_Update.js
// @homeURL         https://openuserjs.org/scripts/TimidScript/[TS]_deviantART_Download_Link
// @grant           GM_xmlhttpRequest
// @grant           GM_info
// @grant           GM_getMetadata
// @grant           GM_getValue
// @grant           GM_setValue
// @grant           GM_deleteValue
// @grant           GM_registerMenuCommand
// @grant           GM_addStyle
// @grant           GM_setClipboard
// @icon            
// ==/UserScript==


/* License + Copyright Notice
********************************************************************************************
Copyright © TimidScript, All Rights Reserved.
[Creative Commons BY-NC-SA](http://en.wikipedia.org/wiki/Creative_Commons_license)

Redistribution and use in source and binary forms, with or without modification, are permitted provided that the
following conditions are met:

1) This copyright must be included
2) Due credits and link to original author's homepage (included in copyright).
3) Notify the original author of redistribution

TimidScript's Homepages:  [GitHub](https://github.com/TimidScript)
                          [OpenUserJS](https://openuserjs.org/users/TimidScript)
                          [GreasyFork](https://greasyfork.org/users/1455-timidscript)
*/

/* Information
**************************************************************************************************
 Version History
------------------------------------
1.1.13 (2015-07-16)
 - Bug Fix: Redirect countdown timer is reset when illustration changes
 - Colour of download button text is changed when redirection is removed
1.1.12 (2015-06-21)
 - Redirect counter interval went back to 1000ms
1.1.11 (2015-06-19)
 - Bug Fix: Correct illustration data is collected. The site stores the content of the previous page which cause parsing problems.
 Fixed by checking if the main container display is set to none.
 - Bug Fix: Redirect check is also done before timer starts
 - Added few illustration page buttons that copy text to clipboard
    * Fav.me id
    * fav.me link
    * thumbnail
    * embed code
 - Removed legacy code
 - Script icon is base64 rather than imgur link
1.1.10 (2015-05-25)
 - Added timer to direct links.
1.1.9 (2015-03-21)
 - Bug fix due to changes in layout
 - Color of download button text becomes red if max download dimensions are not available or different
 - Added a button to allow direct open of full sized image
1.0.8 (2014-08-29)
 - Added GM_update
1.0.7 (2014-08-19)
 - Cleaned up header for OUJS
1.0.6 (2014-08-06)
 - Bug fix on direct link beside the title. It does not link to the largest format of the image. Removed
 it for now.
1.0.5 (2014-07-11)
 - Bug Fix on direct links
 - Optimised code
1.0.4 (2014-07-03)
 - Added direct link without button next to the title
1.0.3 (2013-12-07)
 - Need MutationObserver on all the site. (*//*.deviantart.com/*)
1.0.2 (2013-12-07)
 - Support of flash download added
 - BugFix: the illustration page contains hidden elements that we need to check for, as we
 are trying to only change the visible one.
 - Not using TimidScript Library so remove requirement
1.0.1 (2013-10-07)
 - Initial Release
 - Add Missing download link
 - Stopped download from opening in new tab
**************************************************************************************************/
var DisplayImageOnly = GM_getValue("ReDirect"), redirectInterval;

function CreateDownloadButton(src, imgWidth, imgHeight)
{
    var holder = document.querySelector('body > div:not([style*="none"]) .dev-meta-actions');
    if (!holder) return;
    var btn = holder.querySelector("a.dev-page-button.dev-page-button-with-text.dev-page-download");
    MO.disconnect();
    if (btn && !btn.cleaned)
    {
        console.log("Remove onclick tracker");
        btn.querySelector(".label").style.color = "blue";

        btn.addEventListener("click", function (e)
        {
            e.stopImmediatePropagation();
            return false;
        }, true);

        btn.cleaned = true;
        redirectToImage();
    }
    else if (!btn)
    {
        console.log("Adding missing Download Link");
        var btn = document.createElement("a");
        btn.className = "dev-page-button dev-page-button-with-text dev-page-download";
        btn.id = "dDLButton";

        if (imgWidth != undefined) btn.innerHTML = '<i></i><span class="label" style="color:hotpink;">Download</span><span class="text">' + imgWidth + " × " + imgHeight + '</span>';
        else btn.innerHTML = '<i></i><span class="label" style="color:hotpink;>Download</span><span class="text">Flash</span>';

        btn.href = src;
        holder.appendChild(btn);
        btn.cleaned = true;
        btn.added = true;
        redirectToImage();
    }

    if (btn && !btn.adjusted)
    {
        btn.adjusted = true;
        //console.log(imgHeight, imgWidth);
        if (imgWidth != undefined)
        {
            var size = document.evaluate('//body/div[not(contains(@style,"none"))]//div[contains(@class,"dev-metainfo-details")]//dt[contains(text(),"Resolution")]/following-sibling::dd[1]', document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;
            if (size)
            {
                var notsame = true;
                var dimensions = btn.lastElementChild.textContent.match(/(\d+) × (\d+)/);
                if (dimensions)
                {
                    dimensionsO = size.textContent.match(/(\d+)×(\d+)/);

                    if (dimensionsO[1] == dimensions[1] && dimensionsO[2] == dimensions[2]) notsame = false;
                }

                if (notsame)
                {
                    btn.lastElementChild.innerHTML += " (<span style='color:red;'>" + size.textContent + "</span>)";
                }
            }
        }

        GM_addStyle("#ExtraControls {text-align:center; background-color:rgba(255,255,255,0.5); padding: 0 0 5px 0; border-radius: 2px;}");
        GM_addStyle("#ExtraControls button {font-size: 12px;}");
        var el = document.createElement("div");
        el.id = "ExtraControls";
        el.innerHTML = '<div style="weight:600;font-size:16px;">Copy To Clipboard</div><button>fav.id</button><button>fav.me</button><button>thumbnail</button><button>embed</button>';
        holder.appendChild(el);
        var btns = document.querySelectorAll("#ExtraControls button");
        for (var i = 0; i < 4; i++) btns[i].onclick = copyToClipboard;

        el = document.querySelector('body > div:not([style*="none"]) .dev-metainfo-copy-control  input[data-ga_click_event*=favme]');
        if (el)
        {
            btns[0].value = el.value.match(/fav\.me\/(.+)/i)[1];
            btns[1].value = el.value;
        }
        else
        {
            btns[0].disabled = true;
            btns[1].disabled = true;
        }
        el = document.querySelector('body > div:not([style*="none"]) .dev-metainfo-copy-control  input[data-ga_click_event*=thumbcode]');
        if (el) btns[2].value = el.value;
        else btns[2].disabled = true;
        el = document.querySelector('body > div:not([style*="none"]) .dev-metainfo-copy-control  input[data-ga_click_event*=embed]');
        if (el) btns[3].value = el.value;
        else btns[3].disabled = true;
    }

    function copyToClipboard(e)
    {
        console.log("Copied to clipboard: " + this.value);
        GM_setClipboard(this.value);
    }

    function redirectToImage()
    {
        if (!DisplayImageOnly) return;
        var countdown = GM_getValue("ReDirect-Timer");
        if (countdown == 0) btn.click();
        else
        {
            var s = document.querySelector("#oh-menu-direct span span")
            s.setAttribute("style", "color: white;background-color:red;border-color:red;font-weight:900;");
            //console.log(countdown);
            var timer = 0;
            redirectInterval = setInterval(function ()
            {
                timer++;
                s.textContent = (countdown - timer);
                if (s.textContent == 0)
                {
                    clearInterval(redirectInterval);
                    btn.click();
                }

                if (!DisplayImageOnly)
                {
                    clearInterval(redirectInterval);
                    s.removeAttribute("style");
                    s.textContent = GM_getValue("ReDirect-Timer");
                }
            }, 1000);
        }
    }

    MO.observe();
}

function AddReDirectButton()
{
    var md = document.getElementById("oh-menu-direct");
    if (md && md.URL == document.URL) return;
    else if (md)
    {
        clearInterval(redirectInterval);
        md.parentElement.removeChild(md);
    }
    else
    {
        GM_addStyle(".mItem {background-image: none; display: block; height: 49px; line-height: 49px; padding: 0px 10px; text-transform: uppercase; cursor: pointer;}");
        GM_addStyle(".mItem > span {background-color:gray; padding: 8px 10px; border-radius: 3px; font-weight: 700;}");
        GM_addStyle(".mItem > span > span {padding: 2px 4px; background-color: lightgray; border-radius: 4px; border: 2px solid transparent;}");
        GM_addStyle(".mItem.directImage > span {background-color: #4A9E18 !important; color: white;} .mItem.directImage > span > span {background-color: #82DF4B;}");
        GM_addStyle(".mItem:hover {background-color: black !important; color: white !important;}");
        GM_addStyle(".mItem.directImage:hover {color: lime!important;}");
        GM_addStyle(".mItem span > span:first-child:hover {border-color: yellow; color: yellow;}");
    }

    md = document.createElement("td");
    md.URL = document.URL;
    md.innerHTML = '<a class="oh-l mItem"><span>ReDirect <span>0</span></span></a>';
    md.className = "oh-keep";
    md.id = "oh-menu-direct";
    var ms = document.getElementById("oh-menu-submit");
    ms.parentElement.insertBefore(md, ms);

    if (GM_getValue("ReDirect", 0) == 1) md.firstElementChild.className = "oh-l mItem directImage";
    md.querySelector("span span").textContent = GM_getValue("ReDirect-Timer", 0);

    md.onclick = function (e)
    {
        if (GM_getValue("ReDirect", 0) == 0)
        {
            GM_setValue("ReDirect", 1);
            DisplayImageOnly = 1;
            md.firstElementChild.className = "oh-l mItem directImage";
        }
        else
        {
            GM_setValue("ReDirect", 0);
            DisplayImageOnly = 0;
            md.firstElementChild.className = "oh-l mItem";
        }
    };

    md.querySelector("span span").onclick = function (e)
    {
        e.stopImmediatePropagation();
        switch (parseInt(this.textContent))
        {
            case 0:
                this.textContent = 2;
                break;
            case 2:
                this.textContent = 4;
                break;
            case 4:
                this.textContent = 8;
                break;
            default:
                this.textContent = 0;
                break;
        }

        GM_setValue("ReDirect-Timer", this.textContent);
    };
}

// MutationObserver Control
var MO =
{
    observer: null,

    disconnect: function ()
    {
        if (MO.observer)
        {
            MO.observer.disconnect();
            //console.log("deviantArt Download Link: Observer Disconnected");
        }
    },

    //Observes body changes
    observe: function ()
    {
        //MO.callback(); //Just in case it gets missed. Happens occasionally

        var mo = window.MutationObserver || window.MozMutationObserver || window.WebKitMutationObserver;
        if (mo)
        {
            MO.observer = null;
            MO.observer = new mo(MO.callback);
            MO.observer.observe(document.body, { characterData: true, attributes: true, childList: true, subtree: true });
            //console.warn("deviantArt Download Link: Observer Connected");
        }
    },

    callback: function (mutations)
    {
        AddReDirectButton();
        if (!document.URL.match(/\.deviantart\.com\/(#\/)?art/i)) return;

        if (document.querySelector('body > div:not([style*="none"]) .dev-view-deviation #flashed-in iframe')) //Illustration contains a frame with flash content
        {
            console.log("Illustration Type: flash");
            window.addEventListener('message', function (event)
            {
                //console.log(event.data);
                if (event.data.match(/^dDL_SWFurl:/i))
                    CreateDownloadButton(event.data.replace(/^dDL_SWFurl:/i, ""));
            }, false);
            return
        }

        var img = document.querySelector('body > div:not([style*="none"]) .dev-content-full');
        if (img)
        {
            CreateDownloadButton(img.src, img.naturalWidth, img.naturalHeight);
        }
    }
}

console.info("[TS] deviantArt Download Link");
(function ()
{
    if (window === window.top)
    {
        MO.observe();
    }
    else if (document.URL.match(/sandbox\.deviantart\.com\/\?fileheight/i))
    {
        //Pass flash src information that is stored in the iframe
        var flash = document.getElementsByTagName("embed")[0];

        //Do a delay to allow observe to hook in
        if (flash)
            setTimeout(function () { window.top.postMessage('dDL_SWFurl:' + flash.src, '*'); }, 1000);
    }
})();