digitaledgestudiosgmail.com / Apple Trailer Download HD+

// ==UserScript==
// @name            Apple Trailer Download HD+
// @namespace       http://www.digitaledgestudios.nl/
// @author          mhu
// @description     Download movie trailers from the Apple iTunes Movie Trailers site
// @include         http://trailers.apple.com/trailers/*/*
// @exclude         http://trailers.apple.com/trailers/*/*/gallery/*
// @version         2.0.25
// @require         https://greasyfork.org/libraries/GM_config/20131122/GM_config.js
// @grant           GM_addStyle
// @grant           GM_xmlhttpRequest
// @grant           GM_registerMenuCommand
// @grant           GM_setClipboard
// @noframes
// ==/UserScript==


// based on jQuery.browser
browserInfo = function () {
    var browser = {},
        uaMatch = function (ua) {
            ua = ua.toLowerCase();

            var match = /(chrome)[ \/]([\w.]+)/.exec(ua) ||
                /(webkit)[ \/]([\w.]+)/.exec(ua) ||
                /(opera)(?:.*version|)[ \/]([\w.]+)/.exec(ua) ||
                /(msie) ([\w.]+)/.exec(ua) ||
                /(trident)[ \/](?:.*? rv:([\w.]+))/.exec(ua) ||
                ua.indexOf("compatible") < 0 && /(mozilla)(?:.*? rv:([\w.]+)|)/.exec(ua) ||
                [];

            return {
                browser: match[1] || "",
                version: match[2] || "0"
            };
        },
        matched = uaMatch(navigator.userAgent);

    browser.version = 0;
    if (matched.browser) {
        browser[matched.browser] = true;
        browser.version = parseFloat(matched.version);

        if (browser.trident) {
            // IE11+
            browser.msie = true;
        }

        // Chrome is Webkit, but Webkit is also Safari.
        if (browser.chrome) {
            browser.webkit = true;
        } else if (browser.webkit) {
            browser.safari = true;
        }
    }

    return browser;
}();

//
// Workaround for GM_xmlhttpRequest with GreaseMonkey 1.x on Firefox 32
// Cheating GM_xmlhttpRequest leaking checking use arguments which could be converted to false.
// https://gist.github.com/tiansh/bbe60ec5c9c0531643db
//
var GM_xmlhttpRequest = (function () {
    var old = GM_xmlhttpRequest;

    // only install patch on Firefox 32
    if (!browserInfo.mozilla || browserInfo.version != 32) {
        return old;
    }

    return function (details) {
        var i, x = new Number(0);
        for (i in details) {
            if (details.hasOwnProperty(i)) {
                x[i] = details[i];
            }
        }
        return old(x);
    };
}());

