NOTICE: By continued use of this site you understand and agree to the binding Terms of Service and Privacy Policy.
// ==UserScript== // @name Pandora music replay and download // @version 1.3.3 // @require https://code.jquery.com/jquery-3.2.1.min.js // @require https://cdn.jsdelivr.net/npm/vue@2.6.14 // @author Thesunfei // @grant none // @license MIT; https://opensource.org/licenses/MIT // @include http://*.pandora.com/* // @include https://*.pandora.com/* // ==/UserScript== // ==Revision History== // 1.0 Added playmode switch and now-playing title // 0.8 Added album name between Title and Artist for reference // ==/Revision History== /*jshint multistr: true */ if (self !== top) return; $(function () { var styleele = $("<style></style>"); styleele.html( ` #audioitems { position:fixed; right:100px; top:10%; background-color:rgba(0,0,0,.3); z-index:1000; width:400px; box-sizing:border-box; padding:0 20px 20px 20px; cursor:move; opacity:.5; transition:opacity .5s,box-shadow .5s; border-radius:3px; max-height:80%; overflow-y:auto; color:white; box-shadow:1px 1px 2px rgba(0,0,0,.2); display:flex; flex-direction:column; user-select:none; } #audiolist { flex:1; overflow-y:auto; position:relative; } #audiolist::-webkit-scrollbar { width:5px; } #audiolist::-webkit-scrollbar-track { background-color:rgba(0,0,0,.3); } #audiolist::-webkit-scrollbar-thumb { background-color:rgba(255,255,255,.3); } #audioitems:hover { opacity:1; box-shadow:2px 2px 15px rgba(0,0,0,.4); } #audioitems audio { width:100%; } .audioalbum { overflow:hidden; text-overflow:ellipsis; white-space:nowrap; } .audioartist { flex:1; overflow: hidden; text-overflow: ellipsis; white-space: nowrap; } .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:0 10px; font-size:13px; overflow:hidden; } .audiotitle { font-weight:bold; overflow:hidden; white-space:nowrap; text-overflow:ellipsis; } .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-color:transparent; justify-content:center; align-items:center; } .loading .audiocontrol.audioload:after { content:""; display:block; width:50px; height:50px; border-radius:50%; box-sizing:border-box; border-left:4px solid white; border-right:4px solid rgba(255,255,255,.3); border-top:4px solid rgba(255,255,255,.3); border-bottom:4px solid rgba(255,255,255,.3); animation:rotate .6s linear infinite; -webkit-animation:rotate .6s linear infinite; } @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; } .playing .audiopause{ display:block; } .paused .audioplay { display:block; } .loading .audioload { display:flex; } .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; transition:all .2s; } .audiops { height:100%; position:absolute; width:1px; background-color:white; opacity:.5; } #topinfo { display:flex; align-items:center; flex-shrink:0; padding:5px 0; } #playmode { border:none; background-color:transparent; background-position:center; background-repeat:no-repeat; background-size:contain; width:40px; height:40px; margin-right:10px; outline:none; opacity:.6; transition:all .2s; } #playmode:hover { opacity:1; } #playmode.loop { background-image:url(); background-size:30px; } #playmode.repeat { background-image:url(); background-size:30px; } #playmode.shuffle { background-image:url(); background-size:30px; } #playingtitle { margin:0; flex:1; overflow:hidden; text-overflow:ellipsis; white-space:nowrap; font-size:15px; padding:0; cursor:pointer; } #playingtitle:hover { text-decoration:underline; } ` ); var audioitems = $( ` <div id='audioitems' :style="{left:this.mpos.left+'px',top:this.mpos.top+'px'}" @mousedown="mDown"> <div id="topinfo"> <button id="playmode" :class="playmode" @click="playModeSwitch"></button> <h3 id="playingtitle" @click="showPlayingItem">{{currentaudioobj.title?"Now Playing : "+currentaudioobj.title:""}}</h3> </div> <div id="audiolist"> <div :class="['audiowrap',i.status]" v-for="i in audioobjs"> <div class='imgwrap'> <img class='audioimg' :src='i.image'> <div class="audiocontrol audioplay" @click="currentaudioobj.domobj&¤taudioobj.domobj.pause();i.domobj.play()"></div> <div class="audiocontrol audiopause" @click="i.domobj.pause()"></div> <div class="audiocontrol audioload"></div> </div> <div class='audioinfo'> <div class='audiotitle' :title="i.title">{{i.title}}</div> <div class='audioalbum' :title="i.album">{{i.album}}</div> <div class='audioartist' :title="i.artist">{{i.artist}}</div> <div class="audiofns" v-if="i.src"> <div class="audiotrack" @click="i.domobj.currentTime=$event.offsetX/$event.target.clientWidth*i.totaltime"> <div class="audioposition" :style="{left:i.currenttime/i.totaltime*100+'%'}"></div> </div> <a class="audiodownload" :href="i.src" :download="getFormatedSongFilename(i)"></a> </div> </div> <audio preload class='audiocloned' :src="i.src" v-if="i.src" v-bindele="i" @loadedmetadata="i.totaltime=$event.srcElement.duration;i.status='paused'" @timeupdate="i.currenttime=$event.target.currentTime" @play="pausePandora();i.status='playing';currentaudioobj=i;" @pause="currentaudioobj={};i.status='paused'" @ended="i.status='paused';playNext(i)"></audio> </div> </div> </div> ` ); $("body").append(styleele).append(audioitems); Vue.directive("bindele",{ bind:function(el,binding){ binding.value.domobj=el; } }); window.vm = new Vue({ el: "#audioitems", data: { playmode: "loop", mpos: { start: { x: 0, y: 0 }, offset: { x: 0, y: 0 }, last: { x: 0, y: 0 }, movable: false, left:parseFloat(getComputedStyle($("#audioitems")[0]).left.replace("px","")), top:parseFloat(getComputedStyle($("#audioitems")[0]).top.replace("px","")) }, audiourls: [], audioobjs: [], currentaudioobj: {}, pandora:window.Pandora }, methods: { mDown:function(e){ this.mpos.start.x = e.clientX; this.mpos.start.y = e.clientY; this.mpos.last.x = e.clientX; this.mpos.last.y = e.clientY; this.mpos.movable = true; }, mMove:function(e){ if (!this.mpos.movable) return; this.mpos.offset.x = e.clientX - this.mpos.last.x; this.mpos.offset.y = e.clientY - this.mpos.last.y; this.mpos.left+=this.mpos.offset.x; this.mpos.top+=this.mpos.offset.y; this.mpos.last.x = e.clientX; this.mpos.last.y = e.clientY; }, mUp:function(){ this.mpos.movable = false; }, getAlbum: function () { //Pull the album information var 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.first().text(); } else if (album.length == 1) { album = album.text(); } else { album = ""; } return album; }, getAudioURL: function () { var audios = document.querySelectorAll("body>audio"), that = this; $.each(audios, function (index, item) { if (that.audiourls.indexOf(item.src) == -1) { that.audiourls.push(item.src); } }); }, getAudio: function (audioobj) { var that=this; var xhr = new XMLHttpRequest(); xhr.open("get", audioobj.httpsrc); xhr.responseType = "blob"; xhr.onreadystatechange = function () { if (this.status == 200 && this.readyState == 4) { audio = this.response; //Get the url of the audio object audiourl = URL.createObjectURL(audio); //Set the audio element with the url to get it audioobj.src = audiourl; } else if (this.status != 200) { that.audioobjs.splice(that.audioobjs.indexOf(audioobj),1); } }; xhr.send(); }, pausePandora:function(){ window.Pandora?Pandora.pauseTrack():null; }, playNext: function (last) { switch (this.playmode) { case "repeat": last.domobj.play(); break; case "shuffle": this.audioobjs.filter(function(v){return v!=last})[Math.round(Math.random() * (this.audioobjs.length - 2))].domobj.play(); break; case "loop": this.audioobjs.indexOf(last) == this.audioobjs.length - 1 ? (this.audioobjs[0].domobj.play()) : (this.audioobjs[this.audioobjs.indexOf(last) + 1].domobj.play()); break; } }, getFormatedSongFilename: function (obj) { //What separates artist, album, and track in the filename var downloadElementSeparator = " - "; //Include a spot for an album, if missing, in the download filename. var includeAlbumPlaceholder = true; var filename = this.sanitizeString(downloadElementSeparator, obj.artist) + downloadElementSeparator; //Add the artist if (obj.album) { //See if we have an album to add filename = filename + this.sanitizeString(downloadElementSeparator, obj.album) + downloadElementSeparator; // Album object exists so add it } else if (includeAlbumPlaceholder == true) { // Album object does not exist, see if we need to add an album placeholder filename = filename + downloadElementSeparator; // Add album placeholder by just adding another separator } filename = filename + this.sanitizeString(downloadElementSeparator, obj.title) + ".m4a"; //Add title and extension return filename; }, sanitizeString: function (downloadElementSeparator, dirtyString) { //Remove any illegal characters based on the operating system. dirtyString = dirtyString.replace(/[*?"|]/g, ""); //windows filename restrictions -> replace with space * ? | dirtyString = dirtyString.replace(/["]/g, "''"); //windows filename restrictions -> replace with '' " dirtyString = dirtyString.replace(/[<>]/g, "_"); //windows filename restrictions -> replace with underscore < > dirtyString = dirtyString.replace(/[\\\/]/g, ","); //windows filename restrictions -> replace with comma \ / dirtyString = dirtyString.replace(/[:]/g, ";"); //windows filename restrictions -> replace with semicolon : var sepRegEx = new RegExp(downloadElementSeparator, "g"); //create RegExp object to find downloadElementSeparator dirtyString = dirtyString.replace(sepRegEx, "-"); //downloadElementSeparator -> replace with dash - return dirtyString; }, playModeSwitch: function () { switch (this.playmode) { case "loop": this.playmode = "repeat"; break; case "repeat": this.playmode = "shuffle"; break; case "shuffle": this.playmode = "loop"; } }, showPlayingItem:function(){ $("#audiolist").animate({scrollTop:this.currentaudioobj.domobj.parentElement.offsetTop},500); } }, watch: { audiourls: function () { var httpsrc = this.audiourls[this.audiourls.length - 1], that = this, audioobj = { domobj: null, title: $("[data-qa='mini_track_title']").text(), album: that.getAlbum(), artist: $("[data-qa='mini_track_artist_name']").text(), image: $("[data-qa='mini_track_image']").prop("src"), httpsrc: httpsrc, src: "", currenttime: 0, totaltime: 0, status: "loading" }; if (audioobj.title=="Advertisement") return; this.audioobjs.push(audioobj); this.getAudio(audioobj); } }, mounted: function () { setInterval(this.getAudioURL, 1000); $("body").on("mousemove",function(e){ this.mMove(e); }.bind(this)); $("body").on("mouseup",function(e){ this.mUp(e); }.bind(this)); } }); });