2021-08-18 00:01:11 +02:00
|
|
|
<template>
|
2024-04-04 01:41:41 +02:00
|
|
|
<div v-if="streamLibraryItem" id="mediaPlayerContainer" class="w-full fixed bottom-0 left-0 right-0 h-48 lg:h-40 z-50 bg-primary px-2 lg:px-4 pb-1 lg:pb-4 pt-2">
|
2022-05-31 02:26:53 +02:00
|
|
|
<div id="videoDock" />
|
2024-04-04 01:41:41 +02:00
|
|
|
<div class="absolute left-2 top-2 lg:left-4 cursor-pointer">
|
2023-10-27 23:51:44 +02:00
|
|
|
<covers-book-cover expand-on-click :library-item="streamLibraryItem" :width="bookCoverWidth" :book-cover-aspect-ratio="coverAspectRatio" />
|
|
|
|
</div>
|
2024-04-04 01:41:41 +02:00
|
|
|
<div class="flex items-start mb-6 lg:mb-0" :class="playerHandler.isVideo ? 'ml-4 pl-96' : isSquareCover ? 'pl-18 sm:pl-24' : 'pl-12 sm:pl-16'">
|
2024-06-06 23:56:57 +02:00
|
|
|
<div class="min-w-0 w-full">
|
2024-05-12 20:35:03 +02:00
|
|
|
<div class="flex items-center">
|
|
|
|
<nuxt-link :to="`/item/${streamLibraryItem.id}`" class="hover:underline cursor-pointer text-sm sm:text-lg block truncate">
|
|
|
|
{{ title }}
|
|
|
|
</nuxt-link>
|
|
|
|
<widgets-explicit-indicator v-if="isExplicit" />
|
|
|
|
</div>
|
2024-06-06 23:56:57 +02:00
|
|
|
<div v-if="!playerHandler.isVideo" class="text-gray-400 flex items-center w-1/2 sm:w-4/5 lg:w-2/5">
|
2024-07-08 18:36:37 +02:00
|
|
|
<span class="material-symbols text-sm">person</span>
|
2024-06-06 23:56:57 +02:00
|
|
|
<div v-if="podcastAuthor" class="pl-1 sm:pl-1.5 text-xs sm:text-base">{{ podcastAuthor }}</div>
|
|
|
|
<div v-else-if="musicArtists" class="pl-1 sm:pl-1.5 text-xs sm:text-base">{{ musicArtists }}</div>
|
|
|
|
<div v-else-if="authors.length" class="pl-1 sm:pl-1.5 text-xs sm:text-base truncate">
|
|
|
|
<nuxt-link v-for="(author, index) in authors" :key="index" :to="`/author/${author.id}`" class="hover:underline">{{ author.name }}<span v-if="index < authors.length - 1">, </span></nuxt-link>
|
2023-02-27 03:56:07 +01:00
|
|
|
</div>
|
2024-06-06 23:56:57 +02:00
|
|
|
<div v-else class="text-xs sm:text-base cursor-pointer pl-1 sm:pl-1.5">{{ $strings.LabelUnknown }}</div>
|
2021-10-27 01:12:19 +02:00
|
|
|
</div>
|
|
|
|
|
|
|
|
<div class="text-gray-400 flex items-center">
|
2024-07-08 18:36:37 +02:00
|
|
|
<span class="material-symbols text-xs">schedule</span>
|
2022-06-19 17:04:15 +02:00
|
|
|
<p class="font-mono text-xs sm:text-sm pl-1 sm:pl-1.5 pb-px">{{ totalDurationPretty }}</p>
|
2021-10-27 01:12:19 +02:00
|
|
|
</div>
|
2021-08-18 00:01:11 +02:00
|
|
|
</div>
|
|
|
|
<div class="flex-grow" />
|
2022-11-20 18:50:34 +01:00
|
|
|
<ui-tooltip direction="top" :text="$strings.LabelClosePlayer">
|
2024-07-08 18:36:37 +02:00
|
|
|
<button :aria-label="$strings.LabelClosePlayer" class="material-symbols sm:px-2 py-1 lg:p-4 cursor-pointer text-xl sm:text-2xl" @click="closePlayer">close</button>
|
2022-11-20 18:50:34 +01:00
|
|
|
</ui-tooltip>
|
2021-08-18 00:01:11 +02:00
|
|
|
</div>
|
2022-06-01 03:13:46 +02:00
|
|
|
<player-ui
|
2022-03-05 19:30:46 +01:00
|
|
|
ref="audioPlayer"
|
|
|
|
:chapters="chapters"
|
2024-07-14 20:56:48 +02:00
|
|
|
:current-chapter="currentChapter"
|
2022-03-05 19:30:46 +01:00
|
|
|
:paused="!isPlaying"
|
|
|
|
:loading="playerLoading"
|
|
|
|
:bookmarks="bookmarks"
|
|
|
|
:sleep-timer-set="sleepTimerSet"
|
|
|
|
:sleep-timer-remaining="sleepTimerRemaining"
|
2024-07-14 20:56:48 +02:00
|
|
|
:sleep-timer-type="sleepTimerType"
|
2022-04-12 02:42:09 +02:00
|
|
|
:is-podcast="isPodcast"
|
2024-08-17 20:32:00 +02:00
|
|
|
:hasNextItemInQueue="hasNextItemInQueue"
|
2022-03-05 19:30:46 +01:00
|
|
|
@playPause="playPause"
|
|
|
|
@jumpForward="jumpForward"
|
|
|
|
@jumpBackward="jumpBackward"
|
|
|
|
@setVolume="setVolume"
|
|
|
|
@setPlaybackRate="setPlaybackRate"
|
|
|
|
@seek="seek"
|
2024-08-17 20:32:00 +02:00
|
|
|
@nextItemInQueue="playNextItemInQueue"
|
2022-03-05 19:30:46 +01:00
|
|
|
@close="closePlayer"
|
|
|
|
@showBookmarks="showBookmarks"
|
|
|
|
@showSleepTimer="showSleepTimerModal = true"
|
2022-08-28 20:12:38 +02:00
|
|
|
@showPlayerQueueItems="showPlayerQueueItemsModal = true"
|
2024-07-13 00:52:48 +02:00
|
|
|
@showPlayerSettings="showPlayerSettingsModal = true"
|
2022-03-05 19:30:46 +01:00
|
|
|
/>
|
2021-10-25 01:25:44 +02:00
|
|
|
|
2022-03-18 02:28:04 +01:00
|
|
|
<modals-bookmarks-modal v-model="showBookmarksModal" :bookmarks="bookmarks" :current-time="bookmarkCurrentTime" :library-item-id="libraryItemId" @select="selectBookmark" />
|
2022-03-05 19:30:46 +01:00
|
|
|
|
2024-07-14 20:56:48 +02:00
|
|
|
<modals-sleep-timer-modal v-model="showSleepTimerModal" :timer-set="sleepTimerSet" :timer-type="sleepTimerType" :remaining="sleepTimerRemaining" :has-chapters="!!chapters.length" @set="setSleepTimer" @cancel="cancelSleepTimer" @increment="incrementSleepTimer" @decrement="decrementSleepTimer" />
|
2022-08-28 20:12:38 +02:00
|
|
|
|
2024-08-17 20:32:00 +02:00
|
|
|
<modals-player-queue-items-modal v-model="showPlayerQueueItemsModal" />
|
2024-07-13 00:52:48 +02:00
|
|
|
|
|
|
|
<modals-player-settings-modal v-model="showPlayerSettingsModal" />
|
2021-08-18 00:01:11 +02:00
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<script>
|
2022-02-23 00:33:55 +01:00
|
|
|
import PlayerHandler from '@/players/PlayerHandler'
|
|
|
|
|
2021-08-18 00:01:11 +02:00
|
|
|
export default {
|
|
|
|
data() {
|
|
|
|
return {
|
2022-02-23 00:33:55 +01:00
|
|
|
playerHandler: new PlayerHandler(this),
|
2021-10-25 01:25:44 +02:00
|
|
|
totalDuration: 0,
|
|
|
|
showBookmarksModal: false,
|
|
|
|
bookmarkCurrentTime: 0,
|
2022-02-23 00:33:55 +01:00
|
|
|
playerLoading: false,
|
|
|
|
isPlaying: false,
|
2022-03-05 19:30:46 +01:00
|
|
|
currentTime: 0,
|
|
|
|
showSleepTimerModal: false,
|
2022-08-28 20:12:38 +02:00
|
|
|
showPlayerQueueItemsModal: false,
|
2024-07-13 00:52:48 +02:00
|
|
|
showPlayerSettingsModal: false,
|
2022-03-05 19:30:46 +01:00
|
|
|
sleepTimerSet: false,
|
|
|
|
sleepTimerRemaining: 0,
|
2024-07-14 20:56:48 +02:00
|
|
|
sleepTimerType: null,
|
2022-04-02 17:26:42 +02:00
|
|
|
sleepTimer: null,
|
2022-04-23 23:51:13 +02:00
|
|
|
displayTitle: null,
|
2023-04-08 01:05:23 +02:00
|
|
|
currentPlaybackRate: 1,
|
2024-04-24 00:12:13 +02:00
|
|
|
syncFailedToast: null,
|
|
|
|
coverAspectRatio: 1
|
2021-08-18 00:01:11 +02:00
|
|
|
}
|
|
|
|
},
|
|
|
|
computed: {
|
2023-01-08 16:38:37 +01:00
|
|
|
isSquareCover() {
|
|
|
|
return this.coverAspectRatio === 1
|
|
|
|
},
|
|
|
|
isMobile() {
|
|
|
|
return this.$store.state.globals.isMobile
|
2021-11-04 23:35:59 +01:00
|
|
|
},
|
2023-01-08 16:38:37 +01:00
|
|
|
bookCoverWidth() {
|
|
|
|
if (this.isMobile) return 64 / this.coverAspectRatio
|
|
|
|
return 77 / this.coverAspectRatio
|
2021-12-03 02:02:38 +01:00
|
|
|
},
|
2021-08-18 00:01:11 +02:00
|
|
|
cover() {
|
2022-03-13 02:59:35 +01:00
|
|
|
if (this.media.coverPath) return this.media.coverPath
|
2021-08-18 00:01:11 +02:00
|
|
|
return 'Logo.png'
|
|
|
|
},
|
|
|
|
user() {
|
2021-08-24 01:31:04 +02:00
|
|
|
return this.$store.state.user.user
|
2021-08-18 00:01:11 +02:00
|
|
|
},
|
2022-03-26 17:59:34 +01:00
|
|
|
userMediaProgress() {
|
2022-03-13 02:59:35 +01:00
|
|
|
if (!this.libraryItemId) return
|
2022-03-26 17:59:34 +01:00
|
|
|
return this.$store.getters['user/getUserMediaProgress'](this.libraryItemId)
|
2021-10-25 01:25:44 +02:00
|
|
|
},
|
2022-03-18 01:10:47 +01:00
|
|
|
userItemCurrentTime() {
|
2022-03-26 17:59:34 +01:00
|
|
|
return this.userMediaProgress ? this.userMediaProgress.currentTime || 0 : 0
|
2022-02-23 00:33:55 +01:00
|
|
|
},
|
2021-10-25 01:25:44 +02:00
|
|
|
bookmarks() {
|
2022-03-18 02:28:04 +01:00
|
|
|
if (!this.libraryItemId) return []
|
|
|
|
return this.$store.getters['user/getUserBookmarksForItem'](this.libraryItemId)
|
2021-10-25 01:25:44 +02:00
|
|
|
},
|
2022-03-13 02:59:35 +01:00
|
|
|
streamLibraryItem() {
|
|
|
|
return this.$store.state.streamLibraryItem
|
2021-08-18 00:01:11 +02:00
|
|
|
},
|
2023-04-09 21:32:51 +02:00
|
|
|
streamEpisode() {
|
|
|
|
if (!this.$store.state.streamEpisodeId) return null
|
|
|
|
const episodes = this.streamLibraryItem.media.episodes || []
|
|
|
|
return episodes.find((ep) => ep.id === this.$store.state.streamEpisodeId)
|
|
|
|
},
|
2022-03-13 02:59:35 +01:00
|
|
|
libraryItemId() {
|
2023-04-09 21:32:51 +02:00
|
|
|
return this.streamLibraryItem?.id || null
|
2021-10-25 01:25:44 +02:00
|
|
|
},
|
2022-03-13 02:59:35 +01:00
|
|
|
media() {
|
2023-04-09 21:32:51 +02:00
|
|
|
return this.streamLibraryItem?.media || {}
|
2022-03-13 02:59:35 +01:00
|
|
|
},
|
2022-03-27 00:30:58 +01:00
|
|
|
isPodcast() {
|
2023-04-09 21:32:51 +02:00
|
|
|
return this.streamLibraryItem?.mediaType === 'podcast'
|
2022-03-27 00:30:58 +01:00
|
|
|
},
|
2022-12-22 23:38:55 +01:00
|
|
|
isMusic() {
|
2023-04-09 21:32:51 +02:00
|
|
|
return this.streamLibraryItem?.mediaType === 'music'
|
2022-12-22 23:38:55 +01:00
|
|
|
},
|
2023-02-27 03:56:07 +01:00
|
|
|
isExplicit() {
|
2024-05-12 20:35:03 +02:00
|
|
|
return !!this.mediaMetadata.explicit
|
2023-02-27 03:56:07 +01:00
|
|
|
},
|
2022-03-13 02:59:35 +01:00
|
|
|
mediaMetadata() {
|
|
|
|
return this.media.metadata || {}
|
2021-08-18 00:01:11 +02:00
|
|
|
},
|
2021-09-08 16:15:54 +02:00
|
|
|
chapters() {
|
2023-04-09 21:32:51 +02:00
|
|
|
if (this.streamEpisode) return this.streamEpisode.chapters || []
|
2022-03-13 02:59:35 +01:00
|
|
|
return this.media.chapters || []
|
2021-09-08 16:15:54 +02:00
|
|
|
},
|
2024-07-14 20:56:48 +02:00
|
|
|
currentChapter() {
|
|
|
|
return this.chapters.find((chapter) => chapter.start <= this.currentTime && this.currentTime < chapter.end)
|
|
|
|
},
|
2021-08-18 00:01:11 +02:00
|
|
|
title() {
|
2022-04-02 17:26:42 +02:00
|
|
|
if (this.playerHandler.displayTitle) return this.playerHandler.displayTitle
|
2022-03-13 02:59:35 +01:00
|
|
|
return this.mediaMetadata.title || 'No Title'
|
2021-10-31 22:40:27 +01:00
|
|
|
},
|
2022-03-13 02:59:35 +01:00
|
|
|
authors() {
|
|
|
|
return this.mediaMetadata.authors || []
|
2021-10-31 22:40:27 +01:00
|
|
|
},
|
2021-10-06 04:10:49 +02:00
|
|
|
libraryId() {
|
2022-03-13 02:59:35 +01:00
|
|
|
return this.streamLibraryItem ? this.streamLibraryItem.libraryId : null
|
2021-10-25 01:25:44 +02:00
|
|
|
},
|
|
|
|
totalDurationPretty() {
|
2023-04-08 01:05:23 +02:00
|
|
|
// Adjusted by playback rate
|
|
|
|
return this.$secondsToTimestamp(this.totalDuration / this.currentPlaybackRate)
|
2022-03-27 00:30:58 +01:00
|
|
|
},
|
|
|
|
podcastAuthor() {
|
|
|
|
if (!this.isPodcast) return null
|
|
|
|
return this.mediaMetadata.author || 'Unknown'
|
2022-08-28 20:12:38 +02:00
|
|
|
},
|
2023-01-04 01:00:01 +01:00
|
|
|
musicArtists() {
|
|
|
|
if (!this.isMusic) return null
|
|
|
|
return this.mediaMetadata.artists.join(', ')
|
|
|
|
},
|
2024-08-17 20:32:00 +02:00
|
|
|
hasNextItemInQueue() {
|
|
|
|
return this.currentPlayerQueueIndex < this.playerQueueItems.length - 1
|
|
|
|
},
|
|
|
|
currentPlayerQueueIndex() {
|
|
|
|
if (!this.libraryItemId) return -1
|
|
|
|
return this.playerQueueItems.findIndex((i) => {
|
|
|
|
if (this.streamEpisode?.id) return i.episodeId === this.streamEpisode.id
|
|
|
|
return i.libraryItemId === this.libraryItemId
|
|
|
|
})
|
|
|
|
},
|
2022-08-28 20:12:38 +02:00
|
|
|
playerQueueItems() {
|
|
|
|
return this.$store.state.playerQueueItems || []
|
2021-08-18 00:01:11 +02:00
|
|
|
}
|
|
|
|
},
|
|
|
|
methods: {
|
2022-08-28 21:21:28 +02:00
|
|
|
mediaFinished(libraryItemId, episodeId) {
|
|
|
|
// Play next item in queue
|
|
|
|
if (!this.playerQueueItems.length || !this.$store.state.playerQueueAutoPlay) {
|
|
|
|
// TODO: Set media finished flag so play button will play next queue item
|
|
|
|
return
|
|
|
|
}
|
2022-08-28 20:12:38 +02:00
|
|
|
var currentQueueIndex = this.playerQueueItems.findIndex((i) => {
|
|
|
|
if (episodeId) return i.libraryItemId === libraryItemId && i.episodeId === episodeId
|
|
|
|
return i.libraryItemId === libraryItemId
|
|
|
|
})
|
|
|
|
if (currentQueueIndex < 0) {
|
2022-11-12 22:41:41 +01:00
|
|
|
console.error('Media finished not found in queue - using first in queue', this.playerQueueItems)
|
|
|
|
currentQueueIndex = -1
|
2022-08-28 20:12:38 +02:00
|
|
|
}
|
|
|
|
if (currentQueueIndex === this.playerQueueItems.length - 1) {
|
|
|
|
console.log('Finished last item in queue')
|
|
|
|
return
|
|
|
|
}
|
|
|
|
const nextItemInQueue = this.playerQueueItems[currentQueueIndex + 1]
|
|
|
|
if (nextItemInQueue) {
|
|
|
|
this.playLibraryItem({
|
|
|
|
libraryItemId: nextItemInQueue.libraryItemId,
|
|
|
|
episodeId: nextItemInQueue.episodeId || null,
|
|
|
|
queueItems: this.playerQueueItems
|
|
|
|
})
|
|
|
|
}
|
|
|
|
},
|
2022-03-26 23:41:26 +01:00
|
|
|
setPlaying(isPlaying) {
|
|
|
|
this.isPlaying = isPlaying
|
|
|
|
this.$store.commit('setIsPlaying', isPlaying)
|
2022-05-15 22:48:41 +02:00
|
|
|
this.updateMediaSessionPlaybackState()
|
2022-03-26 23:41:26 +01:00
|
|
|
},
|
2024-07-14 20:56:48 +02:00
|
|
|
setSleepTimer(time) {
|
2022-03-05 19:30:46 +01:00
|
|
|
this.sleepTimerSet = true
|
|
|
|
this.showSleepTimerModal = false
|
2024-07-14 20:56:48 +02:00
|
|
|
|
|
|
|
this.sleepTimerType = time.timerType
|
|
|
|
if (this.sleepTimerType === this.$constants.SleepTimerTypes.COUNTDOWN) {
|
|
|
|
this.runSleepTimer(time)
|
|
|
|
}
|
2022-03-05 19:30:46 +01:00
|
|
|
},
|
2024-07-14 20:56:48 +02:00
|
|
|
runSleepTimer(time) {
|
|
|
|
this.sleepTimerRemaining = time.seconds
|
|
|
|
|
2022-03-05 19:30:46 +01:00
|
|
|
var lastTick = Date.now()
|
|
|
|
clearInterval(this.sleepTimer)
|
|
|
|
this.sleepTimer = setInterval(() => {
|
|
|
|
var elapsed = Date.now() - lastTick
|
|
|
|
lastTick = Date.now()
|
|
|
|
this.sleepTimerRemaining -= elapsed / 1000
|
|
|
|
|
|
|
|
if (this.sleepTimerRemaining <= 0) {
|
2024-07-14 20:56:48 +02:00
|
|
|
this.sleepTimerEnd()
|
2022-03-05 19:30:46 +01:00
|
|
|
}
|
|
|
|
}, 1000)
|
|
|
|
},
|
2024-07-14 20:56:48 +02:00
|
|
|
checkChapterEnd(time) {
|
|
|
|
if (!this.currentChapter) return
|
|
|
|
const chapterEndTime = this.currentChapter.end
|
|
|
|
const tolerance = 0.75
|
|
|
|
if (time >= chapterEndTime - tolerance) {
|
|
|
|
this.sleepTimerEnd()
|
|
|
|
}
|
|
|
|
},
|
|
|
|
sleepTimerEnd() {
|
|
|
|
this.clearSleepTimer()
|
|
|
|
this.playerHandler.pause()
|
|
|
|
this.$toast.info('Sleep Timer Done.. zZzzZz')
|
|
|
|
},
|
2022-03-05 19:30:46 +01:00
|
|
|
cancelSleepTimer() {
|
|
|
|
this.showSleepTimerModal = false
|
|
|
|
this.clearSleepTimer()
|
|
|
|
},
|
|
|
|
clearSleepTimer() {
|
|
|
|
clearInterval(this.sleepTimer)
|
|
|
|
this.sleepTimerRemaining = 0
|
|
|
|
this.sleepTimer = null
|
|
|
|
this.sleepTimerSet = false
|
2024-07-14 20:56:48 +02:00
|
|
|
this.sleepTimerType = null
|
2022-03-05 19:30:46 +01:00
|
|
|
},
|
|
|
|
incrementSleepTimer(amount) {
|
|
|
|
if (!this.sleepTimerSet) return
|
|
|
|
this.sleepTimerRemaining += amount
|
|
|
|
},
|
|
|
|
decrementSleepTimer(amount) {
|
|
|
|
if (this.sleepTimerRemaining < amount) {
|
|
|
|
this.sleepTimerRemaining = 3
|
|
|
|
return
|
|
|
|
}
|
|
|
|
this.sleepTimerRemaining = Math.max(0, this.sleepTimerRemaining - amount)
|
|
|
|
},
|
2022-02-23 00:33:55 +01:00
|
|
|
playPause() {
|
|
|
|
this.playerHandler.playPause()
|
2021-11-13 02:43:16 +01:00
|
|
|
},
|
2022-02-23 00:33:55 +01:00
|
|
|
jumpForward() {
|
|
|
|
this.playerHandler.jumpForward()
|
2021-10-25 01:25:44 +02:00
|
|
|
},
|
2022-02-23 00:33:55 +01:00
|
|
|
jumpBackward() {
|
|
|
|
this.playerHandler.jumpBackward()
|
2021-10-25 01:25:44 +02:00
|
|
|
},
|
2022-02-23 00:33:55 +01:00
|
|
|
setVolume(volume) {
|
|
|
|
this.playerHandler.setVolume(volume)
|
2021-09-21 23:42:01 +02:00
|
|
|
},
|
2022-02-23 00:33:55 +01:00
|
|
|
setPlaybackRate(playbackRate) {
|
2023-04-08 01:05:23 +02:00
|
|
|
this.currentPlaybackRate = playbackRate
|
2022-02-23 00:33:55 +01:00
|
|
|
this.playerHandler.setPlaybackRate(playbackRate)
|
2021-08-18 00:01:11 +02:00
|
|
|
},
|
2022-02-23 00:33:55 +01:00
|
|
|
seek(time) {
|
|
|
|
this.playerHandler.seek(time)
|
2021-08-18 00:01:11 +02:00
|
|
|
},
|
2023-05-28 00:21:43 +02:00
|
|
|
playbackTimeUpdate(time) {
|
|
|
|
// When updating progress from another session
|
|
|
|
this.playerHandler.seek(time, false)
|
|
|
|
},
|
2022-02-23 00:33:55 +01:00
|
|
|
setCurrentTime(time) {
|
|
|
|
this.currentTime = time
|
2021-08-18 00:01:11 +02:00
|
|
|
if (this.$refs.audioPlayer) {
|
2022-02-23 00:33:55 +01:00
|
|
|
this.$refs.audioPlayer.setCurrentTime(time)
|
2021-08-18 00:01:11 +02:00
|
|
|
}
|
2024-07-14 20:56:48 +02:00
|
|
|
|
|
|
|
if (this.sleepTimerType === this.$constants.SleepTimerTypes.CHAPTER && this.sleepTimerSet) {
|
|
|
|
this.checkChapterEnd(time)
|
|
|
|
}
|
2021-08-18 00:01:11 +02:00
|
|
|
},
|
2022-02-23 00:33:55 +01:00
|
|
|
setDuration(duration) {
|
|
|
|
this.totalDuration = duration
|
|
|
|
if (this.$refs.audioPlayer) {
|
|
|
|
this.$refs.audioPlayer.setDuration(duration)
|
2021-08-18 00:01:11 +02:00
|
|
|
}
|
2022-02-23 00:33:55 +01:00
|
|
|
},
|
|
|
|
setBufferTime(buffertime) {
|
|
|
|
if (this.$refs.audioPlayer) {
|
|
|
|
this.$refs.audioPlayer.setBufferTime(buffertime)
|
2021-08-19 01:31:19 +02:00
|
|
|
}
|
2021-08-18 00:01:11 +02:00
|
|
|
},
|
2022-03-05 19:30:46 +01:00
|
|
|
showBookmarks() {
|
|
|
|
this.bookmarkCurrentTime = this.currentTime
|
2022-02-23 00:33:55 +01:00
|
|
|
this.showBookmarksModal = true
|
|
|
|
},
|
|
|
|
selectBookmark(bookmark) {
|
|
|
|
this.seek(bookmark.time)
|
|
|
|
this.showBookmarksModal = false
|
|
|
|
},
|
|
|
|
closePlayer() {
|
|
|
|
this.playerHandler.closePlayer()
|
2022-03-26 23:41:26 +01:00
|
|
|
this.$store.commit('setMediaPlaying', null)
|
2022-02-23 00:33:55 +01:00
|
|
|
},
|
2022-05-15 22:48:41 +02:00
|
|
|
mediaSessionPlay() {
|
|
|
|
console.log('Media session play')
|
|
|
|
this.playerHandler.play()
|
|
|
|
},
|
|
|
|
mediaSessionPause() {
|
|
|
|
console.log('Media session pause')
|
|
|
|
this.playerHandler.pause()
|
|
|
|
},
|
|
|
|
mediaSessionStop() {
|
|
|
|
console.log('Media session stop')
|
|
|
|
this.playerHandler.pause()
|
|
|
|
},
|
|
|
|
mediaSessionSeekBackward() {
|
|
|
|
console.log('Media session seek backward')
|
|
|
|
this.playerHandler.jumpBackward()
|
|
|
|
},
|
|
|
|
mediaSessionSeekForward() {
|
|
|
|
console.log('Media session seek forward')
|
|
|
|
this.playerHandler.jumpForward()
|
|
|
|
},
|
|
|
|
mediaSessionSeekTo(e) {
|
|
|
|
console.log('Media session seek to', e)
|
|
|
|
if (e.seekTime !== null && !isNaN(e.seekTime)) {
|
|
|
|
this.playerHandler.seek(e.seekTime)
|
|
|
|
}
|
|
|
|
},
|
2022-11-22 23:57:18 +01:00
|
|
|
mediaSessionPreviousTrack() {
|
|
|
|
if (this.$refs.audioPlayer) {
|
|
|
|
this.$refs.audioPlayer.prevChapter()
|
|
|
|
}
|
|
|
|
},
|
|
|
|
mediaSessionNextTrack() {
|
|
|
|
if (this.$refs.audioPlayer) {
|
|
|
|
this.$refs.audioPlayer.nextChapter()
|
|
|
|
}
|
|
|
|
},
|
2022-05-15 22:48:41 +02:00
|
|
|
updateMediaSessionPlaybackState() {
|
|
|
|
if ('mediaSession' in navigator) {
|
|
|
|
navigator.mediaSession.playbackState = this.isPlaying ? 'playing' : 'paused'
|
|
|
|
}
|
|
|
|
},
|
|
|
|
setMediaSession() {
|
|
|
|
if (!this.streamLibraryItem) {
|
|
|
|
console.error('setMediaSession: No library item set')
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if ('mediaSession' in navigator) {
|
2024-01-15 15:34:12 +01:00
|
|
|
var coverImageSrc = this.$store.getters['globals/getLibraryItemCoverSrc'](this.streamLibraryItem, '/Logo.png', true)
|
2022-05-15 22:48:41 +02:00
|
|
|
const artwork = [
|
|
|
|
{
|
|
|
|
src: coverImageSrc
|
|
|
|
}
|
|
|
|
]
|
|
|
|
|
|
|
|
navigator.mediaSession.metadata = new MediaMetadata({
|
|
|
|
title: this.title,
|
|
|
|
artist: this.playerHandler.displayAuthor || this.mediaMetadata.authorName || 'Unknown',
|
|
|
|
album: this.mediaMetadata.seriesName || '',
|
|
|
|
artwork
|
|
|
|
})
|
|
|
|
console.log('Set media session metadata', navigator.mediaSession.metadata)
|
|
|
|
|
|
|
|
navigator.mediaSession.setActionHandler('play', this.mediaSessionPlay)
|
|
|
|
navigator.mediaSession.setActionHandler('pause', this.mediaSessionPause)
|
|
|
|
navigator.mediaSession.setActionHandler('stop', this.mediaSessionStop)
|
|
|
|
navigator.mediaSession.setActionHandler('seekbackward', this.mediaSessionSeekBackward)
|
|
|
|
navigator.mediaSession.setActionHandler('seekforward', this.mediaSessionSeekForward)
|
|
|
|
navigator.mediaSession.setActionHandler('seekto', this.mediaSessionSeekTo)
|
2023-05-07 21:23:07 +02:00
|
|
|
navigator.mediaSession.setActionHandler('previoustrack', this.mediaSessionSeekBackward)
|
|
|
|
navigator.mediaSession.setActionHandler('nexttrack', this.mediaSessionSeekForward)
|
2022-05-15 22:48:41 +02:00
|
|
|
} else {
|
|
|
|
console.warn('Media session not available')
|
|
|
|
}
|
|
|
|
},
|
2021-08-18 00:01:11 +02:00
|
|
|
streamProgress(data) {
|
2023-01-15 22:00:18 +01:00
|
|
|
if (this.playerHandler.isPlayingLocalItem && this.playerHandler.currentStreamId === data.stream) {
|
|
|
|
if (!data.numSegments) return
|
|
|
|
var chunks = data.chunks
|
2024-02-15 01:12:35 +01:00
|
|
|
console.log(`[MediaPlayerContainer] Stream Progress ${data.percent}`)
|
2023-01-15 22:00:18 +01:00
|
|
|
if (this.$refs.audioPlayer) {
|
|
|
|
this.$refs.audioPlayer.setChunksReady(chunks, data.numSegments)
|
|
|
|
} else {
|
|
|
|
console.error('No Audio Ref')
|
|
|
|
}
|
2021-08-18 00:01:11 +02:00
|
|
|
}
|
|
|
|
},
|
2022-08-28 20:12:38 +02:00
|
|
|
sessionOpen(session) {
|
|
|
|
// For opening session on init (temporarily unused)
|
2022-03-26 23:41:26 +01:00
|
|
|
this.$store.commit('setMediaPlaying', {
|
|
|
|
libraryItem: session.libraryItem,
|
|
|
|
episodeId: session.episodeId
|
|
|
|
})
|
2023-04-08 01:05:23 +02:00
|
|
|
this.playerHandler.prepareOpenSession(session, this.currentPlaybackRate)
|
2021-08-18 00:01:11 +02:00
|
|
|
},
|
2022-04-16 19:37:10 +02:00
|
|
|
streamOpen(session) {
|
2024-02-15 01:12:35 +01:00
|
|
|
console.log(`[MediaPlayerContainer] Stream session open`, session)
|
2022-04-16 19:37:10 +02:00
|
|
|
},
|
2021-08-18 00:01:11 +02:00
|
|
|
streamClosed(streamId) {
|
2022-02-23 00:33:55 +01:00
|
|
|
// Stream was closed from the server
|
2022-03-13 02:59:35 +01:00
|
|
|
if (this.playerHandler.isPlayingLocalItem && this.playerHandler.currentStreamId === streamId) {
|
2024-02-15 01:12:35 +01:00
|
|
|
console.warn('[MediaPlayerContainer] Closing stream due to request from server')
|
2022-02-23 00:33:55 +01:00
|
|
|
this.playerHandler.closePlayer()
|
2021-08-18 00:01:11 +02:00
|
|
|
}
|
|
|
|
},
|
|
|
|
streamReady() {
|
2024-02-15 01:12:35 +01:00
|
|
|
console.log(`[MediaPlayerContainer] Stream Ready`)
|
2021-08-18 00:01:11 +02:00
|
|
|
if (this.$refs.audioPlayer) {
|
|
|
|
this.$refs.audioPlayer.setStreamReady()
|
2021-08-21 01:29:10 +02:00
|
|
|
} else {
|
|
|
|
console.error('No Audio Ref')
|
2021-08-18 00:01:11 +02:00
|
|
|
}
|
|
|
|
},
|
2021-11-13 22:24:56 +01:00
|
|
|
streamError(streamId) {
|
2022-02-23 00:33:55 +01:00
|
|
|
// Stream had critical error from the server
|
2022-03-13 02:59:35 +01:00
|
|
|
if (this.playerHandler.isPlayingLocalItem && this.playerHandler.currentStreamId === streamId) {
|
2024-02-15 01:12:35 +01:00
|
|
|
console.warn('[MediaPlayerContainer] Closing stream due to stream error from server')
|
2022-02-23 00:33:55 +01:00
|
|
|
this.playerHandler.closePlayer()
|
2021-11-13 22:24:56 +01:00
|
|
|
}
|
|
|
|
},
|
2021-08-18 00:01:11 +02:00
|
|
|
streamReset({ startTime, streamId }) {
|
2022-02-23 00:33:55 +01:00
|
|
|
this.playerHandler.resetStream(startTime, streamId)
|
|
|
|
},
|
|
|
|
castSessionActive(isActive) {
|
2022-03-13 02:59:35 +01:00
|
|
|
if (isActive && this.playerHandler.isPlayingLocalItem) {
|
2022-02-23 00:33:55 +01:00
|
|
|
// Cast session started switch to cast player
|
|
|
|
this.playerHandler.switchPlayer()
|
2022-03-13 02:59:35 +01:00
|
|
|
} else if (!isActive && this.playerHandler.isPlayingCastedItem) {
|
2022-02-23 00:33:55 +01:00
|
|
|
// Cast session ended switch to local player
|
|
|
|
this.playerHandler.switchPlayer()
|
|
|
|
}
|
|
|
|
},
|
2024-08-17 20:32:00 +02:00
|
|
|
playNextItemInQueue() {
|
|
|
|
if (this.hasNextItemInQueue) {
|
|
|
|
this.playQueueItem({ index: this.currentPlayerQueueIndex + 1 })
|
|
|
|
}
|
|
|
|
},
|
|
|
|
/**
|
|
|
|
* @param {{ index: number }} payload
|
|
|
|
*/
|
|
|
|
playQueueItem(payload) {
|
|
|
|
if (payload?.index === undefined) {
|
|
|
|
console.error('playQueueItem: No index provided')
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if (!this.playerQueueItems[payload.index]) {
|
|
|
|
console.error('playQueueItem: No item found at index', payload.index)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
const item = this.playerQueueItems[payload.index]
|
|
|
|
this.playLibraryItem({
|
|
|
|
libraryItemId: item.libraryItemId,
|
|
|
|
episodeId: item.episodeId || null,
|
|
|
|
queueItems: this.playerQueueItems
|
|
|
|
})
|
|
|
|
},
|
2022-03-26 23:41:26 +01:00
|
|
|
async playLibraryItem(payload) {
|
2022-12-22 23:38:55 +01:00
|
|
|
const libraryItemId = payload.libraryItemId
|
|
|
|
const episodeId = payload.episodeId || null
|
2022-03-26 23:41:26 +01:00
|
|
|
|
|
|
|
if (this.playerHandler.libraryItemId == libraryItemId && this.playerHandler.episodeId == episodeId) {
|
2022-07-30 19:25:15 +02:00
|
|
|
if (payload.startTime !== null && !isNaN(payload.startTime)) {
|
|
|
|
this.seek(payload.startTime)
|
|
|
|
} else {
|
2022-08-13 20:56:37 +02:00
|
|
|
this.playerHandler.play()
|
2022-07-30 19:25:15 +02:00
|
|
|
}
|
2022-03-26 23:41:26 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-12-22 23:38:55 +01:00
|
|
|
const libraryItem = await this.$axios.$get(`/api/items/${libraryItemId}?expanded=1`).catch((error) => {
|
2022-03-13 02:59:35 +01:00
|
|
|
console.error('Failed to fetch full item', error)
|
2022-02-23 00:33:55 +01:00
|
|
|
return null
|
|
|
|
})
|
2022-03-13 02:59:35 +01:00
|
|
|
if (!libraryItem) return
|
2022-12-22 23:38:55 +01:00
|
|
|
|
2022-03-26 23:41:26 +01:00
|
|
|
this.$store.commit('setMediaPlaying', {
|
|
|
|
libraryItem,
|
2022-08-28 20:12:38 +02:00
|
|
|
episodeId,
|
|
|
|
queueItems: payload.queueItems || []
|
2022-03-26 23:41:26 +01:00
|
|
|
})
|
2024-04-24 00:12:13 +02:00
|
|
|
// Set cover aspect ratio for this item's library since the library may change
|
|
|
|
this.coverAspectRatio = this.$store.getters['libraries/getBookCoverAspectRatio']
|
|
|
|
|
2022-07-14 02:38:34 +02:00
|
|
|
this.$nextTick(() => {
|
|
|
|
if (this.$refs.audioPlayer) this.$refs.audioPlayer.checkUpdateChapterTrack()
|
|
|
|
})
|
|
|
|
|
2023-04-08 01:05:23 +02:00
|
|
|
this.playerHandler.load(libraryItem, episodeId, true, this.currentPlaybackRate, payload.startTime)
|
2022-03-26 23:41:26 +01:00
|
|
|
},
|
|
|
|
pauseItem() {
|
|
|
|
this.playerHandler.pause()
|
2022-06-04 02:11:13 +02:00
|
|
|
},
|
|
|
|
showFailedProgressSyncs() {
|
|
|
|
if (!isNaN(this.syncFailedToast)) this.$toast.dismiss(this.syncFailedToast)
|
|
|
|
this.syncFailedToast = this.$toast('Progress is not being synced. Restart playback', { timeout: false, type: 'error' })
|
2023-04-09 01:01:24 +02:00
|
|
|
},
|
|
|
|
sessionClosedEvent(sessionId) {
|
|
|
|
if (this.playerHandler.currentSessionId === sessionId) {
|
|
|
|
console.log('sessionClosedEvent closing current session', sessionId)
|
|
|
|
this.playerHandler.resetPlayer() // Closes player without reporting to server
|
|
|
|
this.$store.commit('setMediaPlaying', null)
|
|
|
|
}
|
2021-08-18 00:01:11 +02:00
|
|
|
}
|
2022-02-23 00:33:55 +01:00
|
|
|
},
|
|
|
|
mounted() {
|
|
|
|
this.$eventBus.$on('cast-session-active', this.castSessionActive)
|
2022-07-30 02:06:52 +02:00
|
|
|
this.$eventBus.$on('playback-seek', this.seek)
|
2023-05-28 00:21:43 +02:00
|
|
|
this.$eventBus.$on('playback-time-update', this.playbackTimeUpdate)
|
2024-08-17 20:32:00 +02:00
|
|
|
this.$eventBus.$on('play-queue-item', this.playQueueItem)
|
2022-03-13 02:59:35 +01:00
|
|
|
this.$eventBus.$on('play-item', this.playLibraryItem)
|
2022-03-26 23:41:26 +01:00
|
|
|
this.$eventBus.$on('pause-item', this.pauseItem)
|
2022-02-23 00:33:55 +01:00
|
|
|
},
|
|
|
|
beforeDestroy() {
|
|
|
|
this.$eventBus.$off('cast-session-active', this.castSessionActive)
|
2022-07-30 02:06:52 +02:00
|
|
|
this.$eventBus.$off('playback-seek', this.seek)
|
2023-05-28 00:21:43 +02:00
|
|
|
this.$eventBus.$off('playback-time-update', this.playbackTimeUpdate)
|
2024-08-17 20:32:00 +02:00
|
|
|
this.$eventBus.$off('play-queue-item', this.playQueueItem)
|
2022-03-13 02:59:35 +01:00
|
|
|
this.$eventBus.$off('play-item', this.playLibraryItem)
|
2022-03-26 23:41:26 +01:00
|
|
|
this.$eventBus.$off('pause-item', this.pauseItem)
|
2021-08-18 00:01:11 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
</script>
|
|
|
|
|
|
|
|
<style>
|
2024-02-15 01:12:35 +01:00
|
|
|
#mediaPlayerContainer {
|
2021-08-18 00:01:11 +02:00
|
|
|
box-shadow: 0px -6px 8px #1111113f;
|
|
|
|
}
|
2023-02-27 03:56:07 +01:00
|
|
|
</style>
|