(function() {
    "use strict";

    // constants
    var TITLE = 'Apple Trailer Download HD+',
        ERROR_MSG = 'No downloadable trailers found',
        CLICK_MSG = 'Please click the Watch the trailer link',
        APPLEERRORINDICATORS = /(class="page-errors"|pageType="errorPage")/i,
        APPLEERRORINDICATOR_OLD = /pageType="errorPage"/i,
        SIZES = [ 'Small (480p)', 'Normal (720p)', 'Large (1080p)' ],
        reBackToTrailers = new RegExp('<div class="[^"]*back-to-trailers[^"]*">[^<]*<a\b', 'img'),

    // globals
        g_allLinks = {"480p": [], "720p": [], "1080p": []},
        g_atdcontainer,
        g_baseUrl;

    /**
    * This is where we start. Inject styles.
    * Will run automatically after the load event has fired.
    */
    function initScript() {
        var css = '#atdContainer {background-color: rgba(0,0,0, 0.9); background-image:-moz-linear-gradient(right, #B5B5B5, #7D7D7D); background-image:-webkit-linear-gradient(right, #B5B5B5, #7D7D7D); background-image:linear-gradient(right, #B5B5B5, #7D7D7D); border:2px solid white; border-radius:5px; bottom:10px; box-shadow:0 -1px 5px rgba(0, 0, 0, 0.75); color:#333; display:table; font-family:Arial,sans-serif; height:35px; max-width:200px; min-width:120px; opacity:.9; padding:0 5px; position:fixed; right:10px; text-align:left; z-index:900;} ' +
            '#atdHeader { background-color: rgba(0,0,0, 1); background-image: -moz-linear-gradient(right, rgba(120, 120, 120, 0.55), rgba(75, 75, 75, 0.75)); background-image: -webkit-linear-gradient(right, rgba(120, 120, 120, 0.55), rgba(75, 75, 75, 0.75)); background-image: linear-gradient(right, rgba(120, 120, 120, 0.55), rgba(75, 75, 75, 0.75)); border-top-left-radius: 5px; border-top-right-radius: 5px; box-shadow:-1px -1px 2px black inset; margin: 0 -5px; padding: 2px 3px 5px; text-align: center; } ' +
            '#atdHeaderLink { color: #FFFFFF; font: small-caps bold 0.9em/1.1em helvetica,arial,sans-serif; text-shadow: 1px 0 rgba(0, 0, 0, 0.9); } ' +
            '#atdContainer li {display: list-item; font-size:1em; line-height:1.4em; padding-left:17px; background:url("") no-repeat scroll -2px 1px transparent;} ' +
            '#atdContainer li > a {color:#0F0F1F; font-weight:normal; font-size:0.9em; text-decoration:none;vertical-align: middle;cursor:pointer; } ' +
            '#atdContainer li.error {color:#D90000; font-weight:normal; font-size:0.9em; text-decoration:none;} ' +
            '#atd480p, #atd720p, #atd1080p, #atdLoader {box-shadow:-1px -1px 3px #000000 inset;} ' +
            '#atd1080p {padding-bottom:4px; } ' +
            '#atdContainer > div:not(.toggle) {margin:0 -5px; padding:3px 5px;} ' +
            '#atdContainer .toggle {cursor:pointer;border: thin solid black; box-shadow: 0px -1px 3px rgba(255, 255, 255, 0.25) inset; color: white; font-size: 1em; font-weight: bold; margin: 0 -5px; padding: 0 5px 0 0; text-align: right;} ' +
            '#atdContainer .toggle:hover {margin: 0 -6px; box-shadow:0 2px 5px rgba(0, 0, 0, 0.75);} ' +
            '#atdContainer .toggle:last-of-type {margin-bottom:1px;} ' +
            '#atdLoader {background:url("") center 15px no-repeat;} ' +
            '#atdLoaderText {font-size:0.9em; margin:0; padding:30px 0 5px; text-align:center;} ' +
            '#atdError {background-image:-moz-linear-gradient(top, rgba(255, 0, 0, 0.2), rgba(255, 0, 0, 0.6));-webkit-linear-gradient(top, rgba(255, 0, 0, 0.2), rgba(255, 0, 0, 0.6));linear-gradient(top, rgba(255, 0, 0, 0.2), rgba(255, 0, 0, 0.6));color:black;} ' +
            '.atdListing {margin-bottom:0}' +
            '.errorMsg {font-weight:bold; font-size:0.8em;}' +
            '.roundBottomCorners {border-bottom-left-radius:5px; border-bottom-right-radius:5px;} ' +
            '.hidden {display:none;}';

        writeLog(TITLE + " script started");

        //  update the style (repeat after awhile, to detect background changes)
        addNewStyle(css);
        getTrailerListingColor(addNewStyle);
        setTimeout(function() {getTrailerListingColor(addNewStyle);}, 500);
        setInterval(function() {getTrailerListingColor(addNewStyle);}, 1500);

        // add header
        g_atdcontainer.appendChild(createAtdHeader());

        // Show loading animation until fetching necessary data has finished.
        g_atdcontainer.appendChild(createLoader());
        document.body.insertBefore(g_atdcontainer, document.body.firstChild);
    }

    /**
    * Loads the backbone collection containing the download links (async)
    */
    function getBackboneTrailerPage() {
        writeLog(TITLE + ": Downloading '" + g_baseUrl + "data/page.json'...");
        GM_xmlhttpRequest({
            "url": g_baseUrl + "data/page.json",
            "method": "GET",
            "synchronous": false,
            "timeout": 60*1000,
            "onload": function(xhr) {
                // analyse the contents
                if (!xhr.responseText || APPLEERRORINDICATORS.test(xhr.responseText)) {
                    if (xhr.responseText && APPLEERRORINDICATOR_OLD.test(xhr.responseText))
                    {
                        parseOldTrailerPage(); // try old page type
                    }
                    else
                    {
                        getTrailerPage(); // try trailer page
                    }
                }
                else {
                    parseBackboneTrailerPage(xhr.responseText);
                }
            },
            "ontimeout": function() {
                showError("Timeout occurred");
            },
            "onerror": function() {
                showError("Could not download data/page.json");
            }
        });
    }

    function parseBackboneTrailerPage(page) {
        var collection, i, clip, src_title, sizes;

        writeLog(TITLE + ": Analyzing downloaded page...");

        try {
            // convert to json
            collection = JSON.parse(page);

            // get all the HD links from the json collection
            if (collection.clips && collection.clips.length) {
                for(i = 0; i < collection.clips.length; i++) {
                    clip = collection.clips[i];
                    src_title = clip.title || "Trailer";

                    if (clip.versions && clip.versions.enus && clip.versions.enus.sizes) {
                        sizes = clip.versions.enus.sizes;
                        if (sizes.sd) {
                            addDownloadLink(sizes.sd.src, src_title, "480p");
                        }
                        if (sizes.hd720) {
                            addDownloadLink(sizes.hd720.src, src_title, "720p");
                        }
                        if (sizes.hd1080) {
                            addDownloadLink(sizes.hd1080.src, src_title, "1080p");
                        }
                    }
                }

                showAdtContainer();
            } else {
                showError('');
            }
        }
        catch (err) {
            showError("Could not parse page.json: " + (err.description || err.message));
        }
    }

    // start getting the links
    function getTrailerPage() {
        writeLog(TITLE + ": Downloading '" + g_baseUrl + "includes/large.html'...");
        GM_xmlhttpRequest({
            "method": "GET",
            "synchronous": false,
            "url": g_baseUrl + "includes/large.html",
            "timeout": 60*1000,
            "onload": function(xhr, status) {
                // analyse the contents
                if (!xhr.responseText || APPLEERRORINDICATORS.test(xhr.responseText)) {
                    // check if a "watch the trailer" link exists
                    if ($('#showtimesmain').getElementsByClassName('back-to-trailers')) {
                        showMessage(CLICK_MSG);
                    }
                    else {
                        showError("Could not find a valid trailer page");
                    }
                }
                else {
                    getTrailerSubPages(xhr.responseText);
                }
            },
            "ontimeout": function() {
                showError("Timeout occurred");
            },
            "onerror": function() {
                showError("Could not download includes/large.html");
            }
        });
    }

    /**
    * Loads the trailer subpage containing the download links (async)
    */
    function getDownloadlinks(url, title, cb) {
        GM_xmlhttpRequest({
            "url": g_baseUrl + url,
            "method": "GET",
            "synchronous": false,
            "timeout": 60*1000,
            "onload": function(xhr) {
                // analyse the contents
                if (!xhr.responseText || APPLEERRORINDICATORS.test(xhr.responseText)) {
                    showError("Could not download includes/large.html");
                }
                else {
                    insertDownloadLink(xhr.responseText, title);
                }
                if (cb) { cb(); }
            },
            "ontimeout": function() {
                showError("Timeout occurred");
                if (cb) { cb(); }
            },
            "onerror": function() {
                showError("Could not download trailer subpage");
                if (cb) { cb(); }
            }
        });
    }

    /**
    * Adds the trailer links to the respective arrays
    */
    function insertDownloadLink(page, title) {
        var reLink = new RegExp('<a class="movieLink" href="([^\\?"]+).mov', 'img'),
            result;

        // get all the HD links from the trailer page
        reLink.lastIndex = 0;
        if ((result = reLink.exec(page))) {
            if (result.length >= 2) {
                addDownloadLink(result[1], title);
            }
        }
    }

    function addDownloadLink(url, title, curSize)
    {
        var size, i, size_url, size_links, duplicate;

        try
        {
            for(size in g_allLinks) {
                if (!g_allLinks.hasOwnProperty(size)) {
                    continue;
                }
                if (curSize && curSize != size) {
                    continue;
                }

                size_url = normalizeTrailerLink(url, size);
                size_links = g_allLinks[size];

                // avoid duplicates
                duplicate = false;
                for (i = 0; i < size_links.length; i++) {
                    if (size_links[i] && size_links[i].href == url) {
                        duplicate = true;
                        break;
                    }
                }

                if (!duplicate) {
                    writeLog(TITLE + ": - Trailer link found: '" + size_url + "' (" + url + ")");
                    size_links.push(createElem('a', { 'textContent': title, 'href': size_url, 'title': 'Download ' + size }));
                }
            }
        }
        catch (err)
        {
            writeLog(TITLE + ": " + (err.message || err.description));
        }
    }

    function normalizeTrailerLink(url, size) {
        // url = url.toLowerCase();  don't change the case!!!

        // remove size
        url = url.replace(/_h?([0-9]+p?)(.mov)?$/, '');

        // workaround some incorrect links
        url = url.replace("http://movietrailers.apple.com", "http://trailers.apple.com");
        url = url.replace("http://trailers.apple.com/movies//trailers/independent", "http://trailers.apple.com/movies/independent");

        // add requested size
        url = url + "_h" + size + ".mov";

        return url;
    }

    /**
    * Scan the page for HD quicktime movies. Get all <a> tags with specific href's
    */
    function getTrailerSubPages(page) {
        var src_url, src_title,
        reLink = new RegExp('<a href="includes/([^#"]+)#?.[^"]*"', 'img'),
        reTitle_old = new RegExp('<h4>([^<]+)</h4>', 'img'),
        reTitle_new = new RegExp('<h3 title="[^"]+">([^<]+)</h3>', 'img'),
        result = null, requests, j,
        links = [],
        titles = [];

        writeLog(TITLE + ": Analyzing downloaded page...");

        // get all the HD links from the trailer page
        reLink.lastIndex = 0;
        while ((result = reLink.exec(page))) {
            if (result.length >= 2 && links.indexOfCI(result[1]) < 0) {
                writeLog(TITLE + ": - Trailer page found: '" + result[1] + "'");
                links.push(result[1]);
            }
        }

        // get all the titles from the trailer page
        reTitle_old.lastIndex = 0;
        while ((result = reTitle_old.exec(page))) {
            if (result.length >= 2) {
                writeLog(TITLE + ": - Trailer title found: '" + result[1] + "'");
                titles.push(result[1]);
            }
        }

        if (titles.length === 0) {
            reTitle_new.lastIndex = 0;
            while ((result = reTitle_new.exec(page))) {
                if (result.length >= 2) {
                    writeLog(TITLE + ": - Trailer title found: '" + result[1] + "'");
                    titles.push(result[1]);
                }
            }
        }

        if (links.length > 0) {
            requests = links.length;
            for (j = 0; j < links.length; j++) {
                src_url = "includes/" + links[j];
                src_title = (links.length == titles.length ? titles[j] : ("Trailer " + (j + 1)));

                getDownloadlinks(src_url, src_title, function() { requests--; });
            }

            showAdtContainer(function() {return requests});
        } else {
            // check if a "watch the trailer" link exists
            if (reBackToTrailers.test(page)) {
                showMessage(CLICK_MSG);
            }
            else {
                showError('');
            }
        }
    }

    /**
    * Scan the old style page for HD quicktime movies. Get all <a> tags with specific href's
    * (eg: http://trailers.apple.com/trailers/disney/ponyo/)
    */
    function parseOldTrailerPage() {
        var src_url, src_title, links, titles, j;

        writeLog(TITLE + ": Analyzing current page...");

        try {
            links = Array.filter($('#content').getElementsByClassName('hd'), function(elem) {
                // we're only interested in links with a href pointing to a .mov
                if (elem.nodeName !== 'A' || !elem.hasAttribute('href')) {
                    return false;
                }
                return (elem.getAttribute('href').endsWith('1080p.mov'));
            });

            titles = Array.filter($('.trailer-nav').getElementsByClassName('text'), function(elem) {
                // we're only interested in spans with class text
                return (elem.nodeName === 'SPAN');
            });
        }
        catch (err) {
            links = [];
        }

        if (links.length > 0) {
            for (j = 0; j < links.length; j++) {
                src_url = links[j].getAttribute('href').replace(".mov", "");
                src_title = (links.length == titles.length ? titles[j].textContent || titles[j].innerText : ("Trailer " + (j + 1)));

                addDownloadLink(src_url, src_title);
            }

            showAdtContainer();
        } else {
            // check if a "watch the trailer" link exists
            if ($('.back-to-trailers')) {
                showMessage(CLICK_MSG);
            }
            else {
                showError('');
            }
        }
    }

    /**
    * Try to fill the container
    */
    function showAdtContainer(checkPendingRequests)
    {
        var retryCount = 0;
        var si = setInterval(function() {
            if (!checkPendingRequests || checkPendingRequests() === 0) {
                clearInterval(si);
                fillAdtContainer();
                return;
            }

            retryCount++;
            if (retryCount > 10) {
                clearInterval(si);
                showError("Timeout occurred");
            }
        }, 500);
    }

    /**
    * Fill the already prepared container with all the trailer listings and
    * the respective toggles.
    */
    function fillAdtContainer() {
        var cont = null, i, size, size_links,
            ul = null, len, li;

        for(size in g_allLinks) {
            if (!g_allLinks.hasOwnProperty(size)) {
                continue;
            }
            size_links = g_allLinks[size];

            cont = prepContainer('atd' + size);
            ul = createElem('ul', { 'className': 'atdListing' });
            g_atdcontainer.appendChild(createToggle(size));
            li = null;
            for (i = 0; i < size_links.length; i++) {
                if (!size_links[i]) {
                    continue;
                }

                if (typeof (size_links[i]) == "string") {
                    li = createElem('li', { 'textContent': size_links[i], 'className': 'error' });
                }
                else {
                    li = createElem('li');
                    li.appendChild(size_links[i]);
                }
                ul.appendChild(li);
            }
            cont.appendChild(ul);
            g_atdcontainer.appendChild(cont);
            if (g_atdcontainer.style.display != 'table') { g_atdcontainer.style.display = 'table'; }
        }
        removeNode($('#atdLoader'));
    }

    function copyToClipboard(e) {
        if (typeof GM_setClipboard === 'undefined') {
            return;
        }

        var s = "", size, size_links,
            count = 0, i, len,
            elem = e.target,
            caption = elem.id;

        if (caption.endsWith('480p')) {
            size = "480p";
        } else if (caption.endsWith('720p')) {
            size = "720p";
        } else {
            size = "1080p";
        }

        size_links = g_allLinks[size];
        for (i = 0; i < size_links.length; i++) {
            if (typeof (size_links[i]) != "string" && size_links[i].href) {
                s += size_links[i].href + "\n";
                count++;
            }
        }

        if (s !== "") {
            GM_setClipboard(s);
            writeLog(TITLE + ": " + count + " movie link(s) copied to clipboard");
        }

        if (e.stopPropagation) {
            e.stopPropagation();
        }
        if (e.preventDefault) {
            e.preventDefault();
        }
        e.cancelBubble = true;
        e.returnValue = false;
        return false;
    }

    function showError(msg) {
        var omg = createElem('div', { 'id': 'atdError', 'textContent': ERROR_MSG, 'className': 'roundBottomCorners' });
        if (msg !== '') {
            omg.appendChild(createElem('br'));
            omg.appendChild(createElem('span', { 'textContent': msg, 'className': 'errorMsg' }));
        }
        g_atdcontainer.appendChild(omg);
        removeNode($('#atdLoader'));
    }

    function showMessage(msg) {
        var omg = createElem('div', { 'id': 'atdError', 'textContent': msg, 'className': 'roundBottomCorners' });
        g_atdcontainer.appendChild(omg);
        removeNode($('#atdLoader'));
    }

    // Prototypes ---------------------------------
    /**
    * Determine whether a string starts with a certain string.
    */
    String.prototype.startsWith = function(str) { return (this.indexOf(str) === 0); };

    /**
    * Determine whether a string ends with a certain string.
    */
    String.prototype.endsWith = function(str) { return this.indexOf(str, this.length - str.length) !== -1; };

    // case-insensitive indexOf function for arrays
    if (typeof Array.prototype.indexOfCI == 'undefined') {
        Array.prototype.indexOfCI = function(s) {
            if (s === null || (typeof s == "undefined")) {
                return -1;
            }
            for (var i = 0; i < this.length; i++) {
                if (this[i].toLowerCase() == s.toLowerCase()) {
                    return i;
                }
            }
            return -1;
        };
    }

    // Helper functions ---------------------------
    /**
    * Creates a new element.
    * @param {String} elem The element to create
    * @param {Object} attrs The new element's attributes
    * @returns {HtmlElement} The created element
    */
    function createElem(elem, attrs) {
        var newElem = document.createElement(elem), a;
        for (a in attrs) {
            if (attrs.hasOwnProperty(a)) {
                if (a === 'textContent') {
                    newElem.appendChild(document.createTextNode(attrs[a]));
                } else {
                    newElem[a] = attrs[a];
                }
            }
        }
        return newElem;
    }

    /**
    * Create the side panel header. Create a P element and add an A element to
    * quickly access this script's preferences.
    * @returns {HtmlElement}
    */
    function createAtdHeader() {
        var atdHeader = createElem('p', { 'id': 'atdHeader' }),
            atdHeaderLink = createElem('a', { 'id': 'atdHeaderLink', 'textContent': 'Apple Trailers' }),
            canvas = createElem('canvas', { 'id': 'tempCanvas', 'width': '10', height: '10', 'className': 'hidden' });

        atdHeaderLink.addEventListener('click', function() { GM_config.open(); }, false);
        atdHeader.appendChild(atdHeaderLink);
        atdHeader.appendChild(canvas);
        return atdHeader;
    }

    /**
    * Create and return the busy animation container. Processing all the
    * links can take a while if there are a lot of trailers.
    * @returns {HtmlElement}
    */
    function createLoader() {
        var ldr = createElem('div', { 'id': 'atdLoader', 'className': 'roundBottomCorners' });
        ldr.appendChild(createElem('p', { 'id': 'atdLoaderText', 'textContent': 'Gathering ...' }));
        return ldr;
    }

    /**
    * Create a DIV element which will serve as a toggle as well as
    * an indicator for the three sections (480p, 720p, 1080p).
    * Also adds an event listener (click) to the DIV element to control
    * the toggling.
    */
    function createToggle(caption) {
        var div = createElem('div', { 'id': 'atdTgl' + caption, 'textContent': caption, 'className': 'toggle', 'title': caption + ' trailers' });
        if (caption == '1080p' && GM_config.get('defaultSize') != 'Large (1080p)') { addClass(div, 'roundBottomCorners'); }
        div.addEventListener('click', toggleVisibility, false);
        div.addEventListener('contextmenu', copyToClipboard, false);
        return div;
    }

    /**
    * Prepare a container (DIV element) for the trailer listings (UL).
    * @param {String} newId The id of the container element
    * @returns {HtmlElement} The created DIV element
    */
    function prepContainer(newId) {
        var elem = createElem('div', { 'id': newId });
        if (GM_config.get('defaultSize').indexOf(newId.substr(3)) < 0) { elem.className = 'hidden'; }
        if (newId.endsWith('1080p')) { addClass(elem, 'roundBottomCorners'); }
        return elem;
    }

    /**
    * Return the final computed value of an element's CSS property.
    * @param {HtmlElement} elem The element
    * @param {String} prop The property
    * @returns {String} The final computed value
    */
    function getCssProp(elem, prop) {
        var cssProp;

        try {
            cssProp = window.getComputedStyle(elem, null).getPropertyValue(prop);
            if (cssProp) {
                cssProp.replace(/\\s+!important/gi, '');
            }
        } catch (err) {
            cssProp = null;
        }

        return cssProp;
    }

    /**
    * Return an element matching the specified selector.
    * @param {String} selector The selector
    * @param {Node} root Start looking here
    * @returns {HtmlElement|null} Search result
    */
    function $(selector, root) {
        var e = null;
        root = root || document;
        if (/^#(?!(?:[\w]+)?[ \.,\+\[~>#])/.test(selector)) {
            e = root.getElementById(selector.substring(1));
        } else {
            e = root.querySelector(selector);
        }
        return e;
    }

    /**
    * Removes a node from the DOM.
    * @param {HTMLElement} nod The node to remove
    */
    function removeNode(nod) { if (nod) { nod.parentNode.removeChild(nod); } }

    /**
    * Add a class to the className attribute.
    * @param {HtmlElement} elem The element to check
    * @param {String} cls The class to add
    */
    function addClass(elem, cls) {
        if (elem.nodeType === 1) {
            if (!elem.className) {
                elem.className = cls;
            } else {
                if (!hasClass(elem, cls)) { elem.className += " " + cls; }
            }
        }
    }

    /**
    * Remove a class from the className attribute. If it is the last attribute or
    * if cls isn't specified, the class attribute will be removed.
    * @param {HtmlElement} elem The element to change
    * @param {String} cls The class to remove
    */
    function remClass(elem, cls) {
        if (elem.nodeType === 1) {
            if (cls) {
                if (elem.className === cls) {
                    elem.removeAttribute('class');
                } else {
                    if (hasClass(elem, cls)) {
                        var cn = ' ' + elem.className + ' ';
                        cn = cn.replace(' ' + cls + ' ', '');
                        elem.className = cn.trim();
                    }
                }
            } else {
                elem.removeAttribute('class');
            }
        }
    }

    /**
    * Determine whether a className attribute has a specific class attached.
    * @param {HtmlElement} elem The element to check
    * @param {String} cls The class to look for
    * @returns {Boolean} Does the element have the class?
    */
    function hasClass(elem, cls) {
        if (!elem.className) {
            return false;
        }

        var cn = ' ' + elem.className + ' ';
        return cn.indexOf(' ' + cls + ' ') > -1;
    }

    /**
    * Event listener for showing/hiding the respective trailer listings.
    * In case of the 1080p section, create rounded bottom borders if closed.
    * @param {Event} e The event
    */
    function toggleVisibility(e) {
        var elem = e.target,
        toggleElem = $('#' + elem.id + ' + div');

        if (!hasClass(toggleElem, 'hidden')) {
            addClass(toggleElem, 'hidden');
            if (elem.id.endsWith('1080p')) { addClass(elem, 'roundBottomCorners'); }
        } else {
            remClass(toggleElem, 'hidden');
            if (elem.id.endsWith('1080p')) { remClass(elem, 'roundBottomCorners'); }
        }
    }

    /**
    * Adds a new CSS ruleset to the page. Uses GM_addStyle API; fallback in place.
    * @param {String} style Contains the CSS rules to add to the page
    */
    function addNewStyle(newStyle) {
        var node, heads;

        if (typeof GM_addStyle !== 'undefined') {
            GM_addStyle(newStyle);
        } else {
            heads = document.getElementsByTagName('head');
            if (heads.length > 0) {
                node = document.createElement('style');
                node.type = 'text/css';
                node.appendChild(document.createTextNode(newStyle));
                heads[0].appendChild(node);
            }
        }
    }

    /**
    * Travel up the DOM until the parent has the specified class and return the node.
    * @param {HtmlElement} elm The starting element
    * @param {String} cls The targeted parent has this class
    * @returns {HtmlElement} The targeted parent element
    */
    function parentUntilClassIs(elm, cls) {
        var p = elm;
        while (p.parentNode) {
            p = p.parentNode;
            if (hasClass(p, cls)) {
                break;
            }
        }
        return p;
    }

    /**
    * Determine whether n is a number or not.
    * @param {String|Number} n The string/number to check
    * @returns {Boolean} n can be interpreted as number
    */
    function isNumber(n) { return !isNaN(parseFloat(n)) && isFinite(n); }

    function writeLog(s) {
        if (window.console) {
            console.log(s);
        }
    }

    /**
    * Return the color used in Apple's trailer listing. If that's not
    * possible, assume Apple's default blue color scheme.
    * @returns {String} A CSS rule containing the trailer listing color
    */
    function getTrailerListingColor(cb) {
        var heading = $('h2', $('#trailers')), cssProp, css, m, url, img,
            validColor = function(css) {
                var valid = false, r, g, b;
                if (css && css.indexOf("rgb(") === 0) {
                    css = css.substr(4, css.length - 2);
                    css = css.split(",");
                    if (css.length == 3) {
                        r = parseInt(css[0], 10);
                        g = parseInt(css[1], 10);
                        b = parseInt(css[2], 10);
                        return !((r < 30 && g < 30 && b < 30) || (r > 230 && g > 230 && b > 230));
                    }
                }
                return false;
            };

        if (!cb) { return; }

        if (heading) {
            cssProp = getCssProp(heading, 'background-color');
        }
        if (!validColor(cssProp)) {
            heading = $('.top-wrapper');
            cssProp = getCssProp(heading, 'background-color');
        }
        if (!validColor(cssProp)) {
            heading = $('.hero', $('#backgrounds')) || $('.top-wrapper');
            if (heading) {
                try {
                    m = getCssProp(heading, 'background-image').match(/url\(([^)]+)\)/i);
                    url = m[1].replace(/"/g, "");
                    img = new Image();
                    img.onload = function(e) {
                        var rgb = new ColorFinder(function favorHue(r, g, b) {
                            return ((r < 30 && g < 30 && b < 30) || (r > 230 && g > 230 && b > 230)) ? 0 : ((Math.abs(r - g) * Math.abs(r - g) + Math.abs(r - b) * Math.abs(r - b) + Math.abs(g - b) * Math.abs(g - b)) / 65535 * 50 + 1);
                        }).getMostProminentColor(img);
                        cssProp = 'rgb(' + rgb.r + ', ' + rgb.g + ', ' + rgb.b + ')';
                        css = '#atdContainer > div.toggle {background:-moz-linear-gradient(left, ' + cssProp + ' 5%, rgba(0,0,0,.75) 85%) repeat scroll 0 0 transparent;background:-webkit-linear-gradient(left, ' + cssProp + ' 5%, rgba(0,0,0,.75) 85%) repeat scroll 0 0 transparent;background:linear-gradient(left, ' + cssProp + ' 5%, rgba(0,0,0,.75) 85%) repeat scroll 0 0 transparent;';
                        cb(css);
                    };
                    img.crossOrigin = '';
                    img.src = url;

                    return;
                }
                catch (err) {
                    writeLog(TITLE + ": Unable to determine background color: " + (err.description || err.message));
                    cssProp = getCssProp(heading, 'background-color');
                }
            }
        }
        if (!validColor(cssProp)) {
            cssProp = 'rgb(40, 60, 60)'; //cssProp = 'rgb(2, 131, 224)';
        }

        css = '#atdContainer > div.toggle {background:-moz-linear-gradient(left, ' + cssProp + ' 5%, rgba(0,0,0,.75) 85%) repeat scroll 0 0 transparent;background:-webkit-linear-gradient(left, ' + cssProp + ' 5%, rgba(0,0,0,.75) 85%) repeat scroll 0 0 transparent;background:linear-gradient(left, ' + cssProp + ' 5%, rgba(0,0,0,.75) 85%) repeat scroll 0 0 transparent;';
        cb(css);
    }

    function getAverageRGB(imgEl) {
        var blockSize = 5, // only visit every 5 pixels
            defaultRGB = 'rgb(2, 131, 224)', // for non-supporting envs
            canvas = document.getElementById("tempCanvas"),
            context = canvas.getContext && canvas.getContext('2d'),
            data, width, height,
            i = -4,
            length,
            rgb = { r: 0, g: 0, b: 0 },
            count = 0;

        if (!context) {
            return defaultRGB;
        }

        height = canvas.height = imgEl.naturalHeight || imgEl.offsetHeight || imgEl.height;
        width = canvas.width = imgEl.naturalWidth || imgEl.offsetWidth || imgEl.width;
        context.drawImage(imgEl, 0, 0);

        try {
            data = context.getImageData(0, 0, width, height);
        }
        catch (e) {
            return defaultRGB;
        }

        length = data.data.length;
        while ((i += blockSize * 4) < length) {
            ++count;
            rgb.r += data.data[i];
            rgb.g += data.data[i + 1];
            rgb.b += data.data[i + 2];
        }

        // ~~ used to floor values
        rgb.r = ~~(rgb.r / count);
        rgb.g = ~~(rgb.g / count);
        rgb.b = ~~(rgb.b / count);

        return 'rgb(' + rgb.r + ', ' + rgb.g + ', ' + rgb.b + ')';
    }

    function removeHash(s) {
        if (!s) { return s; }
        var i = s.lastIndexOf("#");
        return (i >= 0 ? s.substr(0, i) : s);
    }

    // Scriptish users don't need this line because of @noframes
    if (window.unsafeWindow && window.unsafeWindow.top !== window.unsafeWindow.self) { return; }

    // Init
    g_atdcontainer = createElem('div', { 'id': 'atdContainer' });
    g_baseUrl = removeHash(window.location.href);

    // Config
    GM_config.init({
        'id': 'GM_config',
        'title': TITLE,
        'fields': {
            'defaultSize': {
                'section': 'Created by JC2k8, Zatic, Mirzmaster and MHU',
                'type': 'select',
                'label': 'Default trailer size (open panel)',
                'default': 'Large (1080p)',
                'options': SIZES
            }
        },
        css: "#GM_config * { font-family: helvetica,arial,tahoma,myriad pro,sans-serif;}" +
             "#GM_config { background-color: rgba(0,0,0, 0.9); margin: 10% }" +
             "#GM_config_wrapper { background: #eee; padding: 1em;}" +
             "#GM_config .indent40 { margin-left: 40%;}" +
             "#GM_config .field_label { font-weight: bold; font-size: 12px; margin-right: 6px; float: left;}" +
             "#GM_config .block { display: block;}" +
             "#GM_config .saveclose_buttons { margin: 16px 10px 10px; padding: 2px 12px;}" +
             "#GM_config .reset, #GM_config .reset a, #GM_config_buttons_holder { text-align: right; color: #000;}" +
             "#GM_config .config_header { font-size: 20pt; margin: 0; padding: 0; font-weight: bold;}" +
             "#GM_config .config_desc, #GM_config .section_desc, #GM_config .reset { font-size: 9pt;}" +
             "#GM_config .center { text-align: center;}" +
             "#GM_config .section_header_holder { margin-top: 4px;}" +
             "#GM_config .config_var { margin: 0 0 4px; line-height: 22px; }" +
             "#GM_config .section_header { font-size: 13pt; background: #414141; color: #FFF; border: 1px solid #000; margin: 0 0 16px; }" +
             "#GM_config .section_desc { font-size: 9pt; background: #EFEFEF; color: #575757; border: 1px solid #CCC; margin: 0 0 6px; }"
    });

    // register menu command to access preferences
    GM_registerMenuCommand(TITLE +' Preferences...', function() { GM_config.open(); });

    // wait for the document to be fully loaded
    window.addEventListener("load", function() {
        initScript();

        setTimeout(function() {
            try {
                getBackboneTrailerPage();
            }
            catch(err) {
                writeLog(TITLE + ": ERROR: " + (err.description || err.message));
            }
        }, 250);
    }, false);
}());


