2016-11-26 03:25:35 +00:00
|
|
|
|
import { shuffle, orderBy } from 'lodash'
|
|
|
|
|
import plyr from 'plyr'
|
|
|
|
|
import Vue from 'vue'
|
2017-03-26 09:02:03 +00:00
|
|
|
|
import isMobile from 'ismobilejs'
|
2015-12-13 04:42:28 +00:00
|
|
|
|
|
2017-05-07 17:41:12 +00:00
|
|
|
|
import { event, isMediaSessionSupported } from '../utils'
|
2016-11-26 03:25:35 +00:00
|
|
|
|
import { queueStore, sharedStore, userStore, songStore, preferenceStore as preferences } from '../stores'
|
2017-08-25 00:38:02 +00:00
|
|
|
|
import { socket } from '../services'
|
2016-11-26 03:25:35 +00:00
|
|
|
|
import config from '../config'
|
|
|
|
|
import router from '../router'
|
2015-12-13 04:42:28 +00:00
|
|
|
|
|
2016-06-25 10:15:57 +00:00
|
|
|
|
export const playback = {
|
2016-06-25 16:05:24 +00:00
|
|
|
|
player: null,
|
2016-12-20 15:44:47 +00:00
|
|
|
|
volumeInput: null,
|
2016-06-25 16:05:24 +00:00
|
|
|
|
repeatModes: ['NO_REPEAT', 'REPEAT_ALL', 'REPEAT_ONE'],
|
|
|
|
|
initialized: false,
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Initialize the playback service for this whole Koel app.
|
|
|
|
|
*/
|
2016-11-26 03:25:35 +00:00
|
|
|
|
init () {
|
2016-06-25 16:05:24 +00:00
|
|
|
|
// We don't need to init this service twice, or the media events will be duplicated.
|
|
|
|
|
if (this.initialized) {
|
2016-11-26 03:25:35 +00:00
|
|
|
|
return
|
2016-06-25 16:05:24 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this.player = plyr.setup({
|
2016-11-26 03:25:35 +00:00
|
|
|
|
controls: []
|
|
|
|
|
})[0]
|
2016-06-25 16:05:24 +00:00
|
|
|
|
|
2016-12-20 15:44:47 +00:00
|
|
|
|
this.audio = document.querySelector('audio')
|
|
|
|
|
this.volumeInput = document.getElementById('volumeRange')
|
2015-12-13 04:42:28 +00:00
|
|
|
|
|
2017-08-06 09:55:02 +00:00
|
|
|
|
const player = document.querySelector('.plyr')
|
|
|
|
|
|
2015-12-13 04:42:28 +00:00
|
|
|
|
/**
|
2016-06-25 16:05:24 +00:00
|
|
|
|
* Listen to 'error' event on the audio player and play the next song if any.
|
2015-12-13 04:42:28 +00:00
|
|
|
|
*/
|
2017-08-06 09:55:02 +00:00
|
|
|
|
player.addEventListener('error', () => this.playNext(), true)
|
2015-12-13 04:42:28 +00:00
|
|
|
|
|
|
|
|
|
/**
|
2016-06-25 16:05:24 +00:00
|
|
|
|
* Listen to 'ended' event on the audio player and play the next song in the queue.
|
2015-12-13 04:42:28 +00:00
|
|
|
|
*/
|
2017-08-06 09:55:02 +00:00
|
|
|
|
player.addEventListener('ended', e => {
|
2016-06-27 06:11:35 +00:00
|
|
|
|
if (sharedStore.state.useLastfm && userStore.current.preferences.lastfm_session_key) {
|
2016-11-26 03:25:35 +00:00
|
|
|
|
songStore.scrobble(queueStore.current)
|
2016-06-27 06:11:35 +00:00
|
|
|
|
}
|
2015-12-13 04:42:28 +00:00
|
|
|
|
|
2017-05-07 17:41:12 +00:00
|
|
|
|
preferences.repeatMode === 'REPEAT_ONE' ? this.restart() : this.playNext()
|
2016-11-26 03:25:35 +00:00
|
|
|
|
})
|
2015-12-13 04:42:28 +00:00
|
|
|
|
|
|
|
|
|
/**
|
2017-03-22 07:44:04 +00:00
|
|
|
|
* Attempt to preload the next song.
|
2015-12-13 04:42:28 +00:00
|
|
|
|
*/
|
2017-08-06 09:55:02 +00:00
|
|
|
|
player.addEventListener('canplaythrough', e => {
|
2016-11-26 03:25:35 +00:00
|
|
|
|
const nextSong = queueStore.next
|
2017-03-26 09:02:03 +00:00
|
|
|
|
if (!nextSong || nextSong.preloaded || (isMobile.any && preferences.transcodeOnMobile)) {
|
|
|
|
|
// Don't preload if
|
|
|
|
|
// - there's no next song
|
|
|
|
|
// - next song has already been preloaded
|
2017-03-27 10:05:43 +00:00
|
|
|
|
// - we're on mobile and "transcode" option is checked
|
2016-11-26 03:25:35 +00:00
|
|
|
|
return
|
2016-06-25 16:05:24 +00:00
|
|
|
|
}
|
2016-03-14 12:47:45 +00:00
|
|
|
|
|
2017-03-22 07:44:04 +00:00
|
|
|
|
const audio = document.createElement('audio')
|
|
|
|
|
audio.setAttribute('src', songStore.getSourceUrl(nextSong))
|
|
|
|
|
audio.setAttribute('preload', 'auto')
|
|
|
|
|
audio.load()
|
2016-11-26 03:25:35 +00:00
|
|
|
|
nextSong.preloaded = true
|
|
|
|
|
})
|
2015-12-13 04:42:28 +00:00
|
|
|
|
|
2017-08-06 09:55:02 +00:00
|
|
|
|
player.addEventListener('timeupdate', e => {
|
|
|
|
|
const song = queueStore.current
|
|
|
|
|
|
|
|
|
|
if (this.player.media.currentTime > 10 && !song.registeredPlayCount) {
|
|
|
|
|
// After 10 seconds, register a play count and add it into "recently played" list
|
|
|
|
|
songStore.addRecentlyPlayed(song)
|
|
|
|
|
songStore.registerPlay(song)
|
|
|
|
|
|
|
|
|
|
song.registeredPlayCount = true
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
|
2016-06-25 16:05:24 +00:00
|
|
|
|
// On init, set the volume to the value found in the local storage.
|
2016-11-26 03:25:35 +00:00
|
|
|
|
this.setVolume(preferences.volume)
|
2016-06-25 16:05:24 +00:00
|
|
|
|
|
|
|
|
|
// Init the equalizer if supported.
|
2016-11-26 03:25:35 +00:00
|
|
|
|
event.emit('equalizer:init', this.player.media)
|
2016-06-25 16:05:24 +00:00
|
|
|
|
|
2017-05-07 17:41:12 +00:00
|
|
|
|
if (isMediaSessionSupported()) {
|
2017-03-26 03:46:17 +00:00
|
|
|
|
navigator.mediaSession.setActionHandler('play', () => this.resume())
|
|
|
|
|
navigator.mediaSession.setActionHandler('pause', () => this.pause())
|
2017-03-26 03:53:04 +00:00
|
|
|
|
navigator.mediaSession.setActionHandler('previoustrack', () => this.playPrev())
|
|
|
|
|
navigator.mediaSession.setActionHandler('nexttrack', () => this.playNext())
|
2017-03-26 03:25:56 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-08-26 21:42:36 +00:00
|
|
|
|
socket.listen('playback:toggle', () => this.toggle())
|
|
|
|
|
.listen('playback:next', () => this.playNext())
|
|
|
|
|
.listen('playback:prev', () => this.playPrev())
|
2017-09-08 22:15:27 +00:00
|
|
|
|
.listen('status:get', () => {
|
|
|
|
|
const data = queueStore.current ? songStore.generateDataToBroadcast(queueStore.current) : {}
|
|
|
|
|
data.volume = this.volumeInput.value
|
|
|
|
|
socket.broadcast('status', data)
|
|
|
|
|
})
|
2017-08-26 21:42:36 +00:00
|
|
|
|
.listen('song:getcurrent', () => {
|
|
|
|
|
socket.broadcast(
|
2017-08-28 11:13:00 +00:00
|
|
|
|
'song',
|
|
|
|
|
queueStore.current
|
2017-08-27 15:53:17 +00:00
|
|
|
|
? songStore.generateDataToBroadcast(queueStore.current)
|
2017-08-26 21:42:36 +00:00
|
|
|
|
: { song: null }
|
|
|
|
|
)
|
|
|
|
|
})
|
2017-09-08 22:15:27 +00:00
|
|
|
|
.listen('volume:set', ({ volume }) => this.setVolume(volume))
|
2017-08-26 21:42:36 +00:00
|
|
|
|
|
2016-11-26 03:25:35 +00:00
|
|
|
|
this.initialized = true
|
2016-06-25 16:05:24 +00:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Play a song. Because
|
|
|
|
|
*
|
|
|
|
|
* So many adventures couldn't happen today,
|
|
|
|
|
* So many songs we forgot to play
|
|
|
|
|
* So many dreams swinging out of the blue
|
|
|
|
|
* We'll let them come true
|
|
|
|
|
*
|
|
|
|
|
* @param {Object} song The song to play
|
|
|
|
|
*/
|
2016-11-26 03:25:35 +00:00
|
|
|
|
play (song) {
|
2016-06-25 16:05:24 +00:00
|
|
|
|
if (!song) {
|
2016-11-26 03:25:35 +00:00
|
|
|
|
return
|
2016-06-25 16:05:24 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (queueStore.current) {
|
2016-11-26 03:25:35 +00:00
|
|
|
|
queueStore.current.playbackState = 'stopped'
|
2016-06-25 16:05:24 +00:00
|
|
|
|
}
|
|
|
|
|
|
2016-11-26 03:25:35 +00:00
|
|
|
|
song.playbackState = 'playing'
|
2016-06-25 16:05:24 +00:00
|
|
|
|
|
|
|
|
|
// Set the song as the current song
|
2016-11-26 03:25:35 +00:00
|
|
|
|
queueStore.current = song
|
2016-06-25 16:05:24 +00:00
|
|
|
|
|
|
|
|
|
// Manually set the `src` attribute of the audio to prevent plyr from resetting
|
|
|
|
|
// the audio media object and cause our equalizer to malfunction.
|
2016-11-26 03:25:35 +00:00
|
|
|
|
this.player.media.src = songStore.getSourceUrl(song)
|
2016-06-25 16:05:24 +00:00
|
|
|
|
|
2016-12-20 15:44:47 +00:00
|
|
|
|
document.title = `${song.title} ♫ ${config.appTitle}`
|
|
|
|
|
document.querySelector('.plyr audio').setAttribute('title', `${song.artist.name} - ${song.title}`)
|
2016-06-25 16:05:24 +00:00
|
|
|
|
|
|
|
|
|
// We'll just "restart" playing the song, which will handle notification, scrobbling etc.
|
2016-11-26 03:25:35 +00:00
|
|
|
|
this.restart()
|
2016-06-25 16:05:24 +00:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
/**
|
2017-08-06 09:55:02 +00:00
|
|
|
|
* Show the "now playing" notification for a song.
|
|
|
|
|
*
|
2017-08-28 11:13:00 +00:00
|
|
|
|
* @param {Object} song
|
2016-06-25 16:05:24 +00:00
|
|
|
|
*/
|
2017-08-06 09:55:02 +00:00
|
|
|
|
showNotification (song) {
|
2016-06-25 16:05:24 +00:00
|
|
|
|
// Show the notification if we're allowed to
|
|
|
|
|
if (!window.Notification || !preferences.notify) {
|
2016-11-26 03:25:35 +00:00
|
|
|
|
return
|
2016-06-25 16:05:24 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
try {
|
2016-11-26 03:25:35 +00:00
|
|
|
|
const notif = new window.Notification(`♫ ${song.title}`, {
|
2016-06-25 16:05:24 +00:00
|
|
|
|
icon: song.album.cover,
|
|
|
|
|
body: `${song.album.name} – ${song.artist.name}`
|
2016-11-26 03:25:35 +00:00
|
|
|
|
})
|
2016-06-25 16:05:24 +00:00
|
|
|
|
|
2016-11-26 03:25:35 +00:00
|
|
|
|
notif.onclick = () => window.focus()
|
2016-06-25 16:05:24 +00:00
|
|
|
|
|
|
|
|
|
// Close the notif after 5 secs.
|
2016-11-26 03:25:35 +00:00
|
|
|
|
window.setTimeout(() => notif.close(), 5000)
|
2016-06-25 16:05:24 +00:00
|
|
|
|
} catch (e) {
|
|
|
|
|
// Notification fails.
|
|
|
|
|
// @link https://developer.mozilla.org/en-US/docs/Web/API/ServiceWorkerRegistration/showNotification
|
|
|
|
|
}
|
2017-03-26 03:25:56 +00:00
|
|
|
|
|
|
|
|
|
if ('mediaSession' in navigator) {
|
2017-03-26 08:13:13 +00:00
|
|
|
|
/* global MediaMetadata */
|
2017-03-26 03:25:56 +00:00
|
|
|
|
navigator.mediaSession.metadata = new MediaMetadata({
|
|
|
|
|
title: song.title,
|
|
|
|
|
artist: song.artist.name,
|
|
|
|
|
album: song.album.name,
|
|
|
|
|
artwork: [
|
|
|
|
|
{ src: song.album.cover, sizes: '256x256', type: 'image/png' }
|
|
|
|
|
]
|
|
|
|
|
})
|
|
|
|
|
}
|
2016-06-25 16:05:24 +00:00
|
|
|
|
},
|
|
|
|
|
|
2017-08-06 09:55:02 +00:00
|
|
|
|
/**
|
|
|
|
|
* Restart playing a song.
|
|
|
|
|
*/
|
|
|
|
|
restart () {
|
|
|
|
|
const song = queueStore.current
|
|
|
|
|
|
|
|
|
|
this.showNotification(song)
|
|
|
|
|
|
|
|
|
|
// Record the UNIX timestamp the song start playing, for scrobbling purpose
|
|
|
|
|
song.playStartTime = Math.floor(Date.now() / 1000)
|
|
|
|
|
|
|
|
|
|
song.registeredPlayCount = false
|
|
|
|
|
|
|
|
|
|
event.emit('song:played', song)
|
|
|
|
|
|
2017-08-27 15:53:17 +00:00
|
|
|
|
socket.broadcast('song', songStore.generateDataToBroadcast(song))
|
2017-08-25 00:38:02 +00:00
|
|
|
|
|
2017-08-06 09:55:02 +00:00
|
|
|
|
this.player.restart()
|
|
|
|
|
this.player.play()
|
|
|
|
|
},
|
|
|
|
|
|
2016-06-25 16:05:24 +00:00
|
|
|
|
/**
|
|
|
|
|
* The next song in the queue.
|
|
|
|
|
* If we're in REPEAT_ALL mode and there's no next song, just get the first song.
|
|
|
|
|
*
|
|
|
|
|
* @return {Object} The song
|
|
|
|
|
*/
|
2016-11-26 03:25:35 +00:00
|
|
|
|
get next () {
|
2017-05-07 17:41:12 +00:00
|
|
|
|
if (queueStore.next) {
|
|
|
|
|
return queueStore.next
|
2016-06-25 16:05:24 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (preferences.repeatMode === 'REPEAT_ALL') {
|
2016-11-26 03:25:35 +00:00
|
|
|
|
return queueStore.first
|
2016-06-25 16:05:24 +00:00
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* The previous song in the queue.
|
|
|
|
|
* If we're in REPEAT_ALL mode and there's no prev song, get the last song.
|
|
|
|
|
*
|
|
|
|
|
* @return {Object} The song
|
|
|
|
|
*/
|
2016-11-26 03:25:35 +00:00
|
|
|
|
get previous () {
|
2017-05-07 17:41:12 +00:00
|
|
|
|
if (queueStore.previous) {
|
|
|
|
|
return queueStore.previous
|
2016-06-25 16:05:24 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (preferences.repeatMode === 'REPEAT_ALL') {
|
2016-11-26 03:25:35 +00:00
|
|
|
|
return queueStore.last
|
2016-06-25 16:05:24 +00:00
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Circle through the repeat mode.
|
|
|
|
|
* The selected mode will be stored into local storage as well.
|
|
|
|
|
*/
|
2016-11-26 03:25:35 +00:00
|
|
|
|
changeRepeatMode () {
|
2017-01-14 13:09:38 +00:00
|
|
|
|
let index = this.repeatModes.indexOf(preferences.repeatMode) + 1
|
2016-06-25 16:05:24 +00:00
|
|
|
|
|
2017-01-14 13:09:38 +00:00
|
|
|
|
if (index >= this.repeatModes.length) {
|
|
|
|
|
index = 0
|
2016-06-25 16:05:24 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-01-14 13:09:38 +00:00
|
|
|
|
preferences.repeatMode = this.repeatModes[index]
|
2016-06-25 16:05:24 +00:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Play the prev song in the queue, if one is found.
|
|
|
|
|
* If the prev song is not found and the current mode is NO_REPEAT, we stop completely.
|
|
|
|
|
*/
|
2016-11-26 03:25:35 +00:00
|
|
|
|
playPrev () {
|
2016-06-25 16:05:24 +00:00
|
|
|
|
// If the song's duration is greater than 5 seconds and we've passed 5 seconds into it,
|
|
|
|
|
// restart playing instead.
|
|
|
|
|
if (this.player.media.currentTime > 5 && queueStore.current.length > 5) {
|
2016-11-26 03:25:35 +00:00
|
|
|
|
this.player.restart()
|
2016-06-25 16:05:24 +00:00
|
|
|
|
|
2016-11-26 03:25:35 +00:00
|
|
|
|
return
|
2016-06-25 16:05:24 +00:00
|
|
|
|
}
|
|
|
|
|
|
2016-11-26 03:25:35 +00:00
|
|
|
|
const prev = this.previous
|
2017-05-07 17:41:12 +00:00
|
|
|
|
!prev && preferences.repeatMode === 'NO_REPEAT'
|
|
|
|
|
? this.stop()
|
|
|
|
|
: this.play(prev)
|
2016-06-25 16:05:24 +00:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Play the next song in the queue, if one is found.
|
|
|
|
|
* If the next song is not found and the current mode is NO_REPEAT, we stop completely.
|
|
|
|
|
*/
|
2016-11-26 03:25:35 +00:00
|
|
|
|
playNext () {
|
|
|
|
|
const next = this.next
|
2017-05-07 17:41:12 +00:00
|
|
|
|
!next && preferences.repeatMode === 'NO_REPEAT'
|
|
|
|
|
? this.stop() // Nothing lasts forever, even cold November rain.
|
|
|
|
|
: this.play(next)
|
2016-06-25 16:05:24 +00:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Set the volume level.
|
|
|
|
|
*
|
|
|
|
|
* @param {Number} volume 0-10
|
|
|
|
|
* @param {Boolean=true} persist Whether the volume should be saved into local storage
|
|
|
|
|
*/
|
2016-11-26 03:25:35 +00:00
|
|
|
|
setVolume (volume, persist = true) {
|
|
|
|
|
this.player.setVolume(volume)
|
2016-06-25 16:05:24 +00:00
|
|
|
|
|
|
|
|
|
if (persist) {
|
2016-11-26 03:25:35 +00:00
|
|
|
|
preferences.volume = volume
|
2016-06-25 16:05:24 +00:00
|
|
|
|
}
|
|
|
|
|
|
2016-12-20 15:44:47 +00:00
|
|
|
|
this.volumeInput.value = volume
|
2016-06-25 16:05:24 +00:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Mute playback.
|
|
|
|
|
*/
|
2016-11-26 03:25:35 +00:00
|
|
|
|
mute () {
|
|
|
|
|
this.setVolume(0, false)
|
2016-06-25 16:05:24 +00:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Unmute playback.
|
|
|
|
|
*/
|
2016-11-26 03:25:35 +00:00
|
|
|
|
unmute () {
|
2016-06-25 16:05:24 +00:00
|
|
|
|
// If the saved volume is 0, we unmute to the default level (7).
|
|
|
|
|
if (preferences.volume === '0' || preferences.volume === 0) {
|
2016-11-26 03:25:35 +00:00
|
|
|
|
preferences.volume = 7
|
2016-06-25 16:05:24 +00:00
|
|
|
|
}
|
|
|
|
|
|
2016-11-26 03:25:35 +00:00
|
|
|
|
this.setVolume(preferences.volume)
|
2016-06-25 16:05:24 +00:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Completely stop playback.
|
|
|
|
|
*/
|
2016-11-26 03:25:35 +00:00
|
|
|
|
stop () {
|
2016-12-20 15:44:47 +00:00
|
|
|
|
document.title = config.appTitle
|
2016-11-26 03:25:35 +00:00
|
|
|
|
this.player.pause()
|
|
|
|
|
this.player.seek(0)
|
2016-06-25 16:05:24 +00:00
|
|
|
|
|
|
|
|
|
if (queueStore.current) {
|
2016-11-26 03:25:35 +00:00
|
|
|
|
queueStore.current.playbackState = 'stopped'
|
2016-06-25 16:05:24 +00:00
|
|
|
|
}
|
2017-08-27 15:53:17 +00:00
|
|
|
|
|
|
|
|
|
socket.broadcast('playback:stopped')
|
2016-06-25 16:05:24 +00:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Pause playback.
|
|
|
|
|
*/
|
2016-11-26 03:25:35 +00:00
|
|
|
|
pause () {
|
|
|
|
|
this.player.pause()
|
|
|
|
|
queueStore.current.playbackState = 'paused'
|
2017-08-27 15:53:17 +00:00
|
|
|
|
socket.broadcast('song', songStore.generateDataToBroadcast(queueStore.current))
|
2016-06-25 16:05:24 +00:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Resume playback.
|
|
|
|
|
*/
|
2016-11-26 03:25:35 +00:00
|
|
|
|
resume () {
|
|
|
|
|
this.player.play()
|
|
|
|
|
queueStore.current.playbackState = 'playing'
|
|
|
|
|
event.emit('song:played', queueStore.current)
|
2017-08-27 15:53:17 +00:00
|
|
|
|
socket.broadcast('song', songStore.generateDataToBroadcast(queueStore.current))
|
2016-06-25 16:05:24 +00:00
|
|
|
|
},
|
|
|
|
|
|
2017-08-26 21:42:36 +00:00
|
|
|
|
/**
|
|
|
|
|
* Toggle playback.
|
|
|
|
|
*/
|
|
|
|
|
toggle () {
|
|
|
|
|
if (!queueStore.current) {
|
|
|
|
|
this.playFirstInQueue()
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (queueStore.current.playbackState !== 'playing') {
|
|
|
|
|
this.resume()
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this.pause()
|
|
|
|
|
},
|
|
|
|
|
|
2016-06-25 16:05:24 +00:00
|
|
|
|
/**
|
|
|
|
|
* Queue up songs (replace them into the queue) and start playing right away.
|
|
|
|
|
*
|
|
|
|
|
* @param {?Array.<Object>} songs An array of song objects. Defaults to all songs if null.
|
|
|
|
|
* @param {Boolean=false} shuffled Whether to shuffle the songs before playing.
|
|
|
|
|
*/
|
2016-11-26 03:25:35 +00:00
|
|
|
|
queueAndPlay (songs = null, shuffled = false) {
|
2016-06-25 16:05:24 +00:00
|
|
|
|
if (!songs) {
|
2016-11-26 03:25:35 +00:00
|
|
|
|
songs = songStore.all
|
2016-06-25 16:05:24 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!songs.length) {
|
2016-11-26 03:25:35 +00:00
|
|
|
|
return
|
2016-06-25 16:05:24 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (shuffled) {
|
2016-11-26 03:25:35 +00:00
|
|
|
|
songs = shuffle(songs)
|
2016-06-25 16:05:24 +00:00
|
|
|
|
}
|
|
|
|
|
|
2016-11-26 03:25:35 +00:00
|
|
|
|
queueStore.queue(songs, true)
|
2016-06-25 16:05:24 +00:00
|
|
|
|
|
|
|
|
|
// Wrap this inside a nextTick() to wait for the DOM to complete updating
|
|
|
|
|
// and then play the first song in the queue.
|
2016-07-10 17:55:20 +00:00
|
|
|
|
Vue.nextTick(() => {
|
2016-11-26 03:25:35 +00:00
|
|
|
|
router.go('queue')
|
|
|
|
|
this.play(queueStore.first)
|
|
|
|
|
})
|
2016-06-25 16:05:24 +00:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Play the first song in the queue.
|
|
|
|
|
* If the current queue is empty, try creating it by shuffling all songs.
|
|
|
|
|
*/
|
2016-11-26 03:25:35 +00:00
|
|
|
|
playFirstInQueue () {
|
2017-05-07 17:41:12 +00:00
|
|
|
|
queueStore.all.length ? this.play(queueStore.first) : this.queueAndPlay()
|
2016-06-25 16:05:24 +00:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Play all songs by an artist.
|
|
|
|
|
*
|
|
|
|
|
* @param {Object} artist The artist object
|
|
|
|
|
* @param {Boolean=true} shuffled Whether to shuffle the songs
|
|
|
|
|
*/
|
2017-05-07 17:41:12 +00:00
|
|
|
|
playAllByArtist ({ songs }, shuffled = true) {
|
|
|
|
|
shuffled
|
|
|
|
|
? this.queueAndPlay(songs, true)
|
|
|
|
|
: this.queueAndPlay(orderBy(songs, 'album_id', 'track'))
|
2016-06-25 16:05:24 +00:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Play all songs in an album.
|
|
|
|
|
*
|
|
|
|
|
* @param {Object} album The album object
|
|
|
|
|
* @param {Boolean=true} shuffled Whether to shuffle the songs
|
|
|
|
|
*/
|
2017-05-07 17:41:12 +00:00
|
|
|
|
playAllInAlbum ({ songs }, shuffled = true) {
|
|
|
|
|
shuffled
|
|
|
|
|
? this.queueAndPlay(songs, true)
|
|
|
|
|
: this.queueAndPlay(orderBy(songs, 'track'))
|
2016-11-26 03:25:35 +00:00
|
|
|
|
}
|
|
|
|
|
}
|