// ==UserScript==
// @name Pandora music replay and download
// @namespace https://greasyfork.org/users/35010
// @version 1.3.3
// @description Gives pandora music replay and download function!
// @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
// @include http://*.pandora.com/*
// @include https://*.pandora.com/*
// @license MIT
// ==/UserScript==
/*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));
}
});
});