ifugu / Pandora Freemium

// ==UserScript==
// @name        Pandora Freemium
// @namespace   https://openuserjs.org/scripts/ifugu/Pandora_Freemium
// @description Download button. Hide advertisements panel. Block audio advertisements. Search buttons. Enable copying lyrics & track info. Extend play automatically. Show current track info on window title.
// @author      ifugu
// @version     3.9.1
// @include     http://*pandora.com/*
// @include     https://*pandora.com/*
// @grant       GM_getValue
// @grant       GM_setValue
// @grant       GM_deleteValue
// @grant       GM_xmlhttpRequest
// @require     http://ajax.googleapis.com/ajax/libs/jquery/1.7.2/jquery.min.js
// @history     3.9.1 Fixed problem getting station name in free Pandora.
// @history     3.9.0 Automatically skipping repeated tracks. Showing cover art as icon in window title bar.
// @history     3.8.0 Included Hal's scrolling title bar and download icon mods
// @history     3.7.5 Fixed lyrics insertion
// @history     3.7.4 Hid promotional ribbon
// @history     3.7.3 Filtering (Explicit) from filename.
// @history     3.7.2 Added ability to save station as part of filename.
// @history     3.7.1 Added setting to control insertion of missing lyrics.
// @history     3.7.0 Fixed artwork filename issue.
// @history     3.6.9 Hiding settings menu on login screen (only showing menu after user is logged in).
// @history     3.6.8 Updated settings menu link style to match latest design.
// @history     3.6.7 Preventing tracks from being automatically downloaded a second time in a single session.
// @history     3.6.6 Migrating to greasyfork.org from userscripts.org
// @history     3.6.5 Disabled audio ad blocking in all browsers. I think it was the source of interuptions in audio playback.
// @history     3.6.4 Disabled audio ad blocking in webkit (Chrome). It prevented audio ads, but then halted all audio.
// @history     3.6.3 Fixed search links.
// @history     3.6.2 Added ability to customize filename format for downloaded tracks. Requested by several users.
// @history     3.6.1 Changed file extension to CSV for track digest
// @history     3.6.0 Added ability to generate digest of tracks played (enable shortcut keys and press L). Suggestion from e56ythh
// @history     3.5.2 Prevented shortcut keys from interfering with inputs. Added shortcut key for next station. Added download button glow to indicate download shortcut working.
// @history     3.5.1 Fixed problem with forward slash character not being encoded properly which causes downloads to fail. Converting to hyphen.
// @history     3.5.0 Added keyboard shortcuts. Suggestion from e56ythh
// @history     3.4.6 Retrieving lyrics from a secondary service when they are missing. Suggestion from JamesTGriffing
// @history     3.4.5 Fixed problem with some characters not being encoded properly which caused downloads to fail
// @history     3.4.4 Added setting to disable analytics
// @history     3.4.3 Added analytics since install counts are broken on userscripts.org
// @history     3.4.2 Added option to download using window.open() for users having trouble with the download attribute, especially Firefox users waiting for blob (used to get around FF's same-origin restriction on download attribute).
// @history     3.4.1 Added support for downloading tracks automatically only when liked already. Requested by Fing3rZ
// @history     3.4.0 Added option to automatically skip video ads in free Pandora (once skip option is shown).
// @history     3.3.1 Added small delay before automatically downloading tracks to reduce load on browser at start of play and to allow artwork to load.
// @history     3.3.0 Added support for downloading tracks automatically when they start playing. Requested by Fing3rZ
// @history     3.2.0 Added support for blocking audio advertisements. Original code from Swyter
// @history     3.1.1 Removed album art overlays that prevented saving artwork by right-clicking images. Requested by mcrandello
// @history     3.1.0 Added TPB search engine and updated a few others. Added ability to customize search engine URLs through settings dialog. Thanks to Timothy Zorn.
// @history     3.0.3 Added right-click support to download button. Also adds support for download managers like DownThemAll! (when initiated by right-clicking button).
// @history     3.0.2 Fixed ad hiding in Firefox
// @history     3.0.1 Added GM_addStyle() polyfill to fix script in Firefox
// @history     3.0.0 Removed Freemium drop-down menu and replaced it with a settings dialog. Enabled settings for all features.
// @history     2.9.0 Added history to metadata. Added timestamps to auto-continue logging to help debug future issues. Changed version format to standard convention.
// @history     2.8.2 Fixed download button (Pandora made layout changes).
// @history     2.8.1 Add artwork filename as query string when downloading artwork on older browsers. You'll still need to copy and paste, but this should make it less painful.
// @history     2.8.0 Using HTML5 download attribute of anchor tags to automatically download track with correct filename. Added an option to download album artwork.
// @history     2.7.0 Minor code refactoring to make maintaining the search sites easier.
// @history     2.6.0 Fixed bug that generated garbled filenames when saving.
// ==/UserScript==

var scrollerActive = false;		//tracks if the scrolling is active or has been cancelled
var scrollerCallback = -1;		//allows the scrolling function to be cancelled
var titleScroll = "";			//temp to allow the title scroll function to work
var titleScrollIndex = 0;		//how far the title is offset for the current scrolling instance
var titleScrollWidth = -1; //will be overwritten by the getScrollWidth function
var titleScrollDelay = -1; //will be overwritten by the getScrollDelay function


/*
 * jQuery Reveal Plugin 1.0
 * www.ZURB.com
 * Copyright 2010, ZURB
 * Free to use under the MIT license.
 * http://www.opensource.org/licenses/mit-license.php
 */

(function($) {

    $.fn.reveal = function(options) {

        var defaults = {
            animation: 'fadeAndPop', //fade, fadeAndPop, none
            animationspeed: 300, //how fast animtions are
            closeonbackgroundclick: true, //if you click background will modal close?
            dismissmodalclass: 'close-reveal-modal' //the class of a button or element that will close an open modal
        };

        var options = $.extend({}, defaults, options);

        return this.each(function() {

            var modal = $(this),
                topMeasure  = parseInt(modal.css('top')),
                topOffset = modal.height() + topMeasure,
                locked = false,
                modalBG = $('.reveal-modal-bg');

            if(modalBG.length == 0) {
                modalBG = $('<div class="reveal-modal-bg" />').insertAfter(modal);
            }

            modal.bind('reveal:open', function () {
                modalBG.unbind('click.modalEvent');
                $('.' + options.dismissmodalclass).unbind('click.modalEvent');
                if(!locked) {
                    lockModal();
                    if(options.animation == "fadeAndPop") {
                        modal.css({'top': $(document).scrollTop()-topOffset, 'opacity' : 0, 'visibility' : 'visible'});
                        modalBG.fadeIn(options.animationspeed/2);
                        modal.delay(options.animationspeed/2).animate({
                            "top": $(document).scrollTop()+topMeasure + 'px',
                            "opacity" : 1
                        }, options.animationspeed,unlockModal());
                    }
                    if(options.animation == "fade") {
                        modal.css({'opacity' : 0, 'visibility' : 'visible', 'top': $(document).scrollTop()+topMeasure});
                        modalBG.fadeIn(options.animationspeed/2);
                        modal.delay(options.animationspeed/2).animate({
                            "opacity" : 1
                        }, options.animationspeed,unlockModal());
                    }
                    if(options.animation == "none") {
                        modal.css({'visibility' : 'visible', 'top':$(document).scrollTop()+topMeasure});
                        modalBG.css({"display":"block"});
                        unlockModal()
                    }
                }
                modal.unbind('reveal:open');
            });

            modal.bind('reveal:close', function () {
                if(!locked) {
                    lockModal();
                    if(options.animation == "fadeAndPop") {
                        modalBG.delay(options.animationspeed).fadeOut(options.animationspeed);
                        modal.animate({
                            "top":  $(document).scrollTop()-topOffset + 'px',
                            "opacity" : 0
                        }, options.animationspeed/2, function() {
                            modal.css({'top':topMeasure, 'opacity' : 1, 'visibility' : 'hidden'});
                            unlockModal();
                        });
                    }
                    if(options.animation == "fade") {
                        modalBG.delay(options.animationspeed).fadeOut(options.animationspeed);
                        modal.animate({
                            "opacity" : 0
                        }, options.animationspeed, function() {
                            modal.css({'opacity' : 1, 'visibility' : 'hidden', 'top' : topMeasure});
                            unlockModal();
                        });
                    }
                    if(options.animation == "none") {
                        modal.css({'visibility' : 'hidden', 'top' : topMeasure});
                        modalBG.css({'display' : 'none'});
                    }
                }
                modal.unbind('reveal:close');
            });

            modal.trigger('reveal:open')

            var closeButton = $('.' + options.dismissmodalclass).bind('click.modalEvent', function () {
                modal.trigger('reveal:close')
            });

            if(options.closeonbackgroundclick) {
                modalBG.css({"cursor":"pointer"})
                modalBG.bind('click.modalEvent', function () {
                    modal.trigger('reveal:close')
                });
            }
            $('body').keyup(function(e) {
                if(e.which===27){ modal.trigger('reveal:close'); }
            });

            function unlockModal() {
                locked = false;
            }
            function lockModal() {
                locked = true;
            }

        });
    }
})(jQuery);


