1434 lines
64 KiB
JavaScript
1434 lines
64 KiB
JavaScript
Vue.use(VueObserveVisibility);
|
|
|
|
|
|
Vue.component('sidebar-library-item', {
|
|
template: '#sidebar-library-item',
|
|
props: ['name', 'page', 'cd-click'],
|
|
methods: {}
|
|
});
|
|
|
|
Vue.component('lyrics-view', {
|
|
template: '#lyrics-view',
|
|
props: ["time", "lyrics", "translation"],
|
|
methods: {}
|
|
});
|
|
|
|
Vue.component('cider-listen-now', {
|
|
template: "#cider-listen-now",
|
|
props: ["data"]
|
|
})
|
|
|
|
const MusicKitTools = {
|
|
getHeader() {
|
|
return new Headers({
|
|
Authorization: 'Bearer ' + MusicKit.getInstance().developerToken,
|
|
Accept: 'application/json',
|
|
'Content-Type': 'application/json',
|
|
'Music-User-Token': '' + MusicKit.getInstance().musicUserToken
|
|
});
|
|
}
|
|
}
|
|
|
|
// limit an array to a certain number of items
|
|
Array.prototype.limit = function (n) {
|
|
return this.slice(0, n);
|
|
};
|
|
|
|
function msToMinSec(ms) {
|
|
var minutes = Math.floor(ms / 60000);
|
|
var seconds = ((ms % 60000) / 1000).toFixed(0);
|
|
return minutes + ":" + (seconds < 10 ? '0' : '') + seconds;
|
|
}
|
|
|
|
class NavigationEvent {
|
|
constructor(page, onnavigate, scrollPosition) {
|
|
this.page = page;
|
|
this.onnavigate = onnavigate;
|
|
this.scrollPosition = scrollPosition;
|
|
}
|
|
|
|
navigate() {
|
|
this.onnavigate();
|
|
document.querySelector("#app-content").scrollTop = this.scrollPosition;
|
|
}
|
|
}
|
|
|
|
const app = new Vue({
|
|
el: "#app",
|
|
data: {
|
|
drawertest: false,
|
|
mk: {},
|
|
quickPlayQuery: "",
|
|
search: {
|
|
term: "",
|
|
results: {},
|
|
limit: 10
|
|
},
|
|
playerLCD: {
|
|
playbackDuration: 0
|
|
},
|
|
listennow: [],
|
|
radio: {
|
|
personal: []
|
|
},
|
|
webview: {
|
|
url: "",
|
|
title: "",
|
|
loading: false
|
|
},
|
|
showingPlaylist: [],
|
|
artistPage: {
|
|
data: {},
|
|
},
|
|
library: {
|
|
downloadNotification: {
|
|
show: false,
|
|
message: "",
|
|
total: 0,
|
|
progress: 0
|
|
},
|
|
songs: {
|
|
sortingOptions: {
|
|
"albumName": "Album",
|
|
"artistName": "Artist",
|
|
"name": "Name",
|
|
"genre": "Genre",
|
|
"releaseDate": "Release Date",
|
|
"durationInMillis": "Duration"
|
|
},
|
|
sorting: "name",
|
|
sortOrder: "asc",
|
|
listing: [],
|
|
meta: {total: 0, progress: 0},
|
|
search: "",
|
|
displayListing: [],
|
|
downloadState: 0 // 0 = not started, 1 = in progress, 2 = complete, 3 = empty library
|
|
},
|
|
albums: {
|
|
sortingOptions: {
|
|
"artistName": "Artist",
|
|
"name": "Name",
|
|
"genre": "Genre",
|
|
"releaseDate": "Release Date"
|
|
},
|
|
viewAs: 'covers',
|
|
sorting: "name",
|
|
sortOrder: "asc",
|
|
listing: [],
|
|
meta: {total: 0, progress: 0},
|
|
search: "",
|
|
displayListing: [],
|
|
downloadState: 0 // 0 = not started, 1 = in progress, 2 = complete, 3 = empty library
|
|
},
|
|
},
|
|
playlists: {
|
|
listing: [],
|
|
details: {}
|
|
},
|
|
mxmtoken: "",
|
|
lyricon: false,
|
|
lyrics: [],
|
|
currentLyricsLine: 0,
|
|
lyriccurrenttime: 0,
|
|
lyricsMediaItem: {},
|
|
lyricsDebug: {
|
|
current: 0,
|
|
start: 0,
|
|
end: 0
|
|
},
|
|
chrome: {
|
|
hideUserInfo: false,
|
|
artworkReady: false,
|
|
userinfo: {},
|
|
menuOpened: false,
|
|
maximized: false,
|
|
drawerOpened: false,
|
|
drawerState: "queue",
|
|
topChromeVisible: true
|
|
},
|
|
page: "artist-page",
|
|
pageHistory: [],
|
|
songstest: false
|
|
},
|
|
methods: {
|
|
async init() {
|
|
let self = this
|
|
this.mk = MusicKit.getInstance()
|
|
this.mk.authorize()
|
|
this.$forceUpdate()
|
|
this.mk.privateEnabled = true
|
|
// Set profile name
|
|
this.chrome.userinfo = await this.mkapi("personalSocialProfile", false, "")
|
|
|
|
// load cached library
|
|
if (localStorage.getItem("librarySongs") != null) {
|
|
this.library.songs.listing = JSON.parse(localStorage.getItem("librarySongs"))
|
|
this.library.songs.displayListing = this.library.songs.listing
|
|
}
|
|
if (localStorage.getItem("libraryAlbums") != null) {
|
|
this.library.albums.listing = JSON.parse(localStorage.getItem("libraryAlbums"))
|
|
this.library.albums.displayListing = this.library.albums.listing
|
|
}
|
|
|
|
MusicKit.getInstance().videoContainerElement = document.getElementById("apple-music-video-player")
|
|
|
|
this.mk.addEventListener(MusicKit.Events.playbackTimeDidChange, (a) => {
|
|
self.playerLCD.playbackDuration = (self.mk.currentPlaybackTime)
|
|
self.lyriccurrenttime = app.mk.currentPlaybackTime;
|
|
|
|
if (self.lyricon) app.getActiveLyric();
|
|
// animated dot like AM - bad perf
|
|
if (self.lyricon && self.drawertest) {
|
|
let currentLine = document.querySelector(`.lyric-line.active`)
|
|
if (currentLine && currentLine.getElementsByClassName('lyricWaiting').length > 0) {
|
|
let duration = currentLine.getAttribute("end") - currentLine.getAttribute("start");
|
|
let u = (self.lyriccurrenttime - currentLine.getAttribute("start")) / duration;
|
|
if (u < 0.25 && !currentLine.classList.contains('mode1')) {
|
|
try {
|
|
currentLine.classList.add('mode1');
|
|
currentLine.classList.remove('mode3');
|
|
currentLine.classList.remove('mode2');
|
|
} catch (e) {
|
|
}
|
|
currentLine.getElementsByClassName('WaitingDot1')[0].style.animation = `dotOpacity ${0.25 * duration}s cubic-bezier(0.42, 0, 0.58, 1) forwards`;
|
|
currentLine.getElementsByClassName('WaitingDot2')[0].style.animation = ``;
|
|
currentLine.getElementsByClassName('WaitingDot3')[0].style.animation = ``;
|
|
currentLine.getElementsByClassName('WaitingDot2')[0].style.opacity = 0.25;
|
|
currentLine.getElementsByClassName('WaitingDot3')[0].style.opacity = 0.25;
|
|
|
|
} else if (u >= 0.25 && u < 0.5 && !currentLine.classList.contains('mode2')) {
|
|
try {
|
|
currentLine.classList.add('mode2');
|
|
currentLine.classList.remove('mode1');
|
|
currentLine.classList.remove('mode3');
|
|
} catch (e) {
|
|
}
|
|
currentLine.getElementsByClassName('WaitingDot2')[0].style.animation = `dotOpacity ${0.25 * duration}s cubic-bezier(0.42, 0, 0.58, 1) forwards`;
|
|
currentLine.getElementsByClassName('WaitingDot1')[0].style.animation = ``;
|
|
currentLine.getElementsByClassName('WaitingDot3')[0].style.animation = ``;
|
|
currentLine.getElementsByClassName('WaitingDot1')[0].style.opacity = 1;
|
|
currentLine.getElementsByClassName('WaitingDot3')[0].style.opacity = 0.25;
|
|
} else if (u >= 0.5 && u < 0.75 && !currentLine.classList.contains('mode3')) {
|
|
try {
|
|
currentLine.classList.add('mode3');
|
|
currentLine.classList.remove('mode1');
|
|
currentLine.classList.remove('mode2');
|
|
} catch (e) {
|
|
}
|
|
currentLine.getElementsByClassName('WaitingDot3')[0].style.animation = `dotOpacity ${0.25 * duration}s cubic-bezier(0.42, 0, 0.58, 1) forwards`;
|
|
currentLine.getElementsByClassName('WaitingDot1')[0].style.animation = ``;
|
|
currentLine.getElementsByClassName('WaitingDot2')[0].style.animation = ``;
|
|
currentLine.getElementsByClassName('WaitingDot1')[0].style.opacity = 1;
|
|
currentLine.getElementsByClassName('WaitingDot2')[0].style.opacity = 1;
|
|
} else if (u >= 0.75 && currentLine.classList.contains('mode3')) {
|
|
try {
|
|
currentLine.classList.remove('mode1');
|
|
currentLine.classList.remove('mode2');
|
|
currentLine.classList.remove('mode3');
|
|
} catch (e) {
|
|
}
|
|
currentLine.getElementsByClassName('WaitingDot1')[0].style.animation = ``;
|
|
currentLine.getElementsByClassName('WaitingDot2')[0].style.animation = ``;
|
|
currentLine.getElementsByClassName('WaitingDot1')[0].style.opacity = 1;
|
|
currentLine.getElementsByClassName('WaitingDot2')[0].style.opacity = 1;
|
|
|
|
}
|
|
|
|
}
|
|
}
|
|
})
|
|
|
|
this.mk.addEventListener(MusicKit.Events.nowPlayingItemDidChange, (a) => {
|
|
let type = (self.mk.nowPlayingItem != null) ? self.mk.nowPlayingItem["type"] ?? '' : '';
|
|
|
|
if (type.includes("musicVideo") || type.includes("uploadedVideo")) {
|
|
document.getElementById("apple-music-video-container").style.display = "block";
|
|
// app.chrome.topChromeVisible = false
|
|
} else {
|
|
document.getElementById("apple-music-video-container").style.display = "none";
|
|
// app.chrome.topChromeVisible = true
|
|
}
|
|
self.chrome.artworkReady = false
|
|
self.lyrics = []
|
|
app.loadLyrics()
|
|
})
|
|
|
|
this.apiCall('https://api.music.apple.com/v1/me/library/playlists', res => {
|
|
self.playlists.listing = res.data
|
|
})
|
|
document.body.removeAttribute("loading")
|
|
},
|
|
async getPlaylistFromID(id) {
|
|
try {
|
|
this.showingPlaylist = await app.mk.api.library.playlist(id)
|
|
} catch (e) {
|
|
console.log(e);
|
|
try {
|
|
this.showingPlaylist = await app.mk.api.playlist(id)
|
|
} catch (err) {
|
|
console.log(err)
|
|
}
|
|
}
|
|
|
|
},
|
|
async getArtistFromID(id) {
|
|
var artistData = await this.mkapi("artists", false, id, {
|
|
"views": "featured-release,full-albums,appears-on-albums,featured-albums,featured-on-albums,singles,compilation-albums,live-albums,latest-release,top-music-videos,similar-artists,top-songs,playlists,more-to-hear,more-to-see",
|
|
"extend": "artistBio,bornOrFormed,editorialArtwork,editorialVideo,isGroup,origin,hero",
|
|
"extend[playlists]": "trackCount",
|
|
"omit[resource:songs]": "relationships",
|
|
"fields[albums]": "artistName,artistUrl,artwork,contentRating,editorialArtwork,editorialVideo,name,playParams,releaseDate,url,trackCount",
|
|
"limit[artists:top-songs]": 20,
|
|
"art[url]": "f"
|
|
}, {includeResponseMeta: !0})
|
|
this.artistPage.data = artistData.data[0]
|
|
this.page = "artist-page"
|
|
},
|
|
hashCode(str) {
|
|
var hash = 0, i, chr;
|
|
if (str.length === 0) return hash;
|
|
for (i = 0; i < str.length; i++) {
|
|
chr = str.charCodeAt(i);
|
|
hash = ((hash << 5) - hash) + chr;
|
|
hash |= 0; // Convert to 32bit integer
|
|
}
|
|
return hash;
|
|
},
|
|
playAnimatedArtwork(url) {
|
|
if (Hls.isSupported()) {
|
|
var video = document.querySelector(`[vid="${app.hashCode(url)}"] > video`)
|
|
console.log('supported');
|
|
var hls = new Hls();
|
|
// bind them together
|
|
if (video) {
|
|
hls.attachMedia(video);
|
|
hls.on(Hls.Events.MEDIA_ATTACHED, function () {
|
|
console.log('video and hls.js are now bound together !');
|
|
hls.loadSource(url);
|
|
hls.on(Hls.Events.MANIFEST_PARSED, function (event, data) {
|
|
video.play();
|
|
return "";
|
|
});
|
|
});
|
|
} else {
|
|
console.log("hso");
|
|
return "";
|
|
}
|
|
} else {
|
|
return "";
|
|
}
|
|
},
|
|
routeView(item) {
|
|
let self = this
|
|
app.showingPlaylist = [];
|
|
let kind = (item.attributes.playParams ? (item.attributes.playParams.kind ?? (item.type ?? '')) : (item.type ?? ''));
|
|
let id = (item.attributes.playParams ? (item.attributes.playParams.id ?? (item.id ?? '')) : (item.id ?? ''));
|
|
;
|
|
let isLibrary = item.attributes.playParams ? (item.attributes.playParams.isLibrary ?? false) : false;
|
|
console.log(kind, id, isLibrary)
|
|
if (kind.toString().includes("artist")) {
|
|
app.getArtistInfo(id, isLibrary)
|
|
} else if (!kind.toString().includes("radioStation") && !kind.toString().includes("song") && !kind.toString().includes("musicVideo") && !kind.toString().includes("uploadedVideo")) {
|
|
app.page = (kind) + "_" + (id);
|
|
console.log("oks");
|
|
app.getTypeFromID((kind), (id), (isLibrary), {extend: "editorialVideo"});
|
|
} else {
|
|
app.playMediaItemById((id), (kind), (isLibrary), item.attributes.url ?? '')
|
|
}
|
|
document.querySelector("#app-content").scrollTop = 0
|
|
},
|
|
pushNavigationEvent(item) {
|
|
let self = this
|
|
|
|
},
|
|
exitMV() {
|
|
MusicKit.getInstance().stop()
|
|
document.getElementById("apple-music-video-container").style.display = "none";
|
|
},
|
|
getArtistInfo(id, isLibrary) {
|
|
var query = {
|
|
"omit[resource]": "autos",
|
|
views: ["featured-release", "full-albums", "appears-on-albums", "featured-albums", "featured-on-albums", "singles", "compilation-albums", "live-albums", "latest-release", "top-music-videos", "similar-artists", "top-songs", "playlists", "more-to-hear", "more-to-see"],
|
|
extend: ["artistBio", "bornOrFormed", "editorialArtwork", "editorialVideo", "isGroup", "origin", "hero"],
|
|
"extend[playlists]": ["trackCount"],
|
|
"omit[resource:songs]": "relationships",
|
|
"fields[albums]": [...["fields[albums]"], "trackCount"],
|
|
limit: {
|
|
"artists:top-songs": 20
|
|
},
|
|
"art[url]": "f"
|
|
};
|
|
this.getArtistFromID(id)
|
|
//this.getTypeFromID("artist",id,isLibrary,query)
|
|
},
|
|
playMediaItem(item) {
|
|
let kind = (item.attributes.playParams ? (item.attributes.playParams.kind ?? (item.type ?? '')) : (item.type ?? ''));
|
|
let id = (item.attributes.playParams ? (item.attributes.playParams.id ?? (item.id ?? '')) : (item.id ?? ''));
|
|
;
|
|
let isLibrary = item.attributes.playParams ? (item.attributes.playParams.isLibrary ?? false) : false;
|
|
console.log(kind, id, isLibrary)
|
|
app.playMediaItemById((id), (kind), (isLibrary), item.attributes.url ?? '')
|
|
},
|
|
async getTypeFromID(kind, id, isLibrary = false, params = {}) {
|
|
var a;
|
|
try {
|
|
a = await this.mkapi(kind.toString(), isLibrary, id.toString(), params);
|
|
} catch (e) {
|
|
console.log(e);
|
|
try {
|
|
console.log("opp", !isLibrary);
|
|
a = await this.mkapi(kind.toString(), !isLibrary, id.toString(), params);
|
|
} catch (err) {
|
|
console.log(err);
|
|
a = []
|
|
} finally {
|
|
this.showingPlaylist = a
|
|
}
|
|
} finally {
|
|
this.showingPlaylist = a
|
|
}
|
|
;
|
|
},
|
|
searchLibrarySongs() {
|
|
let self = this
|
|
|
|
function sortSongs() {
|
|
if (self.library.songs.sortOrder == "asc") {
|
|
// sort this.library.songs.displayListing by song.attributes[self.library.songs.sorting] in ascending order based on alphabetical order and numeric order
|
|
// check if song.attributes[self.library.songs.sorting] is a number and if so, sort by number if not, sort by alphabetical order ignoring case
|
|
self.library.songs.displayListing.sort((a, b) => {
|
|
let aa = null;
|
|
let bb = null;
|
|
if (self.library.songs.sorting == "genre") {
|
|
aa = a.attributes.genreNames[0]
|
|
bb = b.attributes.genreNames[0]
|
|
}
|
|
aa = a.attributes[self.library.songs.sorting]
|
|
bb = b.attributes[self.library.songs.sorting]
|
|
if (aa == null) {
|
|
aa = ""
|
|
}
|
|
if (bb == null) {
|
|
bb = ""
|
|
}
|
|
if (aa.toString().match(/^\d+$/) && bb.toString().match(/^\d+$/)) {
|
|
return aa - bb
|
|
} else {
|
|
return aa.toString().toLowerCase().localeCompare(bb.toString().toLowerCase())
|
|
}
|
|
})
|
|
}
|
|
if (self.library.songs.sortOrder == "desc") {
|
|
// sort this.library.songs.displayListing by song.attributes[self.library.songs.sorting] in descending order based on alphabetical order and numeric order
|
|
// check if song.attributes[self.library.songs.sorting] is a number and if so, sort by number if not, sort by alphabetical order ignoring case
|
|
self.library.songs.displayListing.sort((a, b) => {
|
|
if (self.library.songs.sorting == "genre") {
|
|
aa = a.attributes.genreNames[0]
|
|
bb = b.attributes.genreNames[0]
|
|
}
|
|
let aa = a.attributes[self.library.songs.sorting]
|
|
let bb = b.attributes[self.library.songs.sorting]
|
|
if (aa == null) {
|
|
aa = ""
|
|
}
|
|
if (bb == null) {
|
|
bb = ""
|
|
}
|
|
if (aa.toString().match(/^\d+$/) && bb.toString().match(/^\d+$/)) {
|
|
return bb - aa
|
|
} else {
|
|
return bb.toString().toLowerCase().localeCompare(aa.toString().toLowerCase())
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
if (this.library.songs.search == "") {
|
|
this.library.songs.displayListing = this.library.songs.listing
|
|
sortSongs()
|
|
} else {
|
|
this.library.songs.displayListing = this.library.songs.listing.filter(item => {
|
|
let itemName = item.attributes.name.toLowerCase()
|
|
let searchTerm = this.library.songs.search.toLowerCase()
|
|
let artistName = ""
|
|
let albumName = ""
|
|
if (item.attributes.artistName != null) {
|
|
artistName = item.attributes.artistName.toLowerCase()
|
|
}
|
|
if (item.attributes.albumName != null) {
|
|
albumName = item.attributes.albumName.toLowerCase()
|
|
}
|
|
|
|
// remove any non-alphanumeric characters and spaces from search term and item name
|
|
searchTerm = searchTerm.replace(/[^a-z0-9 ]/gi, "")
|
|
itemName = itemName.replace(/[^a-z0-9 ]/gi, "")
|
|
artistName = artistName.replace(/[^a-z0-9 ]/gi, "")
|
|
albumName = albumName.replace(/[^a-z0-9 ]/gi, "")
|
|
|
|
if (itemName.includes(searchTerm) || artistName.includes(searchTerm) || albumName.includes(searchTerm)) {
|
|
return item
|
|
}
|
|
})
|
|
sortSongs()
|
|
}
|
|
},
|
|
// make a copy of searchLibrarySongs except use Albums instead of Songs
|
|
searchLibraryAlbums() {
|
|
let self = this
|
|
function sortAlbums() {
|
|
if (self.library.albums.sortOrder == "asc") {
|
|
// sort this.library.albums.displayListing by album.attributes[self.library.albums.sorting] in ascending order based on alphabetical order and numeric order
|
|
// check if album.attributes[self.library.albums.sorting] is a number and if so, sort by number if not, sort by alphabetical order ignoring case
|
|
self.library.albums.displayListing.sort((a, b) => {
|
|
let aa = null;
|
|
let bb = null;
|
|
if (self.library.albums.sorting == "genre") {
|
|
aa = a.attributes.genreNames[0]
|
|
bb = b.attributes.genreNames[0]
|
|
}
|
|
aa = a.attributes[self.library.albums.sorting]
|
|
bb = b.attributes[self.library.albums.sorting]
|
|
if (aa == null) {
|
|
aa = ""
|
|
}
|
|
if (bb == null) {
|
|
bb = ""
|
|
}
|
|
if (aa.toString().match(/^\d+$/) && bb.toString().match(/^\d+$/)) {
|
|
return aa - bb
|
|
} else {
|
|
return aa.toString().toLowerCase().localeCompare(bb.toString().toLowerCase())
|
|
}
|
|
})
|
|
}
|
|
if (self.library.albums.sortOrder == "desc") {
|
|
// sort this.library.albums.displayListing by album.attributes[self.library.albums.sorting] in descending order based on alphabetical order and numeric order
|
|
// check if album.attributes[self.library.albums.sorting] is a number and if so, sort by number if not, sort by alphabetical order ignoring case
|
|
self.library.albums.displayListing.sort((a, b) => {
|
|
if (self.library.albums.sorting == "genre") {
|
|
aa = a.attributes.genreNames[0]
|
|
bb = b.attributes.genreNames[0]
|
|
}
|
|
let aa = a.attributes[self.library.albums.sorting]
|
|
let bb = b.attributes[self.library.albums.sorting]
|
|
if (aa == null) {
|
|
aa = ""
|
|
}
|
|
if (bb == null) {
|
|
bb = ""
|
|
}
|
|
if (aa.toString().match(/^\d+$/) && bb.toString().match(/^\d+$/)) {
|
|
return bb - aa
|
|
} else {
|
|
return bb.toString().toLowerCase().localeCompare(aa.toString().toLowerCase())
|
|
}
|
|
})
|
|
}
|
|
}
|
|
if (this.library.albums.search == "") {
|
|
this.library.albums.displayListing = this.library.albums.listing
|
|
sortAlbums()
|
|
} else {
|
|
this.library.albums.displayListing = this.library.albums.listing.filter(item => {
|
|
let itemName = item.attributes.name.toLowerCase()
|
|
let searchTerm = this.library.albums.search.toLowerCase()
|
|
let artistName = ""
|
|
let albumName = ""
|
|
if (item.attributes.artistName != null) {
|
|
artistName = item.attributes.artistName.toLowerCase()
|
|
}
|
|
if (item.attributes.albumName != null) {
|
|
albumName = item.attributes.albumName.toLowerCase()
|
|
}
|
|
|
|
// remove any non-alphanumeric characters and spaces from search term and item name
|
|
searchTerm = searchTerm.replace(/[^a-z0-9 ]/gi, "")
|
|
itemName = itemName.replace(/[^a-z0-9 ]/gi, "")
|
|
artistName = artistName.replace(/[^a-z0-9 ]/gi, "")
|
|
albumName = albumName.replace(/[^a-z0-9 ]/gi, "")
|
|
|
|
if (itemName.includes(searchTerm) || artistName.includes(searchTerm) || albumName.includes(searchTerm)) {
|
|
return item
|
|
}
|
|
})
|
|
sortAlbums()
|
|
}
|
|
},
|
|
getSidebarItemClass(page) {
|
|
if (this.page == page) {
|
|
return ["active"]
|
|
} else {
|
|
return []
|
|
}
|
|
},
|
|
async mkapi(method, library = false, term, params = {}, params2 = {}, attempts = 0) {
|
|
if (attempts > 3) {
|
|
return
|
|
}
|
|
try {
|
|
if (library) {
|
|
return await this.mk.api.library[method](term, params, params2)
|
|
} else {
|
|
return await this.mk.api[method](term, params, params2)
|
|
}
|
|
} catch (e) {
|
|
console.log(e)
|
|
return await this.mkapi(method, library, term, params, params2, attempts + 1)
|
|
}
|
|
},
|
|
getLibraryGenres() {
|
|
let genres = []
|
|
genres = []
|
|
this.library.songs.listing.forEach((item) => {
|
|
item.attributes.genreNames.forEach((genre) => {
|
|
if (!genres.includes(genre)) {
|
|
genres.push(genre)
|
|
}
|
|
})
|
|
})
|
|
return genres
|
|
},
|
|
async getLibrarySongsFull(force = false) {
|
|
let self = this
|
|
let library = []
|
|
let downloaded = null;
|
|
if ((this.library.songs.downloadState == 2) && !force) {
|
|
return
|
|
}
|
|
if(this.library.songs.downloadState == 1) {
|
|
return
|
|
}
|
|
if (localStorage.getItem("librarySongs") != null) {
|
|
this.library.songs.listing = JSON.parse(localStorage.getItem("librarySongs"))
|
|
this.searchLibrarySongs()
|
|
}
|
|
if (this.songstest) {
|
|
return
|
|
}
|
|
this.library.songs.downloadState = 1
|
|
this.library.downloadNotification.show = true
|
|
this.library.downloadNotification.message = "Updating library songs..."
|
|
|
|
function downloadChunk() {
|
|
self.library.songs.downloadState = 1
|
|
if (downloaded == null) {
|
|
app.mk.api.library.songs("", {limit: 100}, {includeResponseMeta: !0}).then((response) => {
|
|
processChunk(response)
|
|
})
|
|
} else {
|
|
downloaded.next("", {limit: 100}, {includeResponseMeta: !0}).then((response) => {
|
|
processChunk(response)
|
|
})
|
|
}
|
|
}
|
|
|
|
function processChunk(response) {
|
|
downloaded = response
|
|
library = library.concat(downloaded.data)
|
|
self.library.downloadNotification.show = true
|
|
self.library.downloadNotification.message = "Updating library songs..."
|
|
self.library.downloadNotification.total = downloaded.meta.total
|
|
self.library.downloadNotification.progress = library.length
|
|
|
|
if (downloaded.meta.total == 0) {
|
|
self.library.songs.downloadState = 3
|
|
return
|
|
}
|
|
if (typeof downloaded.next == "undefined") {
|
|
console.log("downloaded.next is undefined")
|
|
self.library.songs.listing = library
|
|
self.library.songs.downloadState = 2
|
|
self.library.downloadNotification.show = false
|
|
self.searchLibrarySongs()
|
|
localStorage.setItem("librarySongs", JSON.stringify(library))
|
|
}
|
|
if (downloaded.meta.total > library.length || typeof downloaded.meta.next != "undefined") {
|
|
console.log(`downloading next chunk - ${library.length} songs so far`)
|
|
downloadChunk()
|
|
} else {
|
|
self.library.songs.listing = library
|
|
self.library.songs.downloadState = 2
|
|
self.library.downloadNotification.show = false
|
|
self.searchLibrarySongs()
|
|
localStorage.setItem("librarySongs", JSON.stringify(library))
|
|
console.log(library)
|
|
}
|
|
}
|
|
|
|
downloadChunk()
|
|
},
|
|
// copy the getLibrarySongsFull function except change Songs to Albums
|
|
async getLibraryAlbumsFull(force = false) {
|
|
let self = this
|
|
let library = []
|
|
let downloaded = null;
|
|
if ((this.library.albums.downloadState == 2 || this.library.albums.downloadState == 1) && !force) {
|
|
return
|
|
}
|
|
if (localStorage.getItem("libraryAlbums") != null) {
|
|
this.library.albums.listing = JSON.parse(localStorage.getItem("libraryAlbums"))
|
|
this.searchLibraryAlbums()
|
|
}
|
|
if (this.songstest) {
|
|
return
|
|
}
|
|
this.library.albums.downloadState = 1
|
|
this.library.downloadNotification.show = true
|
|
this.library.downloadNotification.message = "Updating library albums..."
|
|
|
|
function downloadChunk() {
|
|
self.library.albums.downloadState = 1
|
|
if (downloaded == null) {
|
|
app.mk.api.library.albums("", {limit: 100}, {includeResponseMeta: !0}).then((response) => {
|
|
processChunk(response)
|
|
})
|
|
} else {
|
|
downloaded.next("", {limit: 100}, {includeResponseMeta: !0}).then((response) => {
|
|
processChunk(response)
|
|
})
|
|
}
|
|
}
|
|
|
|
function processChunk(response) {
|
|
downloaded = response
|
|
library = library.concat(downloaded.data)
|
|
self.library.downloadNotification.show = true
|
|
self.library.downloadNotification.message = "Updating library albums..."
|
|
self.library.downloadNotification.total = downloaded.meta.total
|
|
self.library.downloadNotification.progress = library.length
|
|
if (downloaded.meta.total == 0) {
|
|
self.library.albums.downloadState = 3
|
|
return
|
|
}
|
|
if (typeof downloaded.next == "undefined") {
|
|
console.log("downloaded.next is undefined")
|
|
self.library.albums.listing = library
|
|
self.library.albums.downloadState = 2
|
|
self.library.downloadNotification.show = false
|
|
localStorage.setItem("libraryAlbums", JSON.stringify(library))
|
|
self.searchLibraryAlbums()
|
|
}
|
|
if (downloaded.meta.total > library.length || typeof downloaded.meta.next != "undefined") {
|
|
console.log(`downloading next chunk - ${library.length
|
|
} albums so far`)
|
|
downloadChunk()
|
|
} else {
|
|
self.library.albums.listing = library
|
|
self.library.albums.downloadState = 2
|
|
self.library.downloadNotification.show = false
|
|
localStorage.setItem("libraryAlbums", JSON.stringify(library))
|
|
self.searchLibraryAlbums()
|
|
console.log(library)
|
|
}
|
|
}
|
|
downloadChunk()
|
|
},
|
|
getTotalTime() {
|
|
if (app.showingPlaylist.relationships.tracks.data.length > 0) {
|
|
time = Math.round([].concat(...app.showingPlaylist.relationships.tracks.data).reduce((a, {attributes: {durationInMillis}}) => a + durationInMillis, 0) / 60000);
|
|
return app.showingPlaylist.relationships.tracks.data.length + " tracks, " + time + " mins.";
|
|
} else return ""
|
|
},
|
|
async getLibrarySongs() {
|
|
var response = await this.mkapi("songs", true, "", {limit: 100}, {includeResponseMeta: !0})
|
|
this.library.songs.listing = response.data
|
|
this.library.songs.meta = response.meta
|
|
},
|
|
async getLibraryAlbums() {
|
|
var response = await this.mkapi("albums", true, "", {limit: 100}, {includeResponseMeta: !0})
|
|
this.library.albums.listing = response.data
|
|
this.library.albums.meta = response.meta
|
|
},
|
|
async getListenNow(attempt = 0) {
|
|
if (attempt > 3) {
|
|
return
|
|
}
|
|
try {
|
|
this.listennow = await this.mk.api.personalRecommendations("",
|
|
{
|
|
name: "listen-now",
|
|
with: "friendsMix,library,social",
|
|
"art[social-profiles:url]": "c",
|
|
"art[url]": "c,f",
|
|
"omit[resource]": "autos",
|
|
"relate[editorial-items]": "contents",
|
|
extend: ["editorialCard", "editorialVideo"],
|
|
"extend[albums]": ["artistUrl"],
|
|
"extend[library-albums]": ["artistUrl", "editorialVideo"],
|
|
"extend[playlists]": ["artistNames", "editorialArtwork", "editorialVideo"],
|
|
"extend[library-playlists]": ["artistNames", "editorialArtwork", "editorialVideo"],
|
|
"extend[social-profiles]": "topGenreNames",
|
|
"include[albums]": "artists",
|
|
"include[songs]": "artists",
|
|
"include[music-videos]": "artists",
|
|
"fields[albums]": ["artistName", "artistUrl", "artwork", "contentRating", "editorialArtwork", "editorialVideo", "name", "playParams", "releaseDate", "url"],
|
|
"fields[artists]": ["name", "url"],
|
|
"extend[stations]": ["airDate", "supportsAirTimeUpdates"],
|
|
"meta[stations]": "inflectionPoints",
|
|
types: "artists,albums,editorial-items,library-albums,library-playlists,music-movies,music-videos,playlists,stations,uploaded-audios,uploaded-videos,activities,apple-curators,curators,tv-shows,social-profiles,social-upsells",
|
|
platform: "web"
|
|
},
|
|
{
|
|
includeResponseMeta: !0,
|
|
reload: !0
|
|
});
|
|
console.log(this.listennow)
|
|
} catch (e) {
|
|
console.log(e)
|
|
this.getListenNow(attempt + 1)
|
|
}
|
|
},
|
|
async getRadioStations(attempt = 0) {
|
|
if (attempt > 3) {
|
|
return
|
|
}
|
|
try {
|
|
this.radio.personal = await this.mkapi("recentRadioStations", false, "",
|
|
{
|
|
"platform": "web",
|
|
"art[url]": "f"
|
|
});
|
|
} catch (e) {
|
|
console.log(e)
|
|
this.getRadioStations(attempt + 1)
|
|
}
|
|
},
|
|
unauthorize() {
|
|
this.mk.unauthorize()
|
|
},
|
|
showSearch() {
|
|
this.page = "search"
|
|
},
|
|
loadLyrics() {
|
|
this.loadMXM();
|
|
},
|
|
loadAMLyrics() {
|
|
const songID = (this.mk.nowPlayingItem != null) ? this.mk.nowPlayingItem["_songId"] ?? -1 : -1;
|
|
// this.getMXM( trackName, artistName, 'en', duration);
|
|
if (songID != -1) {
|
|
MusicKit.getInstance().api.lyric(songID)
|
|
.then((response) => {
|
|
this.lyricsMediaItem = response.attributes["ttml"]
|
|
this.parseTTML()
|
|
})
|
|
}
|
|
},
|
|
addToLibrary(id) {
|
|
let self = this
|
|
this.mk.addToLibrary(id).then((data)=>{
|
|
self.getLibrarySongsFull(true)
|
|
})
|
|
},
|
|
loadMXM() {
|
|
let attempt = 0;
|
|
const track = encodeURIComponent((this.mk.nowPlayingItem != null) ? this.mk.nowPlayingItem.title ?? '' : '');
|
|
const artist = encodeURIComponent((this.mk.nowPlayingItem != null) ? this.mk.nowPlayingItem.artistName ?? '' : '');
|
|
const time = encodeURIComponent((this.mk.nowPlayingItem != null) ? (Math.round((this.mk.nowPlayingItem.attributes["durationInMillis"] ?? -1000) / 1000) ?? -1) : -1);
|
|
var lrcfile = "";
|
|
const lang = "en" // translation language
|
|
function revisedRandId() {
|
|
return Math.random().toString(36).replace(/[^a-z]+/g, '').substr(2, 10);
|
|
}
|
|
|
|
/* get token */
|
|
function getToken(mode, track, artist, songid, lang, time) {
|
|
if (attempt > 2) {
|
|
app.loadAMLyrics();
|
|
} else {
|
|
attempt = attempt + 1;
|
|
let url = "https://apic-desktop.musixmatch.com/ws/1.1/token.get?app_id=web-desktop-app-v1.0&t=" + revisedRandId();
|
|
let req = new XMLHttpRequest();
|
|
req.overrideMimeType("application/json");
|
|
req.open('GET', url, true);
|
|
req.setRequestHeader("authority", "apic-desktop.musixmatch.com");
|
|
req.onload = function () {
|
|
let jsonResponse = JSON.parse(this.responseText);
|
|
let status2 = jsonResponse["message"]["header"]["status_code"];
|
|
if (status2 == 200) {
|
|
let token = jsonResponse["message"]["body"]["user_token"] ?? '';
|
|
if (token != "" && token != "UpgradeOnlyUpgradeOnlyUpgradeOnlyUpgradeOnly") {
|
|
console.log('200 token', mode);
|
|
// token good
|
|
app.mxmtoken = token;
|
|
|
|
if (mode == 1) {
|
|
getMXMSubs(track, artist, app.mxmtoken, lang, time);
|
|
} else {
|
|
getMXMTrans(songid, lang, app.mxmtoken);
|
|
}
|
|
} else {
|
|
console.log('fake 200 token');
|
|
getToken(mode, track, artist, songid, lang, time)
|
|
}
|
|
} else {
|
|
console.log('token 4xx');
|
|
getToken(mode, track, artist, songid, lang, time)
|
|
}
|
|
|
|
};
|
|
req.onerror = function () {
|
|
console.log('error');
|
|
app.loadAMLyrics();
|
|
};
|
|
req.send();
|
|
}
|
|
}
|
|
|
|
function getMXMSubs(track, artist, token, lang, time) {
|
|
var usertoken = encodeURIComponent(token);
|
|
var timecustom = (!time || (time && time < 0)) ? '' : `&f_subtitle_length=${time}&q_duration=${time}&f_subtitle_length_max_deviation=40`;
|
|
var url = "https://apic-desktop.musixmatch.com/ws/1.1/macro.subtitles.get?format=json&namespace=lyrics_richsynched&subtitle_format=lrc&q_artist=" + artist + "&q_track=" + track + "&usertoken=" + usertoken + timecustom + "&app_id=web-desktop-app-v1.0&t=" + revisedRandId();
|
|
var req = new XMLHttpRequest();
|
|
req.overrideMimeType("application/json");
|
|
req.open('GET', url, true);
|
|
req.setRequestHeader("authority", "apic-desktop.musixmatch.com");
|
|
req.onload = function () {
|
|
var jsonResponse = JSON.parse(this.responseText);
|
|
console.log(jsonResponse);
|
|
var status1 = jsonResponse["message"]["header"]["status_code"];
|
|
|
|
if (status1 == 200) {
|
|
let id = '';
|
|
try {
|
|
if (jsonResponse["message"]["body"]["macro_calls"]["matcher.track.get"]["message"]["header"]["status_code"] == 200 && jsonResponse["message"]["body"]["macro_calls"]["track.subtitles.get"]["message"]["header"]["status_code"] == 200) {
|
|
id = jsonResponse["message"]["body"]["macro_calls"]["matcher.track.get"]["message"]["body"]["track"]["track_id"] ?? '';
|
|
lrcfile = jsonResponse["message"]["body"]["macro_calls"]["track.subtitles.get"]["message"]["body"]["subtitle_list"][0]["subtitle"]["subtitle_body"];
|
|
}
|
|
|
|
if (lrcfile == "") {
|
|
console.log('track not found');
|
|
app.loadAMLyrics()
|
|
} else {
|
|
// process lrcfile to json here
|
|
app.lyricsMediaItem = lrcfile
|
|
let u = app.lyricsMediaItem.split(/[\r\n]/);
|
|
let preLrc = []
|
|
for (var i = u.length - 1; i >= 0; i--) {
|
|
let xline = (/(\[[0-9.:\[\]]*\])+(.*)/).exec(u[i])
|
|
let end = (preLrc.length > 0) ? ((preLrc[preLrc.length - 1].startTime) ?? 99999) : 99999
|
|
preLrc.push({
|
|
startTime: app.toMS(xline[1].substring(1, xline[1].length - 2)) ?? 0,
|
|
endTime: end,
|
|
line: xline[2],
|
|
translation: ''
|
|
})
|
|
}
|
|
if (preLrc.length > 0)
|
|
preLrc.push({
|
|
startTime: 0,
|
|
endTime: preLrc[preLrc.length - 1].startTime,
|
|
line: "lrcInstrumental",
|
|
translation: ''
|
|
});
|
|
app.lyrics = preLrc.reverse();
|
|
if (lrcfile != null && lrcfile != '') {
|
|
// load translation
|
|
getMXMTrans(id, lang, token);
|
|
} else {
|
|
app.loadAMLyrics()
|
|
}
|
|
}
|
|
} catch (e) {
|
|
console.log(e);
|
|
console.log('track not found ??');
|
|
app.loadAMLyrics()
|
|
}
|
|
} else { //4xx rejected
|
|
getToken(1, track, artist, '', lang, time);
|
|
}
|
|
}
|
|
req.send();
|
|
}
|
|
|
|
function getMXMTrans(id, lang, token) {
|
|
if (lang != "disabled" && id != '') {
|
|
let usertoken = encodeURIComponent(token);
|
|
let url2 = "https://apic-desktop.musixmatch.com/ws/1.1/crowd.track.translations.get?translation_fields_set=minimal&selected_language=" + lang + "&track_id=" + id + "&comment_format=text&part=user&format=json&usertoken=" + usertoken + "&app_id=web-desktop-app-v1.0&t=" + revisedRandId();
|
|
let req2 = new XMLHttpRequest();
|
|
req2.overrideMimeType("application/json");
|
|
req2.open('GET', url2, true);
|
|
req2.setRequestHeader("authority", "apic-desktop.musixmatch.com");
|
|
req2.onload = function () {
|
|
let jsonResponse2 = JSON.parse(this.responseText);
|
|
console.log(jsonResponse2);
|
|
let status2 = jsonResponse2["message"]["header"]["status_code"];
|
|
if (status2 == 200) {
|
|
try {
|
|
let preTrans = []
|
|
let u = app.lyrics;
|
|
let translation_list = jsonResponse2["message"]["body"]["translations_list"];
|
|
if (translation_list.length > 0) {
|
|
for (var i = 0; i < u.length - 1; i++) {
|
|
preTrans[i] = ""
|
|
for (var trans_line of translation_list) {
|
|
if (u[i].line == " " + trans_line["translation"]["matched_line"]) {
|
|
u[i].translation = trans_line["translation"]["description"];
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
app.lyrics = u;
|
|
}
|
|
} catch (e) {
|
|
/// not found trans -> ignore
|
|
}
|
|
} else { //4xx rejected
|
|
getToken(2, '', '', id, lang, '');
|
|
}
|
|
}
|
|
req2.send();
|
|
}
|
|
|
|
}
|
|
|
|
if (track != "" & track != "No Title Found") {
|
|
if (app.mxmtoken != null && app.mxmtoken != '') {
|
|
console.log("we good");
|
|
getMXMSubs(track, artist, app.mxmtoken, lang, time)
|
|
} else {
|
|
console.log("get token");
|
|
getToken(1, track, artist, '', lang, time);
|
|
}
|
|
}
|
|
},
|
|
toMS(str) {
|
|
let rawTime = str.match(/(\d+:)?(\d+:)?(\d+)(\.\d+)?/);
|
|
let hours = (rawTime[2] != null) ? (rawTime[1].replace(":", "")) : 0;
|
|
let minutes = (rawTime[2] != null) ? (hours * 60 + rawTime[2].replace(":", "") * 1) : ((rawTime[1] != null) ? rawTime[1].replace(":", "") : 0);
|
|
let seconds = (rawTime[3] != null) ? (rawTime[3]) : 0;
|
|
let milliseconds = (rawTime[4] != null) ? (rawTime[4].replace(".", "")) : 0
|
|
return parseFloat(`${minutes * 60 + seconds * 1}.${milliseconds * 1}`);
|
|
},
|
|
parseTTML() {
|
|
this.lyrics = [];
|
|
let preLrc = [];
|
|
let xml = this.stringToXml(this.lyricsMediaItem);
|
|
let lyricsLines = xml.getElementsByTagName('p');
|
|
let synced = true;
|
|
let endTimes = [];
|
|
if (xml.getElementsByTagName('tt')[0].getAttribute("itunes:timing") === "None") {
|
|
synced = false;
|
|
}
|
|
endTimes.push(0);
|
|
if (synced) {
|
|
for (element of lyricsLines) {
|
|
start = this.toMS(element.getAttribute('begin'))
|
|
end = this.toMS(element.getAttribute('end'))
|
|
if (start - endTimes[endTimes.length - 1] > 5 && endTimes[endTimes.length - 1] != 0) {
|
|
preLrc.push({
|
|
startTime: endTimes[endTimes.length - 1],
|
|
endTime: start,
|
|
line: "lrcInstrumental"
|
|
});
|
|
}
|
|
preLrc.push({startTime: start, endTime: end, line: element.textContent});
|
|
endTimes.push(end);
|
|
}
|
|
// first line dot
|
|
if (preLrc.length > 0)
|
|
preLrc.unshift({startTime: 0, endTime: preLrc[0].startTime, line: "lrcInstrumental"});
|
|
} else {
|
|
for (element of lyricsLines) {
|
|
preLrc.push({startTime: 9999999, endTime: 9999999, line: element.textContent});
|
|
}
|
|
}
|
|
this.lyrics = preLrc;
|
|
|
|
},
|
|
parseLyrics() {
|
|
var xml = this.stringToXml(this.lyricsMediaItem)
|
|
var json = xmlToJson(xml);
|
|
this.lyrics = json
|
|
},
|
|
stringToXml(st) {
|
|
// string to xml
|
|
var xml = (new DOMParser()).parseFromString(st, "text/xml");
|
|
return xml;
|
|
|
|
},
|
|
getCurrentTime() {
|
|
return parseFloat(this.hmsToSecondsOnly(this.parseTime(this.mk.nowPlayingItem.attributes.durationInMillis - app.mk.currentPlaybackTimeRemaining * 1000)));
|
|
},
|
|
getActiveLyric() {
|
|
const delayfix = 0.5
|
|
const prevLine = app.currentLyricsLine;
|
|
for (var i = 0; i < app.lyrics.length; i++) {
|
|
if (this.lyriccurrenttime + delayfix >= app.lyrics[i].startTime && this.lyriccurrenttime + delayfix <= app.lyrics[i].endTime) {
|
|
if (app.currentLyricsLine != i) {
|
|
app.currentLyricsLine = i;
|
|
if (app.lyricon && document.querySelector(`.lyric-line[line-index="${i}"]`)) {
|
|
document.querySelector(`.lyric-line[line-index="${prevLine}"]`).classList.remove("active");
|
|
document.querySelector(`.lyric-line[line-index="${i}"]`).classList.add("active");
|
|
if (checkIfScrollIsStatic) {
|
|
document.querySelector(`.lyric-line[line-index="${i}"]`).scrollIntoView({
|
|
behavior: "smooth",
|
|
block: "center"
|
|
})
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
},
|
|
seekTo(time) {
|
|
this.mk.seekToTime(time);
|
|
},
|
|
parseTime(value) {
|
|
var minutes = Math.floor(value / 60000);
|
|
var seconds = ((value % 60000) / 1000).toFixed(0);
|
|
return minutes + ":" + (seconds < 10 ? '0' : '') + seconds;
|
|
},
|
|
parseTimeDecimal(value) {
|
|
var minutes = Math.floor(value / 60000);
|
|
var seconds = ((value % 60000) / 1000).toFixed(0);
|
|
return minutes + "." + (seconds < 10 ? '0' : '') + seconds;
|
|
},
|
|
hmsToSecondsOnly(str) {
|
|
var p = str.split(':'),
|
|
s = 0,
|
|
m = 1;
|
|
|
|
while (p.length > 0) {
|
|
s += m * parseInt(p.pop(), 10);
|
|
m *= 60;
|
|
}
|
|
|
|
return s;
|
|
},
|
|
getLyricBGStyle(start, end) {
|
|
var currentTime = this.getCurrentTime();
|
|
var duration = this.mk.nowPlayingItem.attributes.durationInMillis
|
|
var start2 = this.hmsToSecondsOnly(start)
|
|
var end2 = this.hmsToSecondsOnly(end)
|
|
var currentProgress = ((100 * (currentTime)) / (end2))
|
|
// check if currenttime is between start and end
|
|
this.player.lyricsDebug.start = start2
|
|
this.player.lyricsDebug.end = end2
|
|
this.player.lyricsDebug.current = currentTime
|
|
if (currentTime >= start2 && currentTime <= end2) {
|
|
return {
|
|
"--bgSpeed": `${(end2 - start2)}s`
|
|
}
|
|
} else {
|
|
return {}
|
|
}
|
|
},
|
|
playMediaItemById(id, kind, isLibrary, raurl = "") {
|
|
var truekind = (!kind.endsWith("s")) ? (kind + "s") : kind;
|
|
console.log(id, truekind, isLibrary)
|
|
try {
|
|
if (truekind == "radioStations") {
|
|
this.mk.setStationQueue({url: raurl}).then(function (queue) {
|
|
MusicKit.getInstance().play()
|
|
});
|
|
} else {
|
|
this.mk.setQueue({[truekind]: [id]}).then(function (queue) {
|
|
MusicKit.getInstance().play()
|
|
})
|
|
}
|
|
} catch (err) {
|
|
console.log(err)
|
|
this.playMediaItemById(id, kind, isLibrary, raurl)
|
|
}
|
|
},
|
|
friendlyTypes(type) {
|
|
// use switch statement to return friendly name for media types "songs,artists,albums,playlists,music-videos,stations,apple-curators,curators"
|
|
switch (type) {
|
|
case "song":
|
|
return "Songs"
|
|
break;
|
|
case "artist":
|
|
return "Artists"
|
|
break;
|
|
case "album":
|
|
return "Albums"
|
|
break;
|
|
case "playlist":
|
|
return "Playlists"
|
|
break;
|
|
case "music_video":
|
|
return "Music Videos"
|
|
break;
|
|
case "station":
|
|
return "Stations"
|
|
break;
|
|
case "apple-curator":
|
|
return "Apple Curators"
|
|
break;
|
|
case "radio_show":
|
|
return "Radio Shows"
|
|
break;
|
|
case "record_label":
|
|
return "Record Labels"
|
|
break;
|
|
case "radio_episode":
|
|
return "Episodes"
|
|
break;
|
|
case "video_extra":
|
|
return "Video Extras"
|
|
break;
|
|
case "curator":
|
|
return "Curators"
|
|
break;
|
|
case "top":
|
|
return "Top"
|
|
break;
|
|
default:
|
|
return type
|
|
break;
|
|
}
|
|
},
|
|
async searchQuery() {
|
|
let self = this
|
|
this.mk.api.search(this.search.term,
|
|
{
|
|
types: "activities,albums,apple-curators,artists,curators,editorial-items,music-movies,music-videos,playlists,songs,stations,tv-episodes,uploaded-videos,record-labels",
|
|
"relate[editorial-items]": "contents",
|
|
"include[editorial-items]": "contents",
|
|
"include[albums]": "artists",
|
|
"include[artists]": "artists",
|
|
"include[songs]": "artists",
|
|
"include[music-videos]": "artists",
|
|
"extend": "artistUrl",
|
|
"fields[artists]": "url,name,artwork,hero",
|
|
"fields[albums]": "artistName,artistUrl,artwork,contentRating,editorialArtwork,name,playParams,releaseDate,url",
|
|
"with": "serverBubbles,lyricHighlights",
|
|
"art[url]": "c,f",
|
|
"omit[resource]": "autos",
|
|
"platform": "web",
|
|
limit: 25
|
|
}).then(function (results) {
|
|
self.search.results = results
|
|
})
|
|
},
|
|
isInLibrary(playParams) {
|
|
let self = this
|
|
let id = ""
|
|
// ugly code to check if current playback item is in library
|
|
if (typeof playParams == "undefined") {
|
|
return true
|
|
}
|
|
if (playParams["isLibrary"]) {
|
|
return true
|
|
} else if (playParams["catalogId"]) {
|
|
id = playParams["catalogId"]
|
|
} else if (playParams["id"]) {
|
|
id = playParams["id"]
|
|
}
|
|
var found = this.library.songs.listing.filter((item) => {
|
|
if (item["attributes"]) {
|
|
if (item["attributes"]["playParams"] && (item["attributes"]["playParams"]["catalogId"] == id)) {
|
|
return item;
|
|
}
|
|
}
|
|
})
|
|
if (found.length != 0) {
|
|
return true
|
|
} else {
|
|
return false
|
|
}
|
|
},
|
|
mkReady() {
|
|
if (this.mk["nowPlayingItem"]) {
|
|
return true
|
|
} else {
|
|
return false
|
|
}
|
|
},
|
|
getMediaItemArtwork(url, size = 64) {
|
|
var newurl = `${url.replace('{w}', size).replace('{h}', size).replace('{f}', "webp").replace('{c}', "cc")}`;
|
|
return newurl
|
|
},
|
|
getNowPlayingArtworkBG(size = 600) {
|
|
if (!this.mkReady()) {
|
|
return ""
|
|
}
|
|
try {
|
|
if (this.mk["nowPlayingItem"]["attributes"]["artwork"]["url"]) {
|
|
return `${this.mk["nowPlayingItem"]["attributes"]["artwork"]["url"].replace('{w}', size).replace('{h}', size)}`;
|
|
} else {
|
|
return "";
|
|
}
|
|
} catch (e) {
|
|
return ""
|
|
// Does not work
|
|
// this.mk.api.library.song(this.mk.nowPlayingItem.id).then((data) => {
|
|
// try {
|
|
// if (data != null && data !== "") {
|
|
// //document.getElementsByClassName("bg-artwork")[0].setAttribute('src', `${data["attributes"]["artwork"]["url"]}`)
|
|
// return `${data["attributes"]["artwork"]["url"]}`;
|
|
// } else {
|
|
// return "https://beta.music.apple.com/assets/product/MissingArtworkMusic.svg";
|
|
// }
|
|
// } catch (e) {
|
|
// return "https://beta.music.apple.com/assets/product/MissingArtworkMusic.svg";
|
|
// }
|
|
|
|
// });
|
|
}
|
|
},
|
|
getNowPlayingArtwork(size = 600) {
|
|
try {
|
|
if (this.mk["nowPlayingItem"]["attributes"]["artwork"]["url"]) {
|
|
return `url(${this.mk["nowPlayingItem"]["attributes"]["artwork"]["url"].replace('{w}', size).replace('{h}', size)})`;
|
|
} else {
|
|
return "";
|
|
}
|
|
} catch (e) {
|
|
return ""
|
|
// Does not work
|
|
// this.mk.api.library.song(this.mk.nowPlayingItem.id).then((data) => {
|
|
// try {
|
|
// if (data != null && data !== "") {
|
|
// return `url(${data["attributes"]["artwork"]["url"]})`;
|
|
// } else {
|
|
// return "url(https://beta.music.apple.com/assets/product/MissingArtworkMusic.svg)";
|
|
// }
|
|
// } catch (e) {
|
|
// return "url(https://beta.music.apple.com/assets/product/MissingArtworkMusic.svg)";
|
|
// }
|
|
|
|
// });
|
|
}
|
|
},
|
|
quickPlay(query) {
|
|
let self = this
|
|
MusicKit.getInstance().api.search(query, {limit: 2, types: 'songs'}).then(function (data) {
|
|
MusicKit.getInstance().setQueue({song: data["songs"]['data'][0]["id"]}).then(function (queue) {
|
|
MusicKit.getInstance().play()
|
|
setTimeout(() => {
|
|
self.$forceUpdate()
|
|
}, 1000)
|
|
})
|
|
})
|
|
},
|
|
apiCall(url, callback) {
|
|
const xmlHttp = new XMLHttpRequest();
|
|
|
|
xmlHttp.onreadystatechange = (e) => {
|
|
if (xmlHttp.readyState !== 4) {
|
|
return;
|
|
}
|
|
|
|
if (xmlHttp.status === 200) {
|
|
// console.log('SUCCESS', xmlHttp.responseText);
|
|
callback(JSON.parse(xmlHttp.responseText));
|
|
} else {
|
|
console.warn('request_error');
|
|
}
|
|
};
|
|
|
|
xmlHttp.open("GET", url);
|
|
xmlHttp.setRequestHeader("Authorization", "Bearer " + MusicKit.getInstance().developerToken);
|
|
xmlHttp.setRequestHeader("Music-User-Token", "" + MusicKit.getInstance().musicUserToken);
|
|
xmlHttp.setRequestHeader("Accept", "application/json");
|
|
xmlHttp.setRequestHeader("Content-Type", "application/json");
|
|
xmlHttp.responseType = "text";
|
|
xmlHttp.send();
|
|
},
|
|
fetchPlaylist(id, callback) {
|
|
// id can be found in playlist.attributes.playParams.globalId
|
|
this.mk.api.playlist(id).then(res => {
|
|
callback(res)
|
|
})
|
|
|
|
// tracks are found in relationship.data
|
|
},
|
|
windowFocus(val) {
|
|
if(val) {
|
|
document.querySelectorAll(".animated-artwork-video").forEach(el => {
|
|
el.play()
|
|
})
|
|
} else {
|
|
document.querySelectorAll(".animated-artwork-video").forEach(el => {
|
|
el.pause()
|
|
})
|
|
}
|
|
}
|
|
}
|
|
})
|
|
|
|
document.addEventListener('musickitloaded', function () {
|
|
// MusicKit global is now defined
|
|
function initMusicKit() {
|
|
let parsedJson = JSON.parse(this.responseText)
|
|
MusicKit.configure({
|
|
developerToken: parsedJson.Key,
|
|
app: {
|
|
name: 'My Cool Web App',
|
|
build: '1978.4.1'
|
|
}
|
|
});
|
|
setTimeout(() => {
|
|
app.init()
|
|
}, 1000)
|
|
}
|
|
|
|
const request = new XMLHttpRequest();
|
|
request.addEventListener("load", initMusicKit);
|
|
request.open("GET", "https://api.cider.sh/");
|
|
request.send();
|
|
});
|
|
|
|
function refreshFocus() {
|
|
if (document.hasFocus() == false) {
|
|
app.windowFocus(false)
|
|
} else {
|
|
app.windowFocus(true)
|
|
}
|
|
setTimeout(refreshFocus, 200);
|
|
}
|
|
|
|
refreshFocus();
|
|
|
|
function xmlToJson(xml) {
|
|
|
|
// Create the return object
|
|
var obj = {};
|
|
|
|
if (xml.nodeType == 1) { // element
|
|
// do attributes
|
|
if (xml.attributes.length > 0) {
|
|
obj["@attributes"] = {};
|
|
for (var j = 0; j < xml.attributes.length; j++) {
|
|
var attribute = xml.attributes.item(j);
|
|
obj["@attributes"][attribute.nodeName] = attribute.nodeValue;
|
|
}
|
|
}
|
|
} else if (xml.nodeType == 3) { // text
|
|
obj = xml.nodeValue;
|
|
}
|
|
|
|
// do children
|
|
if (xml.hasChildNodes()) {
|
|
for (var i = 0; i < xml.childNodes.length; i++) {
|
|
var item = xml.childNodes.item(i);
|
|
var nodeName = item.nodeName;
|
|
if (typeof (obj[nodeName]) == "undefined") {
|
|
obj[nodeName] = xmlToJson(item);
|
|
} else {
|
|
if (typeof (obj[nodeName].push) == "undefined") {
|
|
var old = obj[nodeName];
|
|
obj[nodeName] = [];
|
|
obj[nodeName].push(old);
|
|
}
|
|
obj[nodeName].push(xmlToJson(item));
|
|
}
|
|
}
|
|
}
|
|
console.log(obj);
|
|
return obj;
|
|
};
|
|
|
|
var checkIfScrollIsStatic = setInterval(() => {
|
|
try {
|
|
if (position === document.getElementsByClassName('lyric-body')[0].scrollTop) {
|
|
clearInterval(checkIfScrollIsStatic)
|
|
// do something
|
|
}
|
|
position = document.getElementsByClassName('lyric-body')[0].scrollTop
|
|
} catch (e) {
|
|
}
|
|
|
|
}, 50);
|