2022-02-23 00:33:55 +01:00
|
|
|
import LocalPlayer from './LocalPlayer'
|
|
|
|
import CastPlayer from './CastPlayer'
|
|
|
|
import AudioTrack from './AudioTrack'
|
|
|
|
|
|
|
|
export default class PlayerHandler {
|
|
|
|
constructor(ctx) {
|
|
|
|
this.ctx = ctx
|
|
|
|
this.audiobook = null
|
|
|
|
this.playWhenReady = false
|
|
|
|
this.player = null
|
|
|
|
this.playerState = 'IDLE'
|
|
|
|
this.currentStreamId = null
|
|
|
|
this.startTime = 0
|
|
|
|
|
|
|
|
this.lastSyncTime = 0
|
|
|
|
this.lastSyncedAt = 0
|
|
|
|
this.listeningTimeSinceSync = 0
|
|
|
|
|
|
|
|
this.playInterval = null
|
|
|
|
}
|
|
|
|
|
|
|
|
get isCasting() {
|
|
|
|
return this.ctx.$store.state.globals.isCasting
|
|
|
|
}
|
|
|
|
get isPlayingCastedAudiobook() {
|
|
|
|
return this.audiobook && (this.player instanceof CastPlayer)
|
|
|
|
}
|
|
|
|
get isPlayingLocalAudiobook() {
|
|
|
|
return this.audiobook && (this.player instanceof LocalPlayer)
|
|
|
|
}
|
|
|
|
get userToken() {
|
|
|
|
return this.ctx.$store.getters['user/getToken']
|
|
|
|
}
|
|
|
|
get playerPlaying() {
|
|
|
|
return this.playerState === 'PLAYING'
|
|
|
|
}
|
|
|
|
|
|
|
|
load(audiobook, playWhenReady, startTime = 0) {
|
|
|
|
if (!this.player) this.switchPlayer()
|
|
|
|
|
|
|
|
this.audiobook = audiobook
|
|
|
|
this.startTime = startTime
|
|
|
|
this.playWhenReady = playWhenReady
|
|
|
|
this.prepare()
|
|
|
|
}
|
|
|
|
|
|
|
|
switchPlayer() {
|
|
|
|
if (this.isCasting && !(this.player instanceof CastPlayer)) {
|
|
|
|
console.log('[PlayerHandler] Switching to cast player')
|
|
|
|
|
|
|
|
this.stopPlayInterval()
|
|
|
|
this.playerStateChange('LOADING')
|
|
|
|
|
|
|
|
this.startTime = this.player ? this.player.getCurrentTime() : this.startTime
|
|
|
|
if (this.player) {
|
|
|
|
this.player.destroy()
|
|
|
|
}
|
|
|
|
this.player = new CastPlayer(this.ctx)
|
|
|
|
this.setPlayerListeners()
|
|
|
|
|
|
|
|
if (this.audiobook) {
|
|
|
|
// Audiobook was already loaded - prepare for cast
|
|
|
|
this.playWhenReady = false
|
|
|
|
this.prepare()
|
|
|
|
}
|
|
|
|
} else if (!this.isCasting && !(this.player instanceof LocalPlayer)) {
|
|
|
|
console.log('[PlayerHandler] Switching to local player')
|
|
|
|
|
|
|
|
this.stopPlayInterval()
|
|
|
|
this.playerStateChange('LOADING')
|
|
|
|
|
|
|
|
if (this.player) {
|
|
|
|
this.player.destroy()
|
|
|
|
}
|
|
|
|
this.player = new LocalPlayer(this.ctx)
|
|
|
|
this.setPlayerListeners()
|
|
|
|
|
|
|
|
if (this.audiobook) {
|
|
|
|
// Audiobook was already loaded - prepare for local play
|
|
|
|
this.playWhenReady = false
|
|
|
|
this.prepare()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
setPlayerListeners() {
|
|
|
|
this.player.on('stateChange', this.playerStateChange.bind(this))
|
|
|
|
this.player.on('timeupdate', this.playerTimeupdate.bind(this))
|
|
|
|
this.player.on('buffertimeUpdate', this.playerBufferTimeUpdate.bind(this))
|
2022-03-05 22:37:30 +01:00
|
|
|
this.player.on('error', this.playerError.bind(this))
|
|
|
|
}
|
|
|
|
|
|
|
|
playerError() {
|
|
|
|
// Switch to HLS stream on error
|
|
|
|
if (!this.isCasting && !this.currentStreamId && (this.player instanceof LocalPlayer)) {
|
|
|
|
console.log(`[PlayerHandler] Audio player error switching to HLS stream`)
|
|
|
|
this.prepare(true)
|
|
|
|
}
|
2022-02-23 00:33:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
playerStateChange(state) {
|
|
|
|
console.log('[PlayerHandler] Player state change', state)
|
|
|
|
this.playerState = state
|
|
|
|
if (this.playerState === 'PLAYING') {
|
|
|
|
this.startPlayInterval()
|
|
|
|
} else {
|
|
|
|
this.stopPlayInterval()
|
|
|
|
}
|
|
|
|
if (this.playerState === 'LOADED' || this.playerState === 'PLAYING') {
|
|
|
|
this.ctx.setDuration(this.player.getDuration())
|
|
|
|
}
|
|
|
|
if (this.playerState !== 'LOADING') {
|
|
|
|
this.ctx.setCurrentTime(this.player.getCurrentTime())
|
|
|
|
}
|
|
|
|
|
|
|
|
this.ctx.isPlaying = this.playerState === 'PLAYING'
|
|
|
|
this.ctx.playerLoading = this.playerState === 'LOADING'
|
|
|
|
}
|
|
|
|
|
|
|
|
playerTimeupdate(time) {
|
|
|
|
this.ctx.setCurrentTime(time)
|
|
|
|
}
|
|
|
|
|
|
|
|
playerBufferTimeUpdate(buffertime) {
|
|
|
|
this.ctx.setBufferTime(buffertime)
|
|
|
|
}
|
|
|
|
|
2022-03-05 22:37:30 +01:00
|
|
|
async prepare(forceHls = false) {
|
|
|
|
var useHls = false
|
|
|
|
|
|
|
|
var runningTotal = 0
|
|
|
|
var audioTracks = (this.audiobook.tracks || []).map((track) => {
|
|
|
|
var audioTrack = new AudioTrack(track)
|
|
|
|
audioTrack.startOffset = runningTotal
|
|
|
|
audioTrack.contentUrl = `/lib/${this.audiobook.libraryId}/${this.audiobook.folderId}/${track.path}?token=${this.userToken}`
|
|
|
|
audioTrack.mimeType = this.getMimeTypeForTrack(track)
|
|
|
|
audioTrack.canDirectPlay = !!this.player.playableMimetypes[audioTrack.mimeType]
|
|
|
|
|
|
|
|
runningTotal += audioTrack.duration
|
|
|
|
return audioTrack
|
|
|
|
})
|
|
|
|
|
|
|
|
// All html5 audio player plays use HLS unless experimental features is on
|
|
|
|
if (!this.isCasting) {
|
|
|
|
if (forceHls || !this.ctx.showExperimentalFeatures) {
|
|
|
|
useHls = true
|
|
|
|
} else {
|
|
|
|
// Use HLS if any audio track cannot be direct played
|
|
|
|
useHls = !!audioTracks.find(at => !at.canDirectPlay)
|
|
|
|
|
|
|
|
if (useHls) {
|
|
|
|
console.warn(`[PlayerHandler] An audio track cannot be direct played`, audioTracks.find(at => !at.canDirectPlay))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-02-23 00:33:55 +01:00
|
|
|
if (useHls) {
|
|
|
|
var stream = await this.ctx.$axios.$get(`/api/books/${this.audiobook.id}/stream`).catch((error) => {
|
|
|
|
console.error('Failed to start stream', error)
|
|
|
|
})
|
|
|
|
if (stream) {
|
|
|
|
console.log(`[PlayerHandler] prepare hls stream`, stream)
|
|
|
|
this.setHlsStream(stream)
|
2022-03-05 22:37:30 +01:00
|
|
|
} else {
|
|
|
|
console.error(`[PlayerHandler] Failed to start HLS stream`)
|
2022-02-23 00:33:55 +01:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
this.setDirectPlay(audioTracks)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-05 22:37:30 +01:00
|
|
|
getMimeTypeForTrack(track) {
|
|
|
|
var ext = track.ext
|
|
|
|
if (ext === '.mp3' || ext === '.m4b' || ext === '.m4a') {
|
|
|
|
return 'audio/mpeg'
|
|
|
|
} else if (ext === '.mp4') {
|
|
|
|
return 'audio/mp4'
|
|
|
|
} else if (ext === '.ogg') {
|
|
|
|
return 'audio/ogg'
|
|
|
|
} else if (ext === '.aac' || ext === '.m4p') {
|
|
|
|
return 'audio/aac'
|
|
|
|
} else if (ext === '.flac') {
|
|
|
|
return 'audio/flac'
|
|
|
|
}
|
|
|
|
return 'audio/mpeg'
|
|
|
|
}
|
|
|
|
|
2022-02-23 00:33:55 +01:00
|
|
|
closePlayer() {
|
|
|
|
console.log('[PlayerHandler] CLose Player')
|
|
|
|
if (this.player) {
|
|
|
|
this.player.destroy()
|
|
|
|
}
|
|
|
|
this.player = null
|
|
|
|
this.playerState = 'IDLE'
|
|
|
|
this.audiobook = null
|
|
|
|
this.currentStreamId = null
|
|
|
|
this.startTime = 0
|
|
|
|
this.stopPlayInterval()
|
|
|
|
}
|
|
|
|
|
|
|
|
prepareStream(stream) {
|
|
|
|
if (!this.player) this.switchPlayer()
|
|
|
|
this.audiobook = stream.audiobook
|
|
|
|
this.setHlsStream({
|
|
|
|
streamId: stream.id,
|
|
|
|
streamUrl: stream.clientPlaylistUri,
|
|
|
|
startTime: stream.clientCurrentTime
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
setHlsStream(stream) {
|
|
|
|
this.currentStreamId = stream.streamId
|
|
|
|
var audioTrack = new AudioTrack({
|
|
|
|
duration: this.audiobook.duration,
|
|
|
|
contentUrl: stream.streamUrl + '?token=' + this.userToken,
|
|
|
|
mimeType: 'application/vnd.apple.mpegurl'
|
|
|
|
})
|
|
|
|
this.startTime = stream.startTime
|
|
|
|
this.ctx.playerLoading = true
|
|
|
|
this.player.set(this.audiobook, [audioTrack], this.currentStreamId, stream.startTime, this.playWhenReady)
|
|
|
|
}
|
|
|
|
|
|
|
|
setDirectPlay(audioTracks) {
|
|
|
|
this.currentStreamId = null
|
|
|
|
this.ctx.playerLoading = true
|
|
|
|
this.player.set(this.audiobook, audioTracks, null, this.startTime, this.playWhenReady)
|
|
|
|
}
|
|
|
|
|
|
|
|
resetStream(startTime, streamId) {
|
|
|
|
if (this.currentStreamId === streamId) {
|
|
|
|
this.player.resetStream(startTime)
|
|
|
|
} else {
|
|
|
|
console.warn('resetStream mismatch streamId', this.currentStreamId, streamId)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
startPlayInterval() {
|
|
|
|
clearInterval(this.playInterval)
|
|
|
|
var lastTick = Date.now()
|
|
|
|
this.playInterval = setInterval(() => {
|
|
|
|
// Update UI
|
|
|
|
if (!this.player) return
|
|
|
|
var currentTime = this.player.getCurrentTime()
|
|
|
|
this.ctx.setCurrentTime(currentTime)
|
|
|
|
|
|
|
|
var exactTimeElapsed = ((Date.now() - lastTick) / 1000)
|
|
|
|
lastTick = Date.now()
|
|
|
|
this.listeningTimeSinceSync += exactTimeElapsed
|
|
|
|
if (this.listeningTimeSinceSync >= 5) {
|
|
|
|
this.sendProgressSync(currentTime)
|
|
|
|
this.listeningTimeSinceSync = 0
|
|
|
|
}
|
|
|
|
}, 1000)
|
|
|
|
}
|
|
|
|
|
|
|
|
sendProgressSync(currentTime) {
|
|
|
|
var diffSinceLastSync = Math.abs(this.lastSyncTime - currentTime)
|
|
|
|
if (diffSinceLastSync < 1) return
|
|
|
|
|
|
|
|
this.lastSyncTime = currentTime
|
|
|
|
if (this.currentStreamId) { // Updating stream progress (HLS stream)
|
|
|
|
var listeningTimeToAdd = Math.max(0, Math.floor(this.listeningTimeSinceSync))
|
|
|
|
var syncData = {
|
|
|
|
timeListened: listeningTimeToAdd,
|
|
|
|
currentTime,
|
|
|
|
streamId: this.currentStreamId,
|
|
|
|
audiobookId: this.audiobook.id
|
|
|
|
}
|
|
|
|
this.ctx.$axios.$post('/api/syncStream', syncData, { timeout: 1000 }).catch((error) => {
|
|
|
|
console.error('Failed to update stream progress', error)
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
// Direct play via chromecast does not yet have backend stream session model
|
|
|
|
// so the progress update for the audiobook is updated this way (instead of through the stream)
|
|
|
|
var duration = this.getDuration()
|
|
|
|
var syncData = {
|
|
|
|
totalDuration: duration,
|
|
|
|
currentTime,
|
|
|
|
progress: duration > 0 ? currentTime / duration : 0,
|
|
|
|
isRead: false,
|
|
|
|
audiobookId: this.audiobook.id,
|
|
|
|
lastUpdate: Date.now()
|
|
|
|
}
|
|
|
|
this.ctx.$axios.$post('/api/syncLocal', syncData, { timeout: 1000 }).catch((error) => {
|
|
|
|
console.error('Failed to update local progress', error)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
stopPlayInterval() {
|
|
|
|
clearInterval(this.playInterval)
|
|
|
|
this.playInterval = null
|
|
|
|
}
|
|
|
|
|
|
|
|
playPause() {
|
|
|
|
if (this.player) this.player.playPause()
|
|
|
|
}
|
|
|
|
|
|
|
|
play() {
|
2022-03-05 19:30:46 +01:00
|
|
|
if (this.player) this.player.play()
|
2022-02-23 00:33:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
pause() {
|
|
|
|
if (this.player) this.player.pause()
|
|
|
|
}
|
|
|
|
|
|
|
|
getCurrentTime() {
|
|
|
|
return this.player ? this.player.getCurrentTime() : 0
|
|
|
|
}
|
|
|
|
|
|
|
|
getDuration() {
|
|
|
|
return this.player ? this.player.getDuration() : 0
|
|
|
|
}
|
|
|
|
|
|
|
|
jumpBackward() {
|
|
|
|
if (!this.player) return
|
|
|
|
var currentTime = this.getCurrentTime()
|
|
|
|
this.seek(Math.max(0, currentTime - 10))
|
|
|
|
}
|
|
|
|
|
|
|
|
jumpForward() {
|
|
|
|
if (!this.player) return
|
|
|
|
var currentTime = this.getCurrentTime()
|
|
|
|
this.seek(Math.min(currentTime + 10, this.getDuration()))
|
|
|
|
}
|
|
|
|
|
|
|
|
setVolume(volume) {
|
|
|
|
if (!this.player) return
|
|
|
|
this.player.setVolume(volume)
|
|
|
|
}
|
|
|
|
|
|
|
|
setPlaybackRate(playbackRate) {
|
|
|
|
if (!this.player) return
|
|
|
|
this.player.setPlaybackRate(playbackRate)
|
|
|
|
}
|
|
|
|
|
|
|
|
seek(time) {
|
|
|
|
if (!this.player) return
|
|
|
|
this.player.seek(time, this.playerPlaying)
|
|
|
|
this.ctx.setCurrentTime(time)
|
|
|
|
|
|
|
|
// Update progress if paused
|
|
|
|
if (!this.playerPlaying) {
|
|
|
|
this.sendProgressSync(time)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|