// polyfill for GM_addStyle
if (typeof GM_addStyle === 'undefined') 
    GM_addStyle = function(css) {
        var head = document.getElementsByTagName('head')[0], style = document.createElement('style');
        if (!head) {return}
        style.type = 'text/css';
        try {style.innerHTML = css}
        catch(x) {style.innerText = css}
        head.appendChild(style);
    };

var tracksPlayedDigest = [],
    searches = [];

searches.push({
    alt: '4shared',
    template: 'http://search.4shared.com/q/CKADAw/1/%artist%+%song%+%album%',
    imgsrc: '%3D%3D'
});
searches.push({
    alt: 'allmusic',
    artistTemplate: 'http://allmusic.com/search/artist/%artist%',
    albumTemplate: 'http://allmusic.com/search/album/%album%',
    imgsrc: '%3D%3D'
});
searches.push({
    alt: 'Amazon',
    template: 'http://www.amazon.com/s/?search-alias=popular&field-keywords=%artist%+%song%+%album%',
    imgsrc: '%3D'
});
searches.push({
    alt: 'Bit-Torrent.bz',
    template: 'http://www.bit-torrent.bz/browse.php?c36=1&search=%artist%+%song%+%album%',
    imgsrc: ''
});
searches.push({
    alt: 'BitTorrentMonster',
    template: 'http://www.btmon.com/torrent/?f=%artist%+%song%+%album%',
    imgsrc: '%3D%3D'
});
searches.push({
    alt: 'Demonoid',
    template: 'http://www.demonoid.ph/files/?category=2&query=%artist%+%song%+%album%',
    imgsrc: '%3D%3D'
});
searches.push({
    alt: 'Discogs',
    artistTemplate: 'http://www.discogs.com/artist/%artist%',
    template: 'http://www.discogs.com/advanced_search?artist=%artist%&release_title=%album%&track=%song%',
    imgsrc: ''
});
searches.push({
    alt: 'Fenopy',
    template: 'http://fenopy.eu/?keyword=%artist%+%song%+%album%',
    imgsrc: ''
});
searches.push({
    alt: 'Google',
    template: 'https://google.com/search?q=%artist%+%song%+%album%',
    imgsrc: '%3D'
});
searches.push({
    alt: 'Google Blogspot',
    template: 'https://www.google.com/search?q=site%3Ablogspot.com+%artist%+%song%+%album%',
    imgsrc: ''
});
searches.push({
    alt: 'Google Direct Downloads',
    template: 'https://www.google.com/search?q=4shared|indomp3z|indowebster|ziddu|megaupload|rapidshare|mediafire|rayfile|sharebee|zshare|badongo|rayfile|brsbox|asianload|japanimusic|japandata+%artist%+%song%+%album%',
    imgsrc: '%3D%3D'
});
searches.push({
    alt: 'Google Guitar Tablature',
    artistSongTemplate: 'https://www.google.com/search?q=%artist%+%song%+"tablature"',
    imgsrc: ''
});
searches.push({
    alt: 'Google Lyrics',
    template: 'https://google.com/search?q=%artist%+-+%song%+lyrics',
    imgsrc: '%3D'
});
searches.push({
    alt: 'Grooveshark',
    artistTemplate: 'http://listen.grooveshark.com/#/search/artists/?query=%artist%',
    imgsrc: '%3D'
});
searches.push({
    alt: 'isoHunt',
    template: 'http://isohunt.com/torrents/?iht=2&ihq=%artist%+%song%+%album%',
    imgsrc: ''
});
searches.push({
    alt: 'iTunes',
    template: 'http://www.apple.com/search/?q=%artist%+%song%+%album%',
    imgsrc: ''
});
searches.push({
    alt: 'jpopsuki',
    artistTemplate: 'http://jpopsuki.eu/artist.php?name=%artist%',
    artistAlbumTemplate: 'http://jpopsuki.eu/torrents.php?action=advanced&artistname=%artist%&torrentname=%album%',
    imgsrc: '%3D%3D'
});
searches.push({
    alt: 'Last.fm',
    artistTemplate: 'http://www.last.fm/music/%artist%',
    artistAlbumTemplate: 'http://www.last.fm/music/%artist%/%album%',
    imgsrc: '%3D'
});
searches.push({
    alt: 'Myspace',
    template: 'http://www.myspace.com/search/music?q=%artist%+%song%+%album%',
    imgsrc: ''
});
searches.push({
    alt: 'MusicBrainz',
    artistTemplate: 'http://www.musicbrainz.org/search/textsearch.html?type=artist&limit=25&handlearguments=1&query=%artist%',
    artistAlbumTemplate: 'http://musicbrainz.org/search/textsearch.html?type=release&adv=on&handlearguments=1&query=artist:%artist%+AND+%album%',
    imgsrc: '%3D'
});
searches.push({
    alt: 'Rate Your Music',
    artistTemplate: 'http://rateyourmusic.com/search?searchtype=a&searchterm=%artist%',
    artistAlbumTemplate: 'http://rateyourmusic.com/search?searchtype=l&searchterm=%artist%+%album%',
    imgsrc: ''
});
searches.push({
    alt: 'RuTracker',
    template: 'http://rutracker.org/forum/search.php?nm=%artist%+%song%+%album%',
    imgsrc: ''
});
searches.push({
    alt: 'Spotify',
    template: 'spotify:search:%artist%+%song%+%album%',
    imgsrc: '',
    samewindow: true
});
searches.push({
    alt: 'The Pirate Bay',
    template: 'https://thepiratebay.sx/search/%artist%+-+%album%/0/7/0',
    imgsrc: ''
});
searches.push({
    alt: 'Torrent Meta Search',
    template: 'http://metasearch.torrentproject.com/#!search=%artist%+%song%+%album%',
    imgsrc: ''
});
searches.push({
    alt: 'Torrentz',
    template: 'http://torrentz.com/search?q=%artist%+%song%+%album%',
    imgsrc: '%3D%3D'
});
searches.push({
    alt: 'What',
    artistTemplate: 'http://what.cd/artist.php?artistname=%artist%',
    artistAlbumTemplate: 'http://what.cd/torrents.php?action=advanced&artistname=%artist%&torrentname=%album%',
    imgsrc: '%3D%3D'
});
searches.push({
    alt: 'Wikipedia',
    template: 'http://en.wikipedia.org/wiki/Special:Search?search=%artist%',
    imgsrc: ''
});
searches.push({
    alt: 'YouTorrent',
    template: 'http://www.youtorrent.com/tag/?q=%artist%+%song%+%album%',
    imgsrc: '%3D%3D'
});
searches.push({
    alt: 'YouTube',
    template: 'http://www.youtube.com/results?search_query=%artist%+-+%song%',
    imgsrc: ''
});


(function () {
    addStyles();
    addBodyClasses();
    toggleAdvertisementsPanel();
    removeCoverArtOverlays();
//    blockAudioAds();
    autoSkipVideoAds();
    insertSettings();
    insertSearchEngineLinks();
    toggleAllowingTrackInfoToBeCopiedToClipboard();
    extendPlayTime();
    detectSongPlayed('_');
    determineInstallCount();
    toggleKeyboardShortcuts();
})();


function addBodyClasses() {
    if (isSettingChecked('download_button'))
        $('BODY').removeClass('freemium-no-download');
    else
        $('BODY').addClass('freemium-no-download');

    if (isSettingChecked('search_buttons'))
        $('BODY').removeClass('freemium-no-search');
    else
        $('BODY').addClass('freemium-no-search');

    if (isSettingChecked('hide_ads_panel'))
        $('BODY').addClass('freemium-hide-ads');
    else
        $('BODY').removeClass('freemium-hide-ads');
}

function determineInstallCount() {
    if (!isSettingChecked('no_analytics')) {
        $('BODY').prepend('<img height="1" width="1" alt="" src="http://e0.extreme-dm.com/s9.g?login=ifugupf&amp;j=n&amp;jv=n" />');
        /*
        var _gaq = _gaq || [];
        _gaq.push(['_setAccount', 'UA-45783848-1']);
        _gaq.push(['_setDomainName', 'pandora.com']);
        _gaq.push(['_setAllowLinker', true]);
        _gaq.push(['_trackPageview']);

        (function() {
            var document = unsafeWindow.document;
            var ga = document.createElement('script'); ga.type = 'text/javascript'; ga.async = true;
            ga.src = ('https:' == document.location.protocol ? 'https://ssl' : 'http://www') + '.google-analytics.com/ga.js';
            var s = document.getElementsByTagName('script')[0]; s.parentNode.insertBefore(ga, s);
        })();
        */
        //$('BODY').append('<img height="1" width="1" alt="" src="http://www.google-analytics.com/__utm.gif?utmhn=pandora.com&utmac=UA-45783848-1" />');
    }
}

function detectSongPlayed(lastKSN) {
    var ti = getNowPlayingTrackInfo();
    if (ti.keySafeName != lastKSN) {
        allowLyricsToBeCopiedToClipboard();
        updateBrowserTitleWithTrackInfo(ti);
        updateFavIcon();
        if (isSettingChecked('skip_repeats') && hasTrackBeenListenedToThisSession(ti))
            setTimeout(function () {
                console.log('track repeat play detected - attempting to skip');
                skipCurrentTrack();
            }, 100);
        else {
            addTrackToDigest(ti, getNowPlayingStation());
            if (registerCurrentTrack(ti) && isSettingChecked('auto_download') && !hasTrackBeenDownloadedThisSession(ti))
                setTimeout(function () {
                    if (isSettingChecked('auto_download_liked_only') && !nowPlayingTrackIsLiked())
                        return;
                    downloadCurrentTrack();
                }, 2000); // delay auto download to give chance artwork to load if option to d/l artwork is enabled
        }
        removeCoverArtOverlays();
        handleMissingLyrics(ti);
        if (GM_getValue('freemium_playExtendedCount', 0) == -1) {
            console.log('Pandora Freemium: ' + (new Date()) + ' - Song played after maximum extensions (3). User must be interacting. So, we can resume extending play.');
            GM_setValue('freemium_playExtendedCount', 0);
            extendPlayTime();
        }
    }
    setTimeout(function () { detectSongPlayed(ti.keySafeName); }, 750);
}

GM_setValue('freemium_playExtendedCount', 0);
function extendPlayTime() {
    if (isSettingChecked('auto_continue')) {
        var playExtendedCount = GM_getValue('freemium_playExtendedCount', 0);
        var $stillListeningEl = $('A.still_listening');
        if ($stillListeningEl.length) {
            if (playExtendedCount > -1 && playExtendedCount < 3) {
                $stillListeningEl[0].click();
                playExtendedCount++;
                console.log('Pandora Freemium: ' + (new Date()) + ' - play extended ' + playExtendedCount + ' time(s)');
            }
            else
                playExtendedCount = -1;
            GM_setValue('freemium_playExtendedCount', playExtendedCount);
        }
        if (playExtendedCount > -1)
            setTimeout(extendPlayTime, 2000);
    }
}

function allowLyricsToBeCopiedToClipboard() {
    if (isSettingChecked('lyrics_select')) {
        $('.lyricsText')
            .removeClass('unselectable')
            .removeAttr('unselectable')
            .removeAttr('style')
            .prop('onmousedown', null)
            .prop('onclick', null)
            .prop('ondragstart', null)
            .prop('onselectstart', null)
            .prop('onmouseover', null);
    }
}

function toggleAllowingTrackInfoToBeCopiedToClipboard() {
    if (isSettingChecked('lyrics_select'))
        $('#trackInfo').removeClass('unselectable');
    else
        $('#trackInfo').addClass('unselectable');
}

function updateBrowserTitleWithTrackInfo(ti) {
	if (scrollerActive) {
		clearInterval(scrollerCallback);
		scrollerActive = false;
	}
	if (isSettingChecked('window_track_title')) {
		if (ti.song.length && ti.artist.length && ti.album.length)
			titleScroll = constructTitleText(ti);
		else {
			var station = getNowPlayingStation();
			if (station.length)
				titleScroll = station + ' on Pandora';
			else
				titleScroll = 'Pandora';
    	}
	}
	else
		titleScroll = 'Pandora';
	console.log(titleScroll);
	titleScrollIndex = 0;
	setTitleText();
}

function updateFavIcon() {
    $('link[rel="icon"]').attr('href', isSettingChecked('coverart_for_favicon') ? $('.slidesForeground div.slide').eq(1).find('img.art').attr('src') : '/favicon.ico');
}

function handleMissingLyrics(ti) {
    $('#trackDetail .item.lyrics').remove();
    if (GM_getValue('freemium_setting__insert_missing_lyrics', false)) {
        var lyricsLookupURL = 'http://api.lyricsnmusic.com/songs?api_key=c504d0fe52a2c836cdcd0b3ec4e94c&q=' + fixedEncodeURIComponent(ti.artist) + '%20' + fixedEncodeURIComponent(ti.song);
        GM_xmlhttpRequest({
            method: 'GET',
            url: lyricsLookupURL,
            onload: function (resp) {
                var matchedLyricsHtml = '';
                if (typeof resp.status != 'undefined' && resp.status == 200) {
                    var results = $.parseJSON(resp.responseText);
                    if (results.length) {
                        if (typeof results[0].snippet != 'undefined' && typeof results[0].url != 'undefined') {
                            var matchedTrack = findCorrectLyrics(results, ti.song);
                            if (matchedTrack)
                                matchedLyricsHtml = matchedTrack.snippet
                                                  + '<br /><br />'
                                                  + '<a href="' + matchedTrack.url + '" target="_blank">Full Lyrics</a>'
                                                  + ' &nbsp;|&nbsp; ';
                        }
                    }
                }
                var lyricsHtml = '<div class="item lyrics" style="display: block;"><div class="heading">Lyrics</div>'
                               + '<div class="itemContent">'
                                    + '<div class="lyricsText">';
                if (matchedLyricsHtml.length)
                    lyricsHtml      += matchedLyricsHtml;
                else
                    lyricsHtml      += 'lyrics not found'
                                    + '<br /><br />';
                    lyricsHtml      += '<a href="http://search.azlyrics.com/search.php?q=' + fixedEncodeURIComponent(ti.artist) + '+' + fixedEncodeURIComponent(ti.song) + '" target="_blank">Alternate lyrics search</a>'
                                    + '</div>'
                                + '</div>'
                                + '<div class="divider"></div></div>';
                $('#trackDetail .item.artistBio').before($(lyricsHtml));
            }
        });
    }
}

function findCorrectLyrics(lyricsResults, trackTitle) {
    for (var i = 0; i < lyricsResults.length; i++) {
        if (trackTitle.replace(/\W/g, '').toLowerCase() == lyricsResults[i].title.replace(/\W/g, '').toLowerCase())
            return lyricsResults[i];
    }
    return false;
}

function getSearchEngineInfoById(id) {
    for (var i = 0; i < searches.length; ++i)
    {
        if (searches[i].id == id)
            return searches[i];
    }
}

function toggleAdvertisementsPanel() {
    if (isSettingChecked('hide_ads_panel'))
        $('#ad_container').remove();
    else
        $('#ad_container').show();
}

function removeCoverArtOverlays() {
    $('.treatment.current').remove();
}

