TheOpenUserNut / Pandora music replay and download

// ==UserScript==
// @name         Pandora music replay and download
// @version      0.8
// @require https://code.jquery.com/jquery-3.2.1.min.js
// @author       Thesunfei
// @grant        none
// @license      MIT; https://opensource.org/licenses/MIT
// @include      http://*.pandora.com/*
// @include      https://*.pandora.com/*
// ==/UserScript==

// ==Revision History==
// 0.8 Added album name between Title and Artist for reference
// ==/Revision History==

/*jshint multistr: true */
$(function(){
    var styleele=$("<style></style>");
    styleele.html(`
#audioitems {
position:fixed;
right:100px;
top:300px;
background-color:rgba(0,0,0,.1);
z-index:1000;
width:400px;
box-sizing:border-box;
padding:20px;
cursor:move;
opacity:.5;
transition:opacity .3s;
border-radius:3px;
max-height:400px;
overflow-y:auto;
color:white;
box-shadow:1px 1px 10px rgba(255,255,255,.5);
}
#audioitems::-webkit-scrollbar {
width:5px;
}
#audioitems::-webkit-scrollbar-track {
background-color:rgba(0,0,0,.3);
}
#audioitems::-webkit-scrollbar-thumb {
background-color:rgba(255,255,255,.3);
}
#audioitems:hover {
opacity:1;
}
#audioitems audio {
width:100%;
}
.audioartist {
flex:1;
}
.audiowrap:not(:last-child) {
margin-bottom:15px;
}
.audiowrap {
display:flex;
}
.audiocloned {
flex:1;
display:none;
}
.audioinfo {
flex: 1;
display: flex;
flex-direction: column;
justify-content: space-between;
padding:5px 10px;
}
.imgwrap {
width:90px;
height:90px;
position:relative;
}
.audioimg {
width:100%;
height:100%;
object-fit:contain;
}
.audiocontrol {
position:absolute;
width:100%;
height:100%;
left:0;
top:0;
display:none;
background-position:center;
background-size:50px 50px;
background-repeat:no-repeat;
background-color:rgba(0,0,0,.2);
cursor:pointer;
opacity:.6;
transition:all .2s;
}
.audiocontrol:hover {
opacity:1;
}
.audioplay {
background-image:url();
}
.audiopause {
background-image:url();
}
.audiocontrol.audioload {
background-image:url();
animation:rotate .5s linear infinite;
-webkit-animation:rotate .5s linear infinite;
background-color:transparent;
}
@keyframes rotate {
from {transform:rotate(0)}
to {transform:rotate(360deg)}
}
.audiodownload {
background-image:url();
background-position:center;
background-repeat:no-repeat;
background-size:20px;
display:block;
width:30px;
}
.audiofns {
display:flex;
}
.audiotrack {
flex:1;
position:relative;
cursor:default;
}
.onplaying .audioplay{
display:block;
}
.onpaused .audiopause {
display:block;
}
.onloading .audioload {
display:block;
}
.audiotrack {
flex:1;
height:30px;
background-color:rgba(0,0,0,.2);
position:relative;
}
.audioposition {
height:100%;
position:absolute;
width:2px;
background-color:white;
box-shadow:0 0 3px white;
}
.audiops {
height:100%;
position:absolute;
width:1px;
background-color:white;
opacity:.5;
}
@keyframes rotate {
from {transform:rotate(0)}
to {transform:rotate(360deg)}
}
`);
    var audioitems=$("<div id='audioitems'></div>");
    var mpos={
        start:{x:0,y:0},
        offset:{x:0,y:0},
        last:{x:0,y:0},
        movable:false
    };
    audioitems.mousedown(function(){
        mpos.start.x=event.clientX;
        mpos.start.y=event.clientY;
        mpos.last.x=event.clientX;
        mpos.last.y=event.clientY;
        mpos.movable=true;
    });
    $("body").mousemove(function(){
        if (!mpos.movable) return;
        mpos.offset.x=event.clientX-mpos.last.x;
        mpos.offset.y=event.clientY-mpos.last.y;
        audioitems.css({left:"+="+mpos.offset.x,top:"+="+mpos.offset.y});
        mpos.last.x=event.clientX;
        mpos.last.y=event.clientY;
    });
    $("body").mouseup(function(){
        mpos.movable=false;
    });
    $("body").append(audioitems).append(styleele);
    window.audioURLs=[];
    Array.prototype.contains=function(v){
        var ret;
        $.each(this,function(index,item){
            if (item===v) {
                ret= [true,index];
            }
        });
        if (ret) {
            return ret;
        }else {
            return false;
        }
    };
    function getAudioURL (){
        var audios=document.querySelectorAll("body>audio");
        $.each(audios,function(index,item){
            if (!audioURLs.contains(item.src)) {
                audioURLs.push(item.src);
                //Pull the album information
                album = $("[data-qa='playing_album_name']");
                //Make sure only the current album is passed on.
                //  Only take the first in the array to avoid extras that can come in because of timing issues
                //    The additional one is from the previous song
                if(album.length > 1) {
                    album = album[0];
                }
                album = album.text();
                //Do not add album if it is advertisement
                if ($("[data-qa='mini_track_title']").text() == "Advertisement")
                {
                    album = "";
                }
                renderAudioHTML(item.src,$("[data-qa='mini_track_title']").text(),album,$("[data-qa='mini_track_artist_name']").text(),$("[data-qa='mini_track_image']")[0].src);
            }
        });
    }
    function renderAudioHTML (src,title,album,artist,image){
        var ele=$(`<div class='audiowrap onloading'>
<div class='imgwrap'>
<img class='audioimg' src='${image}'>
<div class="audiocontrol audioplay"></div>
<div class="audiocontrol audiopause"></div>
<div class="audiocontrol audioload"></div>
</div>
<div class='audioinfo'>
<div class='audiotitle'>${title}</div>
<div class='audioalbum'>${album}</div>
<div class='audioartist'>${artist}</div>
<div class="audiofns">
<div class="audiotrack">
<div class="audioposition"></div>
</div>
<a class="audiodownload" download></a>
</div>
</div>
<audio preload loop class='audiocloned'>
</audio>
</div>`);
        var totaltime=0,currenttime=0,state=0,audio=null,audiourl=null;
        $("audio",ele)[0].onloadedmetadata=function(e){
            totaltime=e.srcElement.duration;
            updateView();
        };
        $("audio",ele)[0].ontimeupdate=function(e){
            currenttime=e.target.currentTime;
            updateView();
        };
        $("audio",ele)[0].onplay=function(){
            Pandora.pauseTrack();
            $("audio",$(this).parents(".audiowrap").siblings()).each(function(index,item){item.pause();});
            state=1;
            updateView();
        };
        $("audio",ele)[0].onpause=function(e){
            state=0;
            updateView();
        };
        $(".audioplay",ele).click(function(){
            $("audio",ele)[0].play();
        });
        $(".audiopause",ele).click(function(){
            $("audio",ele)[0].pause();
        });
        $(".audiotrack",ele).click(function(e){
            $("audio",ele)[0].currentTime=e.offsetX/e.target.clientWidth*totaltime;
        });
        function getAudio () {
            var xhr=new XMLHttpRequest();
            xhr.open("get",src);
            xhr.responseType="blob";
            xhr.onreadystatechange=function(){
                if (this.status==200&&this.readyState==4) {
                    audio=this.response;
                    audiourl=URL.createObjectURL(audio);
                    $("audio",ele).prop("src",audiourl);
                    $("a",ele).prop("href",audiourl);
                }
            };
            xhr.send();
        }
        function updateView(){
            $(".audioposition",ele).css("left",currenttime/totaltime*100+"%");
            $(ele).removeClass("onplaying onpaused onloading");
            if (state===1) {
                $(ele).addClass("onpaused");
            }else if (state===0) {
                $(ele).addClass("onplaying");
            }else if (state===-1) {
                $(ele).addClass("onloading");
            }
        }
        $("#audioitems").append(ele);
        getAudio();
    }
    setInterval(getAudioURL,1000);
});