import LocalPlayer from './LocalPlayer' import CastPlayer from './CastPlayer' import AudioTrack from './AudioTrack' export default class PlayerHandler { constructor(ctx) { this.ctx = ctx this.libraryItem = 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 isPlayingCastedItem() { return this.libraryItem && (this.player instanceof CastPlayer) } get isPlayingLocalItem() { return this.libraryItem && (this.player instanceof LocalPlayer) } get userToken() { return this.ctx.$store.getters['user/getToken'] } get playerPlaying() { return this.playerState === 'PLAYING' } load(libraryItem, playWhenReady, startTime = 0) { if (!this.player) this.switchPlayer() this.libraryItem = libraryItem 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.libraryItem) { // libraryItem 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.libraryItem) { // libraryItem 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)) 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) } } 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) } async prepare(forceHls = false) { var useHls = false var runningTotal = 0 var audioTracks = (this.libraryItem.media.tracks || []).map((track) => { if (!track.metadata) { console.error('INVALID TRACK', track) return null } var audioTrack = new AudioTrack(track) audioTrack.startOffset = runningTotal audioTrack.contentUrl = `/s/item/${this.libraryItem.id}/${this.ctx.$encodeUriPath(track.metadata.relPath.replace(/^\//, ''))}?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)) } } } if (useHls) { var stream = await this.ctx.$axios.$get(`/api/items/${this.libraryItem.id}/stream`).catch((error) => { console.error('Failed to start stream', error) }) if (stream) { console.log(`[PlayerHandler] prepare hls stream`, stream) this.setHlsStream(stream) } else { console.error(`[PlayerHandler] Failed to start HLS stream`) } } else { this.setDirectPlay(audioTracks) } } getMimeTypeForTrack(track) { var ext = track.metadata.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' } closePlayer() { console.log('[PlayerHandler] Close Player') if (this.player) { this.player.destroy() } this.player = null this.playerState = 'IDLE' this.libraryItem = null this.currentStreamId = null this.startTime = 0 this.stopPlayInterval() } prepareStream(stream) { if (!this.player) this.switchPlayer() this.libraryItem = stream.libraryItem this.setHlsStream({ streamId: stream.id, streamUrl: stream.clientPlaylistUri, startTime: stream.clientCurrentTime }) } setHlsStream(stream) { this.currentStreamId = stream.streamId var audioTrack = new AudioTrack({ duration: this.libraryItem.media.duration, contentUrl: stream.streamUrl + '?token=' + this.userToken, mimeType: 'application/vnd.apple.mpegurl' }) this.startTime = stream.startTime this.ctx.playerLoading = true this.player.set(this.libraryItem, [audioTrack], this.currentStreamId, stream.startTime, this.playWhenReady) } setDirectPlay(audioTracks) { this.currentStreamId = null this.ctx.playerLoading = true this.player.set(this.libraryItem, 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.libraryItem.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 libraryItem 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.libraryItem.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() { if (this.player) this.player.play() } 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) } } }