function blockAudioAds() {
    if (!$.browser.webkit) { // failed in Chrome (successfully prevented audio ad, but the next track would not play ... console shows admanager errors)
        try {
            var proxied = unsafeWindow.XMLHttpRequest.prototype.open;
            unsafeWindow.XMLHttpRequest.prototype.open = function(method, url)
            {
                if (url.match(/proxyAdRequest|mediaserverPublicRedirect|brokenAd/) && isSettingChecked('prevent_audio_ads'))
                {
                    console.info('Pandora Freemium: Audio advertisement blocked.', method, url);
                    this.abort();
                }
                else
                    return proxied.apply(this, [].slice.call(arguments));
            }
        }
        catch (e) {}
    }
}

function autoSkipVideoAds() {
    var $skipHolder = $('DIV.skipHolder');
    if ($skipHolder.length) {
        console.log('Pandora Freemium: Skip option on popup video advertisement detected. Attempting to click "skip".');
        $skipHolder.children('A')[0].click();
    }
    if (isSettingChecked('auto_skip_video_ads'))
        setTimeout(autoSkipVideoAds, 1000);
}

function addStyles() {
    GM_addStyle('.freemium-no-download #freemium_download_button { display: none; }'
        + '.freemium-no-search #dllinks { display: none; }'
        + '#pandoraRibbonContainer { display: none !important; }'
        + '#topnav, .skinContainer { top: 0 !important; }'
        + '.freemium-hide-ads.adSupported-layout #adLayout, .freemium-hide-ads.adSupported-layout .footerContainer { width: 800px !important; }'
        + '.freemium-hide-ads.adSupported-layout .contentContainer { width: 800px !important; float: none !important; }'
        + '#trackInfo .info DIV#dllinks { padding: 4px 10px 0 0; white-space: normal; }'
        + '#dllinks .dllink { -ms-filter: "progid:DXImageTransform.Microsoft.Alpha(Opacity=20)"; cursor: pointer; filter: alpha(opacity=20); height: 16px; margin: 5px 0 0 5px; opacity: 0.2; -webkit-transition: opacity 0.4s; -moz-transition: opacity 0.4s; -o-transition: opacity 0.4s; -ms-transition: opacity 0.4s; transition: opacity 0.4s; width: 16px; }'
        + '#dllinks:hover .dllink { -ms-filter: "progid:DXImageTransform.Microsoft.Alpha(Opacity=100)"; filter: alpha(opacity=100); opacity: 1; }'
        + '#freemium { display: inline-block; line-height: 15px; margin: -6px 4px 0; padding: 8px 10px 7px; }'
        + '#freemium:hover { background: #2D476E; color: #D6DEEA !important; }'
        + 'span.anonymousUser[style*="inline"] + span + span { display: none !important; }');
    // add styles for reveal modal
    GM_addStyle('.reveal-modal-bg{position:fixed;height:100%;width:100%;background:#000;background:rgba(0,0,0,.8);z-index:100000;display:none;top:0;left:0}.reveal-modal{visibility:hidden;top:100px;left:50%;margin-left:-300px;width:520px;background:#eee url() no-repeat -200px -80px;position:absolute;z-index:100001;padding:30px 40px 34px;-moz-border-radius:5px;-webkit-border-radius:5px;border-radius:5px;-moz-box-shadow:0 0 10px rgba(0,0,0,.4);-webkit-box-shadow:0 0 10px rgba(0,0,0,.4);-box-shadow:0 0 10px rgba(0,0,0,.4)}.reveal-modal.small{width:200px;margin-left:-140px}.reveal-modal.medium{width:400px;margin-left:-240px}.reveal-modal.large{width:600px;margin-left:-340px}.reveal-modal.xlarge{width:800px;margin-left:-440px}.reveal-modal .close-reveal-modal{font-size:22px;line-height:.5;position:absolute;top:8px;right:11px;color:#aaa;text-shadow:0 -1px 1px rbga(0,0,0,.6);font-weight:700;cursor:pointer}');
    // add styles for reveal modal content
    GM_addStyle('#freemium_settings{ color: #000; }'
        + '#freemium_settings > h2 { color: #3A5997; margin-top: 0; margin-bottom: 1.1em; font-size: 1.5em; }'
        + '#freemium_settings > div:not(.child-options) { margin-top: 10px; }'
        + '#freemium_settings > div > div:not(.child-options) { padding: 5px; margin-left: 17px; max-height: 125px; overflow-y: scroll; min-width: 200px; display: inline-block; margin-top: 6px; }'
        + '#freemium_settings label { display: block; }'
        + '#freemium_settings input + label { display: inline-block; }'
        + '#freemium_settings > div > label { font-weight: 700; }'
        + '#freemium_settings label span { font-weight: 400; }'
        + '#freemium_settings div.child-options label { padding: 7px 0 0 22px; }'
        + '#freemium_settings input[type="checkbox"] { margin-right: 9px; vertical-align: top; }'
        + '#freemium_settings input:not(:checked) + div.child-options *, #freemium_settings input:not(:checked) + label + div.child-options * { opacity: 0.5; }'
        + '#freemium_enabler label { margin-bottom: 3px; }');
}

function insertSettings() {
    // If user menu is not using all of its available horizontal space, then we'll shrink it
    // so that our new menu will not have an awkward space to the right of it.
    var $userMenu = $('#brandingBar .rightcolumn .user_menu');
    var userMenuCSSWidth = $userMenu.css('width');
    $userMenu.css('width', 'auto');
    if ($userMenu.width() > parseInt(userMenuCSSWidth))
        $userMenu.css('width', userMenuCSSWidth);

    // Add our custom menu item
    $('#brandingBar .rightcolumn').append(
        '<span style="display: inline; float: right;"><a id="freemium" href="#">Freemium</a> &nbsp; | &nbsp; </span>'
    );
    // Add the settings modal
    $('BODY').append(
        '<div id="freemium_settings" class="reveal-modal"><h2>Pandora Freemium Settings</h2>'
        + '<div><label><input type="checkbox" id="freemium_setting__keyboard_shortcuts"/>Enable keyboard shortcuts (<a id="show_shortcuts" href="#shortcuts">list shortcuts</a>)</label></div>'
        + '<div><label><input type="checkbox" id="freemium_setting__download_button"/>Show download button (<a id="edit_ftmpl" href="#edit_filename">edit filename format</a>) <span style="color: #555;">(It is strongly recommended to upgrade to an inexpensive monthly Pandora subscription to support Pandora and receive high quality audio.)</span></label></div>'
        + '<div><label title="Filenames for artwork will be incorrect for older browsers"><input type="checkbox" id="freemium_setting__download_art"/>Download artwork with track</label></div>'
        + '<div>'
        + '<input type="checkbox" id="freemium_setting__auto_download" title="Tracks will be downloaded automatically when they start playing"/><label title="Tracks will be downloaded automatically when they start playing" for="freemium_setting__auto_download">Automatically download tracks</label>'
        + '<div class="child-options"><label title="If you chose to automatically download tracks, select this option to download only tracks that have been previously liked"><input type="checkbox" id="freemium_setting__auto_download_liked_only"/>only if they are liked</label></div>'
        + '</div>'
        + '<div><label title="If downloads lock your browser up for too long or cause it to crash, use this option. Tracks and artwork might open in new tabs or windows. You will need to configure your browser to automatically download the content types."><input type="checkbox" id="freemium_setting__download_via_open"/>Use alternate download method <span>(use this option if you are having problems downloading)</span></label></div>'
        + '<div><label><input type="checkbox" id="freemium_setting__search_buttons"/>Show search buttons</label><div id="freemium_enabler"></div></div>'
        + '<div>'
		+ '<input type="checkbox" id="freemium_setting__window_track_title" title="Show current track title in tab/window"/><label title="Show current track title in tab/window" for="freemium_setting__window_track_title">Show track title in window title  (<a id="edit_ttltmpl" href="#edit_titletemplate">edit title text format</a>)</label>'
        + '<div class="child-options"><label title="Scoll the window"><input type="checkbox" id="freemium_setting__scroll_the_title"/>Scroll the track title (<a id="edit_scrWdth" href="#scroll_width">Set scroll characters</a>)  (<a id="edit_scrDly" href="#scroll_delay">Set scroll step delay</a>)</label></div>'
        + '</div>'
        + '<div><label title="Show current track\'s cover art as an icon in the tab/window (may not work in Chrome)"><input type="checkbox" id="freemium_setting__coverart_for_favicon"/>Show cover art as window icon</label></div>'
        + '<div><label title="The &quot;Are you still listening?&quot; button will be clicked automatically for you up to three times"><input type="checkbox" id="freemium_setting__auto_continue"/>Extend playing time</label></div>'
        + '<div><label title="Tracks already played during a listening session will be skipped automatically, unless the maximum number of skips for the hour has been reached."><input type="checkbox" id="freemium_setting__skip_repeats"/>Automatically skip repeated tracks</label></div>'
        + '<div><label><input type="checkbox" id="freemium_setting__hide_ads_panel"/>Hide advertisements panel <span>(may require reloading Pandora after changing this setting)</span></label></div>'
//        + '<div><label title="Audio advertisements that play regularly on free Pandora should be blocked, however this does not block video popup ads."><input type="checkbox" id="freemium_setting__prevent_audio_ads"/>Block audio advertisements <span>(experimental; disabled in Chrome)</span></label></div>'
        + '<div><label title="Popup video advertisements will be skipped automatically when the option is shown on Pandora (several seconds into the video)."><input type="checkbox" id="freemium_setting__auto_skip_video_ads"/>Automatically click skip on video advertisements <span>(experimental)</span></label></div>'
        + '<div><label title="Lyric selection will become disabled again when navigating through the track history slider. If this happens, lyric selection will be enabled again when the next track begins."><input type="checkbox" id="freemium_setting__lyrics_select"/>Enable selection/copy of lyrics</label> </div>'
        + '<div><label title="This option will attempt to insert lyrics if they are not supplied by Pandora."><input type="checkbox" id="freemium_setting__insert_missing_lyrics"/>Insert lyrics if missing</label> </div>'
        + '<div><label title="Since userscripts.org install counts are broken, an external service is used just to get a count of how many people are using this script. This count inspires my development. The info is anonymous, but if you want, you can turn it off completely."><input type="checkbox" id="freemium_setting__no_analytics"/>Don\'t send anonymous statistics</label> </div>'
        + ' <a class="close-reveal-modal">&#215;</a> </div>'
    );
    initSearchEngineSettingsControls();

    // Wire up link to change filename template
    $('#edit_ftmpl').click(filenameTemplateEditClicked);

    // Wire up links to change title bar behavior
    $('#edit_scrWdth').click(scrollWidthEditClicked);
    $('#edit_scrDly').click(scrollDelayEditClicked);
    $('#edit_ttltmpl').click(titleTemplateEditClicked);

    // Wire up menu item to open the settings modal when clicked
    $('#freemium').click(function(e) {
        e.preventDefault();
        $('#freemium_settings').reveal();
        return false;
    });

    // Wire up settings controls
    $('#freemium_settings INPUT[id^="freemium_setting__"]').each(function () { initSettingControl($(this)); });

    $('#show_shortcuts').click(function (e) {
        e.preventDefault();
        alert('Keyboard Shortcuts:\n--------------------------------------------------\n[UP] = Like\n[DOWN] = Dislike\n[P] = Pause/Play\n[T] = Tired of track\n[S] = Search for track\n[D] = Download\n[SHIFT]-[D] = Like and download\n[N] = Next station\n[L] = List digest of tracks played\n\nBuilt-in Pandora shortcuts:\n--------------------------------------------------\n[RIGHT] = Next track\n[LEFT] = View previous track in play history');
        return false;
    });
}