// Copyright (c) 2013 Pieroxy <pieroxy@pieroxy.net>
// This work is free. You can redistribute it and/or modify it
// under the terms of the WTFPL, Version 2
// For more information see LICENSE.txt or http://www.wtfpl.net/
//
// For more information, the home page:
// http://pieroxy.net/blog/pages/color-finder/index.html
//
// Detection of the most prominent color in an image
// version 1.1.1

function ColorFinder(colorFactorCallback) {
    "use strict";

    this.callback = colorFactorCallback;
    this.getMostProminentColor = function(imgEl) {
        var rgb = null, data;
        if (!this.callback) {
            this.callback = function() { return 1; };
        }
        data = this.getImageData(imgEl);
        rgb = this.getMostProminentRGBImpl(data, 6, rgb, this.callback);
        rgb = this.getMostProminentRGBImpl(data, 4, rgb, this.callback);
        rgb = this.getMostProminentRGBImpl(data, 2, rgb, this.callback);
        rgb = this.getMostProminentRGBImpl(data, 0, rgb, this.callback);
        return rgb;
    };

    this.getImageData = function(imgEl, degrade, rgbMatch, colorFactorCallback) {

        var rgb,
            canvas = document.createElement('canvas'),
            defaultRGB = 'rgb(2, 131, 224)', // for non-supporting envs
            context = canvas.getContext && canvas.getContext('2d'),
            data, width, height, key, factor, result,
            i = -4,
            db = {},
            length, r, g, b,
            count = 0;

        if (!context) {
            return defaultRGB;
        }

        height = canvas.height = imgEl.naturalHeight || imgEl.offsetHeight || imgEl.height;
        width = canvas.width = imgEl.naturalWidth || imgEl.offsetWidth || imgEl.width;

        context.drawImage(imgEl, 0, 0);

        try {
            data = context.getImageData(0, 0, width, height);
        } catch (e) {
            /* security error, img on diff domain */
            return null;
        }

        length = data.data.length;
        factor = Math.max(1, Math.round(length / 5000));
        result = {};
        while ((i += 4 * factor) < length) {
            if (data.data[i + 3] > 32) {
                key = (data.data[i] >> degrade) + "," + (data.data[i + 1] >> degrade) + "," + (data.data[i + 2] >> degrade);
                if (!result.hasOwnProperty(key)) {
                    rgb = { r: data.data[i], g: data.data[i + 1], b: data.data[i + 2], count: 1 };
                    rgb.weight = this.callback(rgb.r, rgb.g, rgb.b);
                    if (rgb.weight <= 0) {
                        rgb.weight = 1e-10;
                    }
                    result[key] = rgb;
                } else {
                    rgb = result[key];
                    rgb.count++;
                }
            }
        }

        return result;

    };

    this.getMostProminentRGBImpl = function(pixels, degrade, rgbMatch, colorFactorCallback) {

        var rgb = { r: 0, g: 0, b: 0, count: 0, d: degrade },
            db = {}, data, i,
            pixel, pixelKey, pixelGroupKey,
            length, r, g, b, totalWeight,
            count = 0;


        for (pixelKey in pixels) {
            if (pixels.hasOwnProperty(pixelKey)) {
                pixel = pixels[pixelKey];
                totalWeight = pixel.weight * pixel.count;
                ++count;
                if (this.doesRgbMatch(rgbMatch, pixel.r, pixel.g, pixel.b)) {
                    pixelGroupKey = (pixel.r >> degrade) + "," + (pixel.g >> degrade) + "," + (pixel.b >> degrade);
                    if (db.hasOwnProperty(pixelGroupKey)) {
                        db[pixelGroupKey] += totalWeight;
                    }
                    else {
                        db[pixelGroupKey] = totalWeight;
                    }
                }
            }
        }

        for (i in db) {
            if (db.hasOwnProperty(i)) {
                data = i.split(",");
                r = data[0];
                g = data[1];
                b = data[2];
                count = db[i];

                if (count > rgb.count) {
                    rgb.count = count;
                    data = i.split(",");
                    rgb.r = r;
                    rgb.g = g;
                    rgb.b = b;
                }
            }
        }

        return rgb;

    };

    this.doesRgbMatch = function(rgb, r, g, b) {
        if (rgb === null) { return true; }
        r = r >> rgb.d;
        g = g >> rgb.d;
        b = b >> rgb.d;
        return rgb.r == r && rgb.g == g && rgb.b == b;
    };
}