function isEngineDisabled(id) {
    return GM_getValue('freemium_' + id, false);
}

function searchEngineEnablerChanged() {
    var val = $(this).val(),
        disabled = isEngineDisabled(val),
        $img = $('#' + val + '_img');

    if (disabled) {
        GM_deleteValue('freemium_' + val);
        $(this).attr('checked', 'checked');
        if (isSettingChecked('search_buttons'))
            $img.fadeIn();
        else
            $img.show();
    }
    else {
        GM_setValue('freemium_' + val, true);
        $(this).removeAttr('checked');
        if (isSettingChecked('search_buttons'))
            $img.fadeOut();
        else
            $img.hide();
    }
}

function searchEngineEditClicked(e) {
    var id = $(this).parent().find('INPUT').val(),
        ssId = 'ss_' + id,
        search = getSearchEngineById(id),
        searchString = getSearchTemplate(search),
        response = prompt('Customize the search URL for ' + search.alt + ' (%artist%, %song%, and %album% will be replaced. A blank string will return the search engine to its default.):', searchString),
        newSearchString = $.trim(response);

    if (newSearchString)
        GM_setValue(ssId, newSearchString);
    else if (response != null) // must be blank
        GM_deleteValue(ssId);

    e.stopPropagation();
    return false;
}

function filenameTemplateEditClicked(e) {
    var response = prompt('Customize the filename for downloaded tracks (%artist%, %song%, %album%, and %station% will be replaced. A blank string will return the filename to its default.):', getFilenameTemplate()),
        newTemplate = $.trim(response);

    if (newTemplate)
        GM_setValue('freemium_setting__download_filename_template', newTemplate);
    else if (response != null) // must be blank
        GM_deleteValue('freemium_setting__download_filename_template');

    e.stopPropagation();
    return false;
}

function titleTemplateEditClicked(e) {
    var response = prompt('Customize the text for the window title (%artist%, %song%, %album%, and %station% will be replaced. A blank string will return the filename to its default.):', getTitleTextTemplate()),
        newTemplate = $.trim(response);

    if (newTemplate)
        GM_setValue('freemium_setting__title_text_template', newTemplate);
    else if (response != null) // must be blank
        GM_deleteValue('freemium_setting__title_text_template');

    updateBrowserTitleWithTrackInfo(getNowPlayingTrackInfo());

    e.stopPropagation();
    return false;
}

function scrollWidthEditClicked(e) {
    var response = prompt('Set the number of characters to display in the title scrolling. (Empty will reset to default).', getScrollWidth()),
        newScroll = parseInt($.trim(response));

    if (Number.isInteger(newScroll))
	{
		titleScrollWidth = newScroll;
        GM_setValue('freemium_setting__title_scroll_width', newScroll);
    }
	else if (response != null) // must be blank
    {
		titleScrollWidth = 25;
		GM_deleteValue('freemium_setting__title_scroll_width');
	}
	
    updateBrowserTitleWithTrackInfo(getNowPlayingTrackInfo());

    e.stopPropagation();
    return false;
}

function scrollDelayEditClicked(e) {
    var response = prompt('Set the delay in milliseconds per movement of the title scrolling. Less than 100 is not recommended due to the processing load. (Empty will reset to default).', getScrollDelay()),
        newScroll = parseInt($.trim(response));

    if (Number.isInteger(newScroll))
	{
		titleScrollDelay = newScroll;
        GM_setValue('freemium_setting__title_scroll_delay', newScroll);
    }
	else if (response != null) // must be blank
    {
		titleScrollDelay = 500;
		GM_deleteValue('freemium_setting__title_scroll_width');
	}
    
    updateBrowserTitleWithTrackInfo(getNowPlayingTrackInfo());

    e.stopPropagation();
    return false;
}

function isSettingChecked(sid) {
    return GM_getValue('freemium_setting__' + sid, false);
}

function initSettingControl($input) {
    var cid = $input.attr('id');

    if (GM_getValue(cid, false)) {
        $('#' + cid).attr('checked', 'checked');
    }
    $('#' + cid).click(settingOptionChanged);
}

function settingOptionChanged(e) {
    e.stopPropagation();
    var cid = $(this).attr('id');

    if ($(this).attr('checked') == 'checked') {
        GM_setValue(cid, true);

        if (cid == 'freemium_setting__auto_continue')
            extendPlayTime();
        else if (cid == 'freemium_setting__auto_skip_video_ads')
            autoSkipVideoAds();
    }
    else {
        GM_deleteValue(cid);

        if (cid == 'freemium_setting__window_track_title')
            document.title = 'Pandora';
    }

    if (cid == 'freemium_setting__hide_ads_panel')
        toggleAdvertisementsPanel();
    else if (cid == 'freemium_setting__lyrics_panel') {
        toggleAllowingTrackInfoToBeCopiedToClipboard();
        allowLyricsToBeCopiedToClipboard();
    }
    else if (cid == 'freemium_setting__keyboard_shortcuts')
        toggleKeyboardShortcuts();
    else if (cid == 'freemium_setting__scroll_the_title' || cid == 'freemium_setting__window_track_title')
        updateBrowserTitleWithTrackInfo(getNowPlayingTrackInfo());
    else if (cid == 'freemium_setting__coverart_for_favicon')
        updateFavIcon();

    addBodyClasses();
}

function initSearchEngineSettingsControls() {
    var idtxt, alttxt, disabled;
    for (var i = 0, max = searches.length; i < max; ++i)
    {
        searches[i].id = stringToID(searches[i].alt);
        idtxt = searches[i].id;
        alttxt = searches[i].alt;

        disabled = isEngineDisabled(idtxt);

        $('#freemium_enabler').append('<label><input type="checkbox" id="' + idtxt + '_en" value="' + idtxt + '"' + (disabled ? '' : ' checked="checked"') + ' />' + alttxt + ' <a href="#edit">edit</a></label>');
    }
    $('#freemium_enabler INPUT').click(searchEngineEnablerChanged);
    $('#freemium_enabler A').click(searchEngineEditClicked);
}

function insertSearchEngineLinks() {
    $('#trackInfo .info').append('<div id="dllinks" />');
    for (var i = 0; i < searches.length; ++i)
        addLink(searches[i]);
    addDownloadLink();
}

function addLink(search) {
    var $img = $('<img class="dllink" />');
    $img.attr('id', search.id + '_img');
    $img.attr('alt', search.alt);
    $img.attr('title', 'Search on ' + search.alt);
    $img.attr('src', search.imgsrc);
    if (isEngineDisabled(search.id))
        $img.css('display', 'none');
    $img.click(function() {
        var srch = getSearchEngineInfoById($(this).attr('id').replace('_img', ''));
        var srchUrl = makeUrl(search, getNowViewingTrackInfo());
        if (srch.samewindow)
            unsafeWindow.open(srchUrl);
        else
            unsafeWindow.open(srchUrl, '_blank');
    });
    $('#dllinks').append($img);
}

function addDownloadLink() {
    $('#trackInfoButtons .buttons').append(
        $('<a href="#" id="freemium_download_button" class="button btn_bg" style="width: 75px; padding: 5px 7px 6px 6px;"><img style="height: 13px; margin-right: 4px; float: left; margin-top: 2px; width: 20px;" title="Download" alt="cloud arrow download icon" src=""><div class="text" style="font-size: 11px; padding-top: 2px;">Download</div></a>')
          .click(downloadCurrentTrack)
          .mouseover(function () {
              try {
                    var trackInfo = getNowViewingTrackInfo(),
                        trackUrl = tracks[trackInfo.keySafeName].src.replace(/(.+)access.+\?(.+)/g, '$1access/' + fixedEncodeURIComponent(trackInfo.filename()) + '?$2');
              }
              catch (err) {
                    var trackUrl = '#';
              }
              $(this).attr('href', trackUrl);
          })
    );
}

function makeUrl(search, parameters, url) {
    if (!url)
        url = getSearchTemplate(search);
    for (p in parameters) {
        if (typeof parameters[p] === 'string')
            url = url.replace('%' + p + '%', fixedEncodeURIComponent(parameters[p]));
    }
    url = url.replace('%artist%+', '');
    url = url.replace('%song%+', '');
    url = url.replace('+%album%', '');
    return url;
}

function getSearchTemplate(search) {
    var searchString = $.trim(GM_getValue('ss_' + stringToID(search.alt), ''));
    return if2(searchString, if2(search.artistSongTemplate, if2(search.artistAlbumTemplate, if2(search.albumTemplate, if2(search.artistTemplate, search.template)))));
}

function if2(e1, e2) {
    return e1 ? e1 : e2;
}

function downloadNowPlayingTrack() {
    var ti = getNowPlayingTrackInfo();
    if (tracks[ti.keySafeName])
        initiateTrackDownload(ti);
}

function downloadCurrentTrack() {
    var ti = getNowViewingTrackInfo();
    if (tracks[ti.keySafeName]) // track was already registered when it began playing
        initiateTrackDownload(ti);
    else { // track wasn't properly registered when it began playing. Let's try to find the URL now
        var npti = getNowPlayingTrackInfo();
        if (ti.song == npti.song && ti.artist == npti.artist && ti.album == npti.album) {
            var npRemainingTime = runtimeToSeconds($('.progress .remainingTime').text());
            // It can take a few seconds for the player to get the song duration and we need it to determine the current track from the jPlayer media.  So, we'll keep scanning for the track time to load.
            if (!npRemainingTime) {
                alert('Please wait for the remaining track time to be determined and then try again.');
                return;
            }
            var jp = unsafeWindow.$.jPlayer,
                npLen = runtimeToSeconds($('.progress .elapsedTime').text()) + npRemainingTime,
                thisSongDiff,
                lastSongDiff = 100000,
                srcStr;
            $.each(jp.prototype.instances, function(i, el) {
                if (el.data('jPlayer').status.srcSet)
                {
                    thisSongDiff = Math.abs(npLen - el.data('jPlayer').status.duration);
                    if (thisSongDiff < lastSongDiff)
                        srcStr = el.data('jPlayer').status.src;
                    lastSongDiff = thisSongDiff;
                }
            });
            if (srcStr) {
                tracks[ti.keySafeName] = { src: srcStr, artSrc: getNowPlayingArtSrc(ti.keySafeName) };
                initiateTrackDownload(ti);
            }
            else
                alert('Unable to locate URL of current track. Allow track to play for a few seconds and try again.');
        }
        else
            alert('Unable to download this track. The URL wasn\'t registered. Try letting tracks play a little longer before skipping ahead. If this keeps happening, please report the problem.');
    }
    return false;
}

function initiateTrackDownload(trackInfo) {
    markTrackAsDownloadedInDigest(trackInfo);

    var trackUrl = tracks[trackInfo.keySafeName].src.replace(/(.+)access.+\?(.+)/g, '$1access/' + fixedEncodeURIComponent(trackInfo.filename()) + '?$2'),
        artSrc = tracks[trackInfo.keySafeName].artSrc;

    // download by opening a new window if the download attribute is not supported or if the user chose simple downloads in settings
    if (typeof document.createElement('a').download == 'undefined' || isSettingChecked('download_via_open')) {
        unsafeWindow.open(trackUrl, '_blank');
        if (artSrc && GM_getValue('freemium_setting__download_art', false))
            unsafeWindow.open(artSrc + '?filename=' + fixedEncodeURIComponent(trackInfo.artFilename), '_blank');
    }
    // download using the download attribute of an anchor
    else {
        download(trackUrl, trackInfo.filename(), isPandoraOne() ? 'audio/mpeg' : 'audio/mp4a-latm');
        if (artSrc && GM_getValue('freemium_setting__download_art', false))
            download(artSrc, trackInfo.artFilename, 'image/jpeg', true); // force XHR method for artwork since download attribute is failing to set filename in Chrome
    }
}

// download using the download attribute of an anchor
function download(src, filename, mime, forceXHR) {
    // Firefox does not honor download attribute if content is not from same origin. So we have to create a local blob.
    if ($.browser.mozilla || forceXHR) {
        GM_xmlhttpRequest({
            method: 'GET',
            url: src,
            onload: function (respDetails) {
                var binResp = customBase64Encode(respDetails.responseText);
                clickTempLink('data:' + mime + ';base64,' + binResp, filename);
            },
            overrideMimeType: 'text/plain; charset=x-user-defined'
        });
    }
    else
        clickTempLink(src, filename);
}

function clickTempLink(src, filename) {
    var firstChild = document.querySelector('BODY *'),
        tag = document.createElement('A');
    tag.href = src;
    tag.download = filename;
    tag.target = '_blank';
    firstChild.parentNode.insertBefore(tag, firstChild);
    tag.click();
    firstChild.parentNode.removeChild(tag);
}

var tracks = {};
function registerCurrentTrack(ti) {
    var foundTrackInfo = false;
    $.each(unsafeWindow.$.jPlayer.prototype.instances, function(i, el) {
        if (el.data('jPlayer').status.srcSet && !trackExists(el.data('jPlayer').status.src)) {
            tracks[ti.keySafeName] = { src: el.data('jPlayer').status.src, artSrc: getNowPlayingArtSrc(ti.keySafeName) };
            foundTrackInfo = true;
            return false; // breaks $.each() loop
        }
    });
    return foundTrackInfo;
}

function trackExists(src) {
    for (key in tracks) {
        if (tracks[key].src == src)
            return true;
    }
    return false;
}

function keySafeTrackFilename(ti) {
    var safeStr = (ti.song + ti.artist).toLowerCase();
    return '_' + safeStr.replace(/[^A-Za-z0-9]/g, '');
}

function addTrackToDigest(ti, station) {
    tracksPlayedDigest.push({ station: station, trackInfo: ti, downloaded: false });
}

function hasTrackBeenListenedToThisSession(ti) {
    var trackDigest = findTrackInDigest(ti);
    return typeof trackDigest === 'object';
}

function hasTrackBeenDownloadedThisSession(ti) {
    var trackDigest = findTrackInDigest(ti);
    return typeof trackDigest === 'object' && trackDigest.downloaded;
}

function markTrackAsDownloadedInDigest(ti) {
    var trackDigest = findTrackInDigest(ti);
    if (typeof trackDigest === 'object')
        trackDigest.downloaded = true;
}

function findTrackInDigest(ti) {
    for (var i = tracksPlayedDigest.length - 1; i > -1; i--) {
        var thisTi = tracksPlayedDigest[i].trackInfo;
        if (ti.keySafeName == thisTi.keySafeName) {
            return tracksPlayedDigest[i];
            break;
        }
    }
}

function outputDigest() {
    if (tracksPlayedDigest.length) {
        var outStr = 'Station,Album,Artist,Title,Downloaded\r\n';
        for (var i = 0, max = tracksPlayedDigest.length; i < max; i++) {
            var entry = tracksPlayedDigest[i],
                entryTi = entry.trackInfo;
            outStr += entry.station + ','
                    + entryTi.album + ','
                    + entryTi.artist + ','
                    + entryTi.song + ','
                    + (entry.downloaded ? 'yes' : 'no') + '\r\n';
        }
        clickTempLink('data:text/octet-stream;base64,' + customBase64Encode(outStr), 'pf-digest_' + getFilenameSafeTimestamp(new Date()) + '.csv');
    }
}

function getFilenameSafeTimestamp(dt) {
    return dt.getFullYear() + (dt.getMonth() + 1).pad() + dt.getDate().pad() + '-' + dt.getHours().pad() + dt.getMinutes().pad() + dt.getSeconds().pad();
}

function pad(padLength, padChar) {
    var str = '' + this;

    if (typeof padLength == 'undefined' || padLength == null || padLength == '')
        padLength = 2;
    if (typeof padChar == 'undefined' || padChar == null || padChar == '')
        padChar = '0';

    while (str.length < padLength)
        str = padChar + str;

    return str;
};
Number.prototype.pad = pad;
String.prototype.pad = pad;

function isPandoraOne() {
    return $('DIV.logosubscriber:visible').length;
}

function runtimeToSeconds(runtime) {
    try {
        runtime = $.trim(runtime.replace('-', ''));
        var timeParts = runtime.split(':');
        return (safeParseInt(timeParts[0]) * 60) + safeParseInt(timeParts[1]);
    }
    catch (err) {
        return 0;
    }
}

function safeParseInt(str) {
    var val = parseInt(str);
    return (isNaN(val) ? 0 : val);
}

function getNowViewingTrackInfo() {
    return constructTrackInfo('#trackInfo', 'A.songTitle', 'A.artistSummary', 'A.albumTitle');
}

function getNowPlayingTrackInfo() {
    return constructTrackInfo('#playerBar', 'A.playerBarSong', 'A.playerBarArtist', 'A.playerBarAlbum');
}

function constructTrackInfo(selTrackInfo, selSong, selArtist, selAlbum, artSrc) {
    var $trackInfo = $(selTrackInfo + ':first'),
        trackInfo = new Object();
    trackInfo.station = $.trim(getNowPlayingStation());
    trackInfo.song = $.trim(filterAddedTitleInfo($trackInfo.find(selSong + ':first').text()));
    trackInfo.artist = $.trim($trackInfo.find(selArtist + ':first').text());
    trackInfo.album = $.trim(filterAddedTitleInfo($trackInfo.find(selAlbum + ':first').text()));
    trackInfo.keySafeName = keySafeTrackFilename(trackInfo);
    // TODO: use el.data('jPlayer').status.formatType for filename instead (test with free Pandora first ... m4a)
    trackInfo.filename = function () {
        return constructFilename(trackInfo) + (isPandoraOne() ? '.mp3' : '.m4a');
    };
    trackInfo.artFilename = constructFilename(trackInfo) + '.jpg';
    return trackInfo;
}

// remove terms added by Pandora like (Explicit) from titles
function filterAddedTitleInfo(str) {
	return str.replace(/\(Explicit\)/gi, '').replace(/\s{2,}/g, ' ');
}

function constructFilename(trackInfo) {
    return getFilenameTemplate()
                .replace(/%artist%/g, trackInfo.artist)
                .replace(/%song%/g, trackInfo.song)
                .replace(/%album%/g, trackInfo.album)
                .replace(/%station%/g, trackInfo.station.replace(/ \/ /g, '-').replace(/\//g, '-'))
                .replace('/', '');
}

function getFilenameTemplate() {
    return GM_getValue('freemium_setting__download_filename_template', '%artist% - %album% - %song%');
}

function constructTitleText(trackInfo) {
    return getTitleTextTemplate()
                .replace(/%artist%/g, trackInfo.artist)
                .replace(/%song%/g, trackInfo.song)
                .replace(/%album%/g, trackInfo.album)
                .replace(/%station%/g, trackInfo.station.replace(/ \/ /g, '-').replace(/\//g, '-'))
                .replace('/', '');
}

function getTitleTextTemplate() {
    return GM_getValue('freemium_setting__title_text_template', '%song% by %artist% on %album%');
}

function getScrollWidth() {
    if(titleScrollWidth == -1) titleScrollWidth = parseInt(GM_getValue('freemium_setting__title_scroll_width', 25));
	return titleScrollWidth;
}

function getScrollDelay() {
	if(titleScrollDelay == -1) titleScrollDelay = parseInt(GM_getValue('freemium_setting__title_scroll_delay', 500));
	return titleScrollDelay;
}


// Reliably getting the album art has turned out to be tricky. When a track first starts playing, the text
// describing the track gets updated, but there is sometimes a delay before the image source is updated.
// This results in the album art for the previous track being associated with the current track. To solve the
// problem, I'm going to wait until the album art changes by comparing the current source with the last one.
// It's not airtight, but hopefully it will work most of the time.
var lastArtSrc = null; // seed the last art source value so there's a difference when the first art appears
function getNowPlayingArtSrc(keySafeName, callCount) {
    // since we're going to keep running this routine to check for the source change, we need to know how
    // many times we've checked. If this is the first time running, then we can let the caller register
    // the art source. We'll also use the count to limit how long we'll look for the album art, in case of
    // failure.
    callCount || (callCount = 0);

    // fetch the art album source
    var src = null,
        $img = $('IMG.playerBarArt:first');
    if ($img.length) {
        src = $.trim($img.attr('src'));
        if (src.indexOf('no_album_art') != -1)
            src = null;
    }

    if (src != lastArtSrc) { // detect new album art shown
        lastArtSrc = src;
        if (callCount > 0) // for increased reliability, we'll only update the tracks register if getNowPlayingArtSrc() called itself
            tracks[keySafeName].artSrc = src;
    }
    else {
        if (callCount == 0) // if the old art source is still showing, we want to return null to the initial caller
            src = null;
        if (callCount < 33) // stop checking after 5 seconds, assume failure
            setTimeout(function () { getNowPlayingArtSrc(keySafeName, ++callCount); }, 150); // wait a little while and then check again for the updated art source
    }

    return src; // the return value is only useful for the initial caller (example: registerCurrentTrack()) to initialize artSrc to null or, if art source is already updated, the actual src
}

function getNowPlayingStation() {
    return $.trim($('.stationChangeSelector DIV.textWithArrow P, .stationChangeSelectorNoMenu P').first().attr('title'));
}

function nowPlayingTrackIsLiked() {
    return $('#playbackControl .buttons .thumbUpButton.indicator').length > 0;
}

function searchForMusic(parameters) {
    var srch = getSearchEngineInfoById(getAdvancedSearchEngineId());

    var url;
    var srchParams = new Object();
    var ti = getNowViewingTrackInfo();
    switch (parameters)
    {
        case 'artist+song+album':
            url = srch.template;
            srchParams.song = ti.song;
            srchParams.artist = ti.artist;
            srchParams.album = ti.album;
            break;
        case 'artist+song':
            url = if2(srch.artistSongTemplate, srch.template);
            srchParams.song = ti.song;
            srchParams.artist = ti.artist;
            break;
        case 'artist+album':
            url = if2(srch.artistAlbumTemplate, srch.template);
            srchParams.artist = ti.artist;
            srchParams.album = ti.album;
            break;
        case 'song':
            url = if2(srch.songTemplate, srch.template);
            srchParams.song = ti.song;
            break;
        case 'album':
            url = if2(srch.albumTemplate, srch.template);
            srchParams.album = ti.album;
            break;
        default:
            url = if2(srch.artistTemplate, srch.template);
            srchParams.artist = ti.artist;
    }
    var srchUrl = makeUrl(srch, srchParams, url);

    if (srch.samewindow)
        unsafeWindow.open(srchUrl);
    else
        unsafeWindow.open(srchUrl, '_blank');

    return false;
}

function stringToID(unsafeString) {
    return '_id_' + unsafeString.replace(/ |\.|-|_/g, '');
}

function getSearchEngineById(id) {
    for (var i = 0, max = searches.length; i < max; ++i)
    {
        var thisId = stringToID(searches[i].alt);
        if (thisId == id)
            return searches[i];
    }
}

function toggleKeyboardShortcuts() {
    if (isSettingChecked('keyboard_shortcuts'))
        $(document).bind('keydown.pf', pfKeydown);
    else
        $(document).unbind('keydown.pf');
}

function pfKeydown(evt) {
    var $target = $(evt.target);
    var kc = evt.keyCode;

    // console.log('pf: ' + kc);

    if ($('INPUT:focus, SELECT:focus, TEXTAREA:focus').length) // disable hotkeys if focused on a form control
        return;


    if (kc == 80) { // P: Pause/Play
        evt.preventDefault();
        pausePlay();
    }
    else if (kc == 38) { // Up Arrow: Like
        evt.preventDefault();
        likeCurrentTrack();
    }
    else if (kc == 40) { // Down Arrow: Dislike
        evt.preventDefault();
        dislikeCurrentTrack();
    }
    else if (kc == 68) { // D: Download
        evt.preventDefault();
        if (evt.shiftKey)
            likeCurrentTrack();
        $('#freemium_download_button').css('box-shadow', '0 0 4px #FC0');
        downloadNowPlayingTrack();
        setTimeout(function () { $('#freemium_download_button').css('box-shadow', 'none') }, 3500);
    }
    else if (kc == 83) { // S: Search
        evt.preventDefault();
        $('DIV.searchBox INPUT.searchInput')[0].click();
    }
    else if (kc == 84) { // T: Tired of track
        evt.preventDefault();
        tiredOfCurrentTrack();
    }
    else if (kc == 78) { // N: Next station
        evt.preventDefault();
        playNextStation();
    }
    else if (kc == 76) { // L: List digest of tracks played
        evt.preventDefault();
        outputDigest();
    }
}

function playNextStation() {
    var $stations = $('#stationList .stationListItem');
    if ($stations.length > 2) { // check to make sure there are more than 2 station items (shuffle is a station item and one station ... need one more in order to have something to switch to)
        var $currentStation = $stations.filter('.selected');
        if ($currentStation.is(':last-child')) // determine if selected station is last station
            $stations.first().next().click(); // if last station, select first station (second station item)
        else
            $currentStation.next().click(); // if not last station, select next station
    }
}

function pausePlay() {
    $('DIV.playButton:visible A, DIV.pauseButton:visible A')[0].click();
}

function likeCurrentTrack() {
    $('DIV.thumbUpButton:visible A')[0].click();
}
function dislikeCurrentTrack() {
    $('DIV.thumbDownButton:visible A')[0].click();
}

function tiredOfCurrentTrack() {
    $('#track_menu_dd').show();
    $('#track_menu_dd').css('visibility', 'visible');
    $('A.tiredOfSong')[0].click();
    $('#track_menu_dd').hide();
    $('#track_menu_dd').css('visibility', 'hidden');
}

function skipCurrentTrack() {
    $('.skipButton A')[0].click();
}

// http://emilsblog.lerch.org/2009/07/javascript-hacks-using-xhr-to-load.html
function customBase64Encode(inputStr) {
    var bbLen = 3,
        enCharLen = 4,
        inpLen = inputStr.length,
        inx = 0,
        jnx,
        keyStr = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz'
            + '0123456789+/=',
        output = '',
        paddingBytes = 0,
        bytebuffer = new Array(bbLen),
        encodedCharIndexes = new Array(enCharLen);

    while (inx < inpLen) {
        for (jnx = 0; jnx < bbLen; ++jnx) {
            // Throw away high-order byte, as documented at:
            // https://developer.mozilla.org/En/Using_XMLHttpRequest#Handling_binary_data
            if (inx < inpLen)
                bytebuffer[jnx] = inputStr.charCodeAt(inx++) & 0xff;
            else
                bytebuffer[jnx] = 0;
        }

        /*--- Get each encoded character, 6 bits at a time.
            index 0: first 6 bits
            index 1: second 6 bits
                (2 least significant bits from inputStr byte 1
                + 4 most significant bits from byte 2)
            index 2: third 6 bits
                (4 least significant bits from inputStr byte 2
                + 2 most significant bits from byte 3)
            index 3: forth 6 bits (6 least significant bits from inputStr byte 3)
        */
        encodedCharIndexes[0] = bytebuffer[0] >> 2;
        encodedCharIndexes[1] = ((bytebuffer[0] & 0x3) << 4) | (bytebuffer[1] >> 4);
        encodedCharIndexes[2] = ((bytebuffer[1] & 0x0f) << 2) | (bytebuffer[2] >> 6);
        encodedCharIndexes[3] = bytebuffer[2] & 0x3f;

        //--- Determine whether padding happened, and adjust accordingly.
        paddingBytes = inx - (inpLen - 1);
        switch (paddingBytes) {
            case 1:
                // Set last character to padding char
                encodedCharIndexes[3] = 64;
                break;
            case 2:
                // Set last 2 characters to padding char
                encodedCharIndexes[3] = 64;
                encodedCharIndexes[2] = 64;
                break;
            default:
                // No padding - proceed
                break;
        }

        // Now grab each appropriate character out of our keystring,
        // based on our index array and append it to the output string.
        for (jnx = 0; jnx < enCharLen; ++jnx)
            output += keyStr.charAt(encodedCharIndexes[jnx]);
    }

    return output;
}

function fixedEncodeURIComponent(str) {
    return encodeURIComponent(str.replace(/\//g, '-')).replace(/[!'().*~]/g, encodeURIChar);
}

function encodeURIChar(str) {
    return '%' + str.charCodeAt(0).toString(16).toUpperCase();
}

function setTitleText() {
	//If title is longer than display, increment it one next time
	if (titleScroll.length > getScrollWidth() && isSettingChecked('scroll_the_title')) {
		var titleScrolldub = titleScroll + ' | ' + titleScroll;
		var displaytitle = titleScrolldub.substr(titleScrollIndex, getScrollWidth());
    
		//Skip spaces on the far left, as they are not displayed anyways and it looks jumpy
		while (displaytitle.substr(0,1) == ' ') {
			titleScrollIndex++;
			displaytitle = titleScrolldub.substr(titleScrollIndex, getScrollWidth());
		}

		titleScrollIndex++;
		if (titleScrollIndex >= (titleScroll.length + 3)) {
			titleScrollIndex = 0;
		}
		displaytitle = displaytitle+'...';
		if(!scrollerActive) {
			scrollerCallback = setInterval(setTitleText,getScrollDelay());
			scrollerActive = true;
		}
	} else {
		displaytitle = titleScroll;
		if(scrollerActive) clearInterval(scrollerCallback);
		scrollerActive = false;
	}
	
	document.title = displaytitle;
 
}