mirror of
https://github.com/advplyr/audiobookshelf-app.git
synced 2025-06-25 22:58:43 +02:00
Add more menu on playlist item row to show library item more menu #914
This commit is contained in:
parent
74b488ad0f
commit
e9251db647
5 changed files with 535 additions and 279 deletions
473
components/modals/ItemMoreMenuModal.vue
Normal file
473
components/modals/ItemMoreMenuModal.vue
Normal file
|
@ -0,0 +1,473 @@
|
||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<modals-dialog v-model="show" :items="moreMenuItems" @action="moreMenuAction" />
|
||||||
|
<modals-item-details-modal v-model="showDetailsModal" :library-item="libraryItem" />
|
||||||
|
<modals-dialog v-model="showSendEbookDevicesModal" title="Select a device" :items="ereaderDeviceItems" @action="sendEbookToDeviceAction" />
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import { Dialog } from '@capacitor/dialog'
|
||||||
|
import { AbsFileSystem } from '@/plugins/capacitor'
|
||||||
|
|
||||||
|
export default {
|
||||||
|
props: {
|
||||||
|
value: Boolean,
|
||||||
|
processing: Boolean,
|
||||||
|
libraryItem: {
|
||||||
|
type: Object,
|
||||||
|
default: () => {}
|
||||||
|
},
|
||||||
|
episode: {
|
||||||
|
type: Object,
|
||||||
|
default: () => {}
|
||||||
|
},
|
||||||
|
rssFeed: {
|
||||||
|
type: Object,
|
||||||
|
default: () => null
|
||||||
|
},
|
||||||
|
hideRssFeedOption: Boolean
|
||||||
|
},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
showDetailsModal: false,
|
||||||
|
showSendEbookDevicesModal: false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
show: {
|
||||||
|
get() {
|
||||||
|
return this.value
|
||||||
|
},
|
||||||
|
set(val) {
|
||||||
|
this.$emit('input', val)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
userIsAdminOrUp() {
|
||||||
|
return this.$store.getters['user/getIsAdminOrUp']
|
||||||
|
},
|
||||||
|
moreMenuItems() {
|
||||||
|
const items = []
|
||||||
|
|
||||||
|
// TODO: Implement on iOS
|
||||||
|
if (this.$platform !== 'ios' && !this.isPodcast) {
|
||||||
|
items.push({
|
||||||
|
text: 'History',
|
||||||
|
value: 'history',
|
||||||
|
icon: 'history'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!this.isPodcast || this.episode) {
|
||||||
|
if (!this.userIsFinished) {
|
||||||
|
items.push({
|
||||||
|
text: 'Mark as Finished',
|
||||||
|
value: 'markFinished',
|
||||||
|
icon: 'beenhere'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.progressPercent > 0) {
|
||||||
|
items.push({
|
||||||
|
text: 'Discard Progress',
|
||||||
|
value: 'discardProgress',
|
||||||
|
icon: 'backspace'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((!this.isPodcast && this.serverLibraryItemId) || (this.episode && this.serverEpisodeId)) {
|
||||||
|
items.push({
|
||||||
|
text: 'Add to Playlist',
|
||||||
|
value: 'playlist',
|
||||||
|
icon: 'playlist_add'
|
||||||
|
})
|
||||||
|
|
||||||
|
if (this.ereaderDeviceItems.length) {
|
||||||
|
items.push({
|
||||||
|
text: 'Send ebook to device',
|
||||||
|
value: 'sendEbook',
|
||||||
|
icon: 'send'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.showRSSFeedOption) {
|
||||||
|
items.push({
|
||||||
|
text: this.rssFeed ? 'RSS Feed' : 'Open RSS Feed',
|
||||||
|
value: 'rssFeed',
|
||||||
|
icon: 'rss_feed'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.localLibraryItemId) {
|
||||||
|
items.push({
|
||||||
|
text: 'Manage Local Files',
|
||||||
|
value: 'manageLocal',
|
||||||
|
icon: 'folder'
|
||||||
|
})
|
||||||
|
|
||||||
|
if (!this.isPodcast) {
|
||||||
|
items.push({
|
||||||
|
text: 'Delete Local Item',
|
||||||
|
value: 'deleteLocal',
|
||||||
|
icon: 'delete'
|
||||||
|
})
|
||||||
|
} else if (this.localEpisodeId) {
|
||||||
|
items.push({
|
||||||
|
text: 'Delete Local Episode',
|
||||||
|
value: 'deleteLocalEpisode',
|
||||||
|
icon: 'delete'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!this.episode) {
|
||||||
|
items.push({
|
||||||
|
text: 'More Info',
|
||||||
|
value: 'details',
|
||||||
|
icon: 'info'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
return items
|
||||||
|
},
|
||||||
|
ereaderDeviceItems() {
|
||||||
|
if (!this.ebookFile || !this.$store.state.libraries.ereaderDevices?.length) return []
|
||||||
|
return this.$store.state.libraries.ereaderDevices.map((d) => {
|
||||||
|
return {
|
||||||
|
text: d.name,
|
||||||
|
value: d.name
|
||||||
|
}
|
||||||
|
})
|
||||||
|
},
|
||||||
|
isConnectedToServer() {
|
||||||
|
if (!this.isLocal) return true
|
||||||
|
if (!this.libraryItem?.serverAddress) return false
|
||||||
|
return this.$store.getters['user/getServerAddress'] === this.libraryItem.serverAddress
|
||||||
|
},
|
||||||
|
isLocal() {
|
||||||
|
return !!this.libraryItem?.isLocal
|
||||||
|
},
|
||||||
|
localLibraryItem() {
|
||||||
|
if (this.isLocal) return this.libraryItem
|
||||||
|
return this.libraryItem?.localLibraryItem || null
|
||||||
|
},
|
||||||
|
localLibraryItemId() {
|
||||||
|
return this.localLibraryItem?.id || null
|
||||||
|
},
|
||||||
|
serverLibraryItemId() {
|
||||||
|
if (!this.isLocal) return this.libraryItem?.id
|
||||||
|
if (this.isConnectedToServer) {
|
||||||
|
return this.libraryItem.libraryItemId
|
||||||
|
}
|
||||||
|
return null
|
||||||
|
},
|
||||||
|
localEpisode() {
|
||||||
|
if (this.isLocal) return this.episode
|
||||||
|
return this.episode.localEpisode
|
||||||
|
},
|
||||||
|
localEpisodeId() {
|
||||||
|
return this.localEpisode?.id || null
|
||||||
|
},
|
||||||
|
serverEpisodeId() {
|
||||||
|
if (!this.isLocal) return this.episode?.id
|
||||||
|
if (this.isConnectedToServer) {
|
||||||
|
return this.episode.serverEpisodeId
|
||||||
|
}
|
||||||
|
return null
|
||||||
|
},
|
||||||
|
mediaType() {
|
||||||
|
return this.libraryItem?.mediaType
|
||||||
|
},
|
||||||
|
isPodcast() {
|
||||||
|
return this.mediaType == 'podcast'
|
||||||
|
},
|
||||||
|
media() {
|
||||||
|
return this.libraryItem?.media || {}
|
||||||
|
},
|
||||||
|
mediaMetadata() {
|
||||||
|
return this.media.metadata || {}
|
||||||
|
},
|
||||||
|
title() {
|
||||||
|
return this.mediaMetadata.title
|
||||||
|
},
|
||||||
|
tracks() {
|
||||||
|
return this.media.tracks || []
|
||||||
|
},
|
||||||
|
episodes() {
|
||||||
|
return this.media.episodes || []
|
||||||
|
},
|
||||||
|
ebookFile() {
|
||||||
|
return this.media.ebookFile
|
||||||
|
},
|
||||||
|
localItemProgress() {
|
||||||
|
if (this.isPodcast) {
|
||||||
|
if (!this.localEpisodeId) return null
|
||||||
|
return this.$store.getters['globals/getLocalMediaProgressById'](this.localLibraryItemId, this.localEpisodeId)
|
||||||
|
}
|
||||||
|
return this.$store.getters['globals/getLocalMediaProgressById'](this.localLibraryItemId)
|
||||||
|
},
|
||||||
|
serverItemProgress() {
|
||||||
|
if (this.isPodcast) {
|
||||||
|
if (!this.serverEpisodeId) return null
|
||||||
|
return this.$store.getters['user/getUserMediaProgress'](this.serverLibraryItemId, this.serverEpisodeId)
|
||||||
|
}
|
||||||
|
return this.$store.getters['user/getUserMediaProgress'](this.serverLibraryItemId)
|
||||||
|
},
|
||||||
|
userItemProgress() {
|
||||||
|
if (this.isLocal) return this.localItemProgress
|
||||||
|
return this.serverItemProgress
|
||||||
|
},
|
||||||
|
userIsFinished() {
|
||||||
|
return !!this.userItemProgress?.isFinished
|
||||||
|
},
|
||||||
|
useEBookProgress() {
|
||||||
|
if (!this.userItemProgress || this.userItemProgress.progress) return false
|
||||||
|
return this.userItemProgress.ebookProgress > 0
|
||||||
|
},
|
||||||
|
progressPercent() {
|
||||||
|
if (this.useEBookProgress) return Math.max(Math.min(1, this.userItemProgress.ebookProgress), 0)
|
||||||
|
return Math.max(Math.min(1, this.userItemProgress?.progress || 0), 0)
|
||||||
|
},
|
||||||
|
showRSSFeedOption() {
|
||||||
|
if (this.hideRssFeedOption) return false
|
||||||
|
if (!this.serverLibraryItemId) return false
|
||||||
|
if (!this.rssFeed && !this.episodes.length && !this.tracks.length) return false // Cannot open RSS feed with no episodes/tracks
|
||||||
|
|
||||||
|
// If rss feed is open then show feed url to users otherwise just show to admins
|
||||||
|
return this.userIsAdminOrUp || this.rssFeed
|
||||||
|
},
|
||||||
|
mediaId() {
|
||||||
|
if (this.isPodcast) return null
|
||||||
|
return this.serverLibraryItemId || this.localLibraryItemId
|
||||||
|
}
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
moreMenuAction(action) {
|
||||||
|
this.show = false
|
||||||
|
if (action === 'manageLocal') {
|
||||||
|
this.$nextTick(() => {
|
||||||
|
this.$router.push(`/localMedia/item/${this.localLibraryItemId}`)
|
||||||
|
})
|
||||||
|
} else if (action === 'details') {
|
||||||
|
this.showDetailsModal = true
|
||||||
|
} else if (action === 'playlist') {
|
||||||
|
this.$store.commit('globals/setSelectedPlaylistItems', [{ libraryItem: this.libraryItem, episode: this.episode }])
|
||||||
|
this.$store.commit('globals/setShowPlaylistsAddCreateModal', true)
|
||||||
|
} else if (action === 'markFinished') {
|
||||||
|
if (this.episode) this.toggleEpisodeFinished()
|
||||||
|
else this.toggleFinished()
|
||||||
|
} else if (action === 'history') {
|
||||||
|
this.$router.push(`/media/${this.mediaId}/history?title=${this.title}`)
|
||||||
|
} else if (action === 'discardProgress') {
|
||||||
|
this.clearProgressClick()
|
||||||
|
} else if (action === 'deleteLocal') {
|
||||||
|
this.deleteLocalItem()
|
||||||
|
} else if (action === 'deleteLocalEpisode') {
|
||||||
|
this.deleteLocalEpisode()
|
||||||
|
} else if (action === 'rssFeed') {
|
||||||
|
this.clickRSSFeed()
|
||||||
|
} else if (action === 'sendEbook') {
|
||||||
|
this.showSendEbookDevicesModal = true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
async toggleFinished() {
|
||||||
|
await this.$hapticsImpact()
|
||||||
|
|
||||||
|
// Show confirm if item has progress since it will reset
|
||||||
|
if (this.userItemProgress && this.userItemProgress.progress > 0 && !this.userIsFinished) {
|
||||||
|
const { value } = await Dialog.confirm({
|
||||||
|
title: 'Confirm',
|
||||||
|
message: 'Are you sure you want to mark this item as Finished?'
|
||||||
|
})
|
||||||
|
if (!value) return
|
||||||
|
}
|
||||||
|
|
||||||
|
this.$emit('update:processing', true)
|
||||||
|
if (this.isLocal) {
|
||||||
|
const isFinished = !this.userIsFinished
|
||||||
|
const payload = await this.$db.updateLocalMediaProgressFinished({ localLibraryItemId: this.localLibraryItemId, isFinished })
|
||||||
|
console.log('toggleFinished payload', JSON.stringify(payload))
|
||||||
|
if (payload?.error) {
|
||||||
|
this.$toast.error(payload?.error || 'Unknown error')
|
||||||
|
} else {
|
||||||
|
const localMediaProgress = payload.localMediaProgress
|
||||||
|
console.log('toggleFinished localMediaProgress', JSON.stringify(localMediaProgress))
|
||||||
|
if (localMediaProgress) {
|
||||||
|
this.$store.commit('globals/updateLocalMediaProgress', localMediaProgress)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
const updatePayload = {
|
||||||
|
isFinished: !this.userIsFinished
|
||||||
|
}
|
||||||
|
await this.$nativeHttp.patch(`/api/me/progress/${this.serverLibraryItemId}`, updatePayload).catch((error) => {
|
||||||
|
console.error('Failed', error)
|
||||||
|
this.$toast.error(`Failed to mark as ${updatePayload.isFinished ? 'Finished' : 'Not Finished'}`)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
this.$emit('update:processing', false)
|
||||||
|
},
|
||||||
|
async toggleEpisodeFinished() {
|
||||||
|
await this.$hapticsImpact()
|
||||||
|
|
||||||
|
this.$emit('update:processing', true)
|
||||||
|
if (this.isLocal || this.localEpisode) {
|
||||||
|
const isFinished = !this.userIsFinished
|
||||||
|
const localLibraryItemId = this.localLibraryItemId
|
||||||
|
const localEpisodeId = this.localEpisodeId
|
||||||
|
const payload = await this.$db.updateLocalMediaProgressFinished({ localLibraryItemId, localEpisodeId, isFinished })
|
||||||
|
console.log('toggleFinished payload', JSON.stringify(payload))
|
||||||
|
|
||||||
|
if (payload?.error) {
|
||||||
|
this.$toast.error(payload?.error || 'Unknown error')
|
||||||
|
} else {
|
||||||
|
const localMediaProgress = payload.localMediaProgress
|
||||||
|
console.log('toggleFinished localMediaProgress', JSON.stringify(localMediaProgress))
|
||||||
|
if (localMediaProgress) {
|
||||||
|
this.$store.commit('globals/updateLocalMediaProgress', localMediaProgress)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
const updatePayload = {
|
||||||
|
isFinished: !this.userIsFinished
|
||||||
|
}
|
||||||
|
await this.$nativeHttp.patch(`/api/me/progress/${this.serverLibraryItemId}/${this.serverEpisodeId}`, updatePayload).catch((error) => {
|
||||||
|
console.error('Failed', error)
|
||||||
|
this.$toast.error(`Failed to mark as ${updatePayload.isFinished ? 'Finished' : 'Not Finished'}`)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
this.$emit('update:processing', false)
|
||||||
|
},
|
||||||
|
async clearProgressClick() {
|
||||||
|
await this.$hapticsImpact()
|
||||||
|
|
||||||
|
const { value } = await Dialog.confirm({
|
||||||
|
title: 'Confirm',
|
||||||
|
message: 'Are you sure you want to reset your progress?'
|
||||||
|
})
|
||||||
|
if (value) {
|
||||||
|
this.$emit('update:processing', true)
|
||||||
|
const serverMediaProgressId = this.serverItemProgress?.id
|
||||||
|
if (this.localItemProgress) {
|
||||||
|
await this.$db.removeLocalMediaProgress(this.localItemProgress.id)
|
||||||
|
this.$store.commit('globals/removeLocalMediaProgressForItem', this.localItemProgress.id)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (serverMediaProgressId) {
|
||||||
|
await this.$nativeHttp
|
||||||
|
.delete(`/api/me/progress/${serverMediaProgressId}`)
|
||||||
|
.then(() => {
|
||||||
|
console.log('Progress reset complete')
|
||||||
|
this.$toast.success(`Your progress was reset`)
|
||||||
|
this.$store.commit('user/removeMediaProgress', serverMediaProgressId)
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
console.error('Progress reset failed', error)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
this.$emit('update:processing', false)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
async deleteLocalEpisode() {
|
||||||
|
await this.$hapticsImpact()
|
||||||
|
|
||||||
|
const localEpisodeAudioTrack = this.localEpisode.audioTrack
|
||||||
|
const localFile = this.localLibraryItem.localFiles.find((lf) => lf.id === localEpisodeAudioTrack.localFileId)
|
||||||
|
if (!localFile) {
|
||||||
|
this.$toast.error('Audio track does not have matching local file..')
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
let confirmMessage = `Remove local episode "${localFile.basePath}" from your device?`
|
||||||
|
if (this.serverLibraryItemId) {
|
||||||
|
confirmMessage += ' The file on the server will be unaffected.'
|
||||||
|
}
|
||||||
|
const { value } = await Dialog.confirm({
|
||||||
|
title: 'Confirm',
|
||||||
|
message: confirmMessage
|
||||||
|
})
|
||||||
|
if (value) {
|
||||||
|
const res = await AbsFileSystem.deleteTrackFromItem({ id: this.localLibraryItemId, trackLocalFileId: localFile.id, trackContentUrl: localEpisodeAudioTrack.contentUrl })
|
||||||
|
if (res?.id) {
|
||||||
|
this.$toast.success('Deleted episode successfully')
|
||||||
|
if (this.isLocal) {
|
||||||
|
// If this is local episode then redirect to server episode when available
|
||||||
|
if (this.serverEpisodeId) {
|
||||||
|
this.$router.replace(`/item/${this.serverLibraryItemId}/${this.serverEpisodeId}`)
|
||||||
|
} else {
|
||||||
|
this.$router.replace(`/item/${this.localLibraryItemId}`)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Update local library item and local episode
|
||||||
|
this.libraryItem.localLibraryItem = res
|
||||||
|
this.$delete(this.episode, 'localEpisode')
|
||||||
|
}
|
||||||
|
} else this.$toast.error('Failed to delete')
|
||||||
|
}
|
||||||
|
},
|
||||||
|
async deleteLocalItem() {
|
||||||
|
await this.$hapticsImpact()
|
||||||
|
|
||||||
|
let confirmMessage = 'Remove local files of this item from your device?'
|
||||||
|
if (this.serverLibraryItemId) {
|
||||||
|
confirmMessage += ' The files on the server and your progress will be unaffected.'
|
||||||
|
}
|
||||||
|
const { value } = await Dialog.confirm({
|
||||||
|
title: 'Confirm',
|
||||||
|
message: confirmMessage
|
||||||
|
})
|
||||||
|
if (value) {
|
||||||
|
const res = await AbsFileSystem.deleteItem(this.localLibraryItem)
|
||||||
|
if (res?.success) {
|
||||||
|
this.$toast.success('Deleted successfully')
|
||||||
|
if (this.isLocal) {
|
||||||
|
// If local then redirect to server version when available
|
||||||
|
if (this.serverLibraryItemId) {
|
||||||
|
this.$router.replace(`/item/${this.serverLibraryItemId}`)
|
||||||
|
} else {
|
||||||
|
this.$router.replace('/bookshelf')
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Remove localLibraryItem
|
||||||
|
this.$delete(this.libraryItem, 'localLibraryItem')
|
||||||
|
}
|
||||||
|
} else this.$toast.error('Failed to delete')
|
||||||
|
}
|
||||||
|
},
|
||||||
|
clickRSSFeed() {
|
||||||
|
this.$store.commit('globals/setRSSFeedOpenCloseModal', {
|
||||||
|
id: this.serverLibraryItemId,
|
||||||
|
name: this.title,
|
||||||
|
type: 'item',
|
||||||
|
feed: this.rssFeed,
|
||||||
|
hasEpisodesWithoutPubDate: this.episodes.some((ep) => !ep.pubDate)
|
||||||
|
})
|
||||||
|
},
|
||||||
|
sendEbookToDeviceAction(deviceName) {
|
||||||
|
this.showSendEbookDevicesModal = false
|
||||||
|
|
||||||
|
const payload = {
|
||||||
|
libraryItemId: this.serverLibraryItemId,
|
||||||
|
deviceName
|
||||||
|
}
|
||||||
|
this.$emit('update:processing', true)
|
||||||
|
this.$nativeHttp
|
||||||
|
.post(`/api/emails/send-ebook-to-device`, payload)
|
||||||
|
.then(() => {
|
||||||
|
this.$toast.success('Ebook sent successfully')
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
console.error('Failed to send ebook to device', error)
|
||||||
|
this.$toast.error('Failed to send ebook to device')
|
||||||
|
})
|
||||||
|
.finally(() => {
|
||||||
|
this.$emit('update:processing', false)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
},
|
||||||
|
mounted() {}
|
||||||
|
}
|
||||||
|
</script>
|
|
@ -1,7 +1,7 @@
|
||||||
<template>
|
<template>
|
||||||
<div class="w-full px-1.5 pb-1.5">
|
<div class="w-full px-1.5 pb-1.5">
|
||||||
<div class="w-full h-full p-2 rounded-lg relative bg-bg overflow-hidden">
|
<div class="w-full h-full p-2 rounded-lg relative bg-bg overflow-hidden">
|
||||||
<nuxt-link v-if="libraryItem" :to="`/item/${libraryItem.id}`" class="flex items-center w-full">
|
<nuxt-link v-if="libraryItem" :to="itemUrl" class="flex items-center w-full">
|
||||||
<div class="h-full relative" :style="{ width: '50px' }">
|
<div class="h-full relative" :style="{ width: '50px' }">
|
||||||
<covers-book-cover :library-item="libraryItem" :width="50" :book-cover-aspect-ratio="bookCoverAspectRatio" />
|
<covers-book-cover :library-item="libraryItem" :width="50" :book-cover-aspect-ratio="bookCoverAspectRatio" />
|
||||||
</div>
|
</div>
|
||||||
|
@ -17,8 +17,13 @@
|
||||||
<span class="material-icons" :class="streamIsPlaying ? '' : 'text-success'">{{ streamIsPlaying ? 'pause' : 'play_arrow' }}</span>
|
<span class="material-icons" :class="streamIsPlaying ? '' : 'text-success'">{{ streamIsPlaying ? 'pause' : 'play_arrow' }}</span>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="w-8 min-w-8 flex justify-center">
|
||||||
|
<button class="w-8 h-8 rounded-full flex items-center justify-center" @click.stop.prevent="showMore">
|
||||||
|
<span class="material-icons">more_vert</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
</nuxt-link>
|
</nuxt-link>
|
||||||
<div v-if="!isPodcast" class="absolute bottom-0 left-0 h-0.5 shadow-sm z-10" :class="userIsFinished ? 'bg-success' : 'bg-yellow-400'" :style="{ width: progressPercent * 100 + '%' }"></div>
|
<div class="absolute bottom-0 left-0 h-0.5 shadow-sm z-10" :class="userIsFinished ? 'bg-success' : 'bg-yellow-400'" :style="{ width: progressPercent * 100 + '%' }"></div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
@ -36,6 +41,10 @@ export default {
|
||||||
return {}
|
return {}
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
|
itemUrl() {
|
||||||
|
if (this.episodeId) return `/item/${this.libraryItem.id}/${this.episodeId}`
|
||||||
|
return `/item/${this.libraryItem.id}`
|
||||||
|
},
|
||||||
libraryItem() {
|
libraryItem() {
|
||||||
return this.item.libraryItem || {}
|
return this.item.libraryItem || {}
|
||||||
},
|
},
|
||||||
|
@ -57,6 +66,12 @@ export default {
|
||||||
mediaMetadata() {
|
mediaMetadata() {
|
||||||
return this.media.metadata || {}
|
return this.media.metadata || {}
|
||||||
},
|
},
|
||||||
|
mediaType() {
|
||||||
|
return this.libraryItem.mediaType
|
||||||
|
},
|
||||||
|
isPodcast() {
|
||||||
|
return this.mediaType === 'podcast'
|
||||||
|
},
|
||||||
tracks() {
|
tracks() {
|
||||||
if (this.episode) return []
|
if (this.episode) return []
|
||||||
return this.media.tracks || []
|
return this.media.tracks || []
|
||||||
|
@ -109,10 +124,23 @@ export default {
|
||||||
return !!this.userItemProgress?.isFinished
|
return !!this.userItemProgress?.isFinished
|
||||||
},
|
},
|
||||||
progressPercent() {
|
progressPercent() {
|
||||||
return this.userItemProgress ? Math.max(Math.min(1, this.userItemProgress.progress), 0) : 0
|
return Math.max(Math.min(1, this.userItemProgress?.progress || 0), 0)
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
|
showMore() {
|
||||||
|
const playlistItem = {
|
||||||
|
libraryItem: this.libraryItem,
|
||||||
|
episode: this.episode
|
||||||
|
}
|
||||||
|
if (this.localLibraryItem) {
|
||||||
|
playlistItem.libraryItem.localLibraryItem = this.localLibraryItem
|
||||||
|
}
|
||||||
|
if (this.localEpisode && playlistItem.episode) {
|
||||||
|
playlistItem.episode.localEpisode = this.localEpisode
|
||||||
|
}
|
||||||
|
this.$emit('showMore', playlistItem)
|
||||||
|
},
|
||||||
async playClick() {
|
async playClick() {
|
||||||
await this.$hapticsImpact()
|
await this.$hapticsImpact()
|
||||||
if (this.streamIsPlaying) {
|
if (this.streamIsPlaying) {
|
||||||
|
@ -138,7 +166,7 @@ export default {
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
.item-table-content {
|
.item-table-content {
|
||||||
width: calc(100% - 82px);
|
width: calc(100% - 114px);
|
||||||
max-width: calc(100% - 82px);
|
max-width: calc(100% - 114px);
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
|
@ -11,7 +11,7 @@
|
||||||
<p v-if="totalDuration" class="text-sm text-gray-200">{{ totalDurationPretty }}</p>
|
<p v-if="totalDuration" class="text-sm text-gray-200">{{ totalDurationPretty }}</p>
|
||||||
</div>
|
</div>
|
||||||
<template v-for="item in items">
|
<template v-for="item in items">
|
||||||
<tables-playlist-item-table-row :key="item.id" :item="item" :playlist-id="playlistId" />
|
<tables-playlist-item-table-row :key="item.id" :item="item" :playlist-id="playlistId" @showMore="showMore" />
|
||||||
</template>
|
</template>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
@ -41,7 +41,11 @@ export default {
|
||||||
return this.$elapsedPrettyExtended(this.totalDuration)
|
return this.$elapsedPrettyExtended(this.totalDuration)
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
methods: {},
|
methods: {
|
||||||
|
showMore(playlistItem) {
|
||||||
|
this.$emit('showMore', playlistItem)
|
||||||
|
}
|
||||||
|
},
|
||||||
mounted() {}
|
mounted() {}
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
|
@ -50,7 +50,7 @@
|
||||||
</ui-btn>
|
</ui-btn>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div v-if="!isPodcast && progressPercent > 0" class="px-4 py-2 bg-primary text-sm font-semibold rounded-md text-gray-200 mt-4 text-center" :class="resettingProgress ? 'opacity-25' : ''">
|
<div v-if="!isPodcast && progressPercent > 0" class="px-4 py-2 bg-primary text-sm font-semibold rounded-md text-gray-200 mt-4 text-center">
|
||||||
<p>Your Progress: {{ Math.round(progressPercent * 100) }}%</p>
|
<p>Your Progress: {{ Math.round(progressPercent * 100) }}%</p>
|
||||||
<p v-if="!useEBookProgress && !userIsFinished" class="text-gray-400 text-xs">{{ $elapsedPretty(userTimeRemaining) }} remaining</p>
|
<p v-if="!useEBookProgress && !userIsFinished" class="text-gray-400 text-xs">{{ $elapsedPretty(userTimeRemaining) }} remaining</p>
|
||||||
<p v-else-if="userIsFinished" class="text-gray-400 text-xs">Finished {{ $formatDate(userProgressFinishedAt) }}</p>
|
<p v-else-if="userIsFinished" class="text-gray-400 text-xs">Finished {{ $formatDate(userProgressFinishedAt) }}</p>
|
||||||
|
@ -126,13 +126,10 @@
|
||||||
<tables-ebook-files-table v-if="ebookFiles.length" :library-item="libraryItem" />
|
<tables-ebook-files-table v-if="ebookFiles.length" :library-item="libraryItem" />
|
||||||
|
|
||||||
<!-- modals -->
|
<!-- modals -->
|
||||||
|
<modals-item-more-menu-modal v-model="showMoreMenu" :library-item="libraryItem" :rss-feed="rssFeed" :processing.sync="processing" />
|
||||||
|
|
||||||
<modals-select-local-folder-modal v-model="showSelectLocalFolder" :media-type="mediaType" @select="selectedLocalFolder" />
|
<modals-select-local-folder-modal v-model="showSelectLocalFolder" :media-type="mediaType" @select="selectedLocalFolder" />
|
||||||
|
|
||||||
<modals-dialog v-model="showMoreMenu" :items="moreMenuItems" @action="moreMenuAction" />
|
|
||||||
<modals-dialog v-model="showSendEbookDevicesModal" title="Select a device" :items="ereaderDeviceItems" @action="sendEbookToDeviceAction" />
|
|
||||||
|
|
||||||
<modals-item-details-modal v-model="showDetailsModal" :library-item="libraryItem" />
|
|
||||||
|
|
||||||
<modals-fullscreen-cover v-model="showFullscreenCover" :library-item="libraryItem" />
|
<modals-fullscreen-cover v-model="showFullscreenCover" :library-item="libraryItem" />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
@ -181,12 +178,8 @@ export default {
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
processing: false,
|
processing: false,
|
||||||
resettingProgress: false,
|
|
||||||
isProcessingReadUpdate: false,
|
|
||||||
showSelectLocalFolder: false,
|
showSelectLocalFolder: false,
|
||||||
showMoreMenu: false,
|
showMoreMenu: false,
|
||||||
showDetailsModal: false,
|
|
||||||
showSendEbookDevicesModal: false,
|
|
||||||
showFullscreenCover: false,
|
showFullscreenCover: false,
|
||||||
coverRgb: 'rgb(55, 56, 56)',
|
coverRgb: 'rgb(55, 56, 56)',
|
||||||
coverBgIsLight: false,
|
coverBgIsLight: false,
|
||||||
|
@ -318,7 +311,7 @@ export default {
|
||||||
return this.$store.getters['user/getUserMediaProgress'](this.serverLibraryItemId)
|
return this.$store.getters['user/getUserMediaProgress'](this.serverLibraryItemId)
|
||||||
},
|
},
|
||||||
userIsFinished() {
|
userIsFinished() {
|
||||||
return this.userItemProgress ? !!this.userItemProgress.isFinished : false
|
return !!this.userItemProgress?.isFinished
|
||||||
},
|
},
|
||||||
userTimeRemaining() {
|
userTimeRemaining() {
|
||||||
if (!this.userItemProgress) return 0
|
if (!this.userItemProgress) return 0
|
||||||
|
@ -331,10 +324,10 @@ export default {
|
||||||
},
|
},
|
||||||
progressPercent() {
|
progressPercent() {
|
||||||
if (this.useEBookProgress) return Math.max(Math.min(1, this.userItemProgress.ebookProgress), 0)
|
if (this.useEBookProgress) return Math.max(Math.min(1, this.userItemProgress.ebookProgress), 0)
|
||||||
return this.userItemProgress ? Math.max(Math.min(1, this.userItemProgress.progress), 0) : 0
|
return Math.max(Math.min(1, this.userItemProgress?.progress || 0), 0)
|
||||||
},
|
},
|
||||||
userProgressFinishedAt() {
|
userProgressFinishedAt() {
|
||||||
return this.userItemProgress ? this.userItemProgress.finishedAt : 0
|
return this.userItemProgress?.finishedAt || 0
|
||||||
},
|
},
|
||||||
isStreaming() {
|
isStreaming() {
|
||||||
return this.isPlaying && !this.$store.getters['getIsCurrentSessionLocal']
|
return this.isPlaying && !this.$store.getters['getIsCurrentSessionLocal']
|
||||||
|
@ -394,100 +387,6 @@ export default {
|
||||||
isCasting() {
|
isCasting() {
|
||||||
return this.$store.state.isCasting
|
return this.$store.state.isCasting
|
||||||
},
|
},
|
||||||
showRSSFeedOption() {
|
|
||||||
if (!this.serverLibraryItemId) return false
|
|
||||||
if (!this.rssFeed && !this.episodes.length && !this.tracks.length) return false // Cannot open RSS feed with no episodes/tracks
|
|
||||||
|
|
||||||
// If rss feed is open then show feed url to users otherwise just show to admins
|
|
||||||
return this.userIsAdminOrUp || this.rssFeed
|
|
||||||
},
|
|
||||||
moreMenuItems() {
|
|
||||||
const items = []
|
|
||||||
|
|
||||||
if (!this.isPodcast) {
|
|
||||||
// TODO: Implement on iOS
|
|
||||||
if (!this.isIos) {
|
|
||||||
items.push({
|
|
||||||
text: 'History',
|
|
||||||
value: 'history',
|
|
||||||
icon: 'history'
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!this.userIsFinished) {
|
|
||||||
items.push({
|
|
||||||
text: 'Mark as Finished',
|
|
||||||
value: 'markFinished',
|
|
||||||
icon: 'beenhere'
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.progressPercent > 0) {
|
|
||||||
items.push({
|
|
||||||
text: 'Discard Progress',
|
|
||||||
value: 'discardProgress',
|
|
||||||
icon: 'backspace'
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!this.isPodcast && this.serverLibraryItemId) {
|
|
||||||
items.push({
|
|
||||||
text: 'Add to Playlist',
|
|
||||||
value: 'playlist',
|
|
||||||
icon: 'playlist_add'
|
|
||||||
})
|
|
||||||
|
|
||||||
if (this.ebookFile && this.$store.state.libraries.ereaderDevices?.length) {
|
|
||||||
items.push({
|
|
||||||
text: 'Send ebook to device',
|
|
||||||
value: 'sendEbook',
|
|
||||||
icon: 'send'
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.showRSSFeedOption) {
|
|
||||||
items.push({
|
|
||||||
text: this.rssFeed ? 'RSS Feed' : 'Open RSS Feed',
|
|
||||||
value: 'rssFeed',
|
|
||||||
icon: 'rss_feed'
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.localLibraryItemId) {
|
|
||||||
items.push({
|
|
||||||
text: 'Manage Local Files',
|
|
||||||
value: 'manageLocal',
|
|
||||||
icon: 'folder'
|
|
||||||
})
|
|
||||||
|
|
||||||
if (!this.isPodcast) {
|
|
||||||
items.push({
|
|
||||||
text: 'Delete Local Item',
|
|
||||||
value: 'deleteLocal',
|
|
||||||
icon: 'delete'
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
items.push({
|
|
||||||
text: 'More Info',
|
|
||||||
value: 'details',
|
|
||||||
icon: 'info'
|
|
||||||
})
|
|
||||||
|
|
||||||
return items
|
|
||||||
},
|
|
||||||
ereaderDeviceItems() {
|
|
||||||
if (!this.ebookFile || !this.$store.state.libraries.ereaderDevices?.length) return []
|
|
||||||
return this.$store.state.libraries.ereaderDevices.map((d) => {
|
|
||||||
return {
|
|
||||||
text: d.name,
|
|
||||||
value: d.name
|
|
||||||
}
|
|
||||||
})
|
|
||||||
},
|
|
||||||
coverWidth() {
|
coverWidth() {
|
||||||
let width = this.windowWidth - 94
|
let width = this.windowWidth - 94
|
||||||
if (width > 325) return 325
|
if (width > 325) return 325
|
||||||
|
@ -495,42 +394,9 @@ export default {
|
||||||
|
|
||||||
if (width * this.bookCoverAspectRatio > 325) width = 325 / this.bookCoverAspectRatio
|
if (width * this.bookCoverAspectRatio > 325) width = 325 / this.bookCoverAspectRatio
|
||||||
return width
|
return width
|
||||||
},
|
|
||||||
mediaId() {
|
|
||||||
if (this.isPodcast) return null
|
|
||||||
return this.serverLibraryItemId || this.localLibraryItemId
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
async deleteLocalItem() {
|
|
||||||
await this.$hapticsImpact()
|
|
||||||
|
|
||||||
let confirmMessage = 'Remove local files of this item from your device?'
|
|
||||||
if (this.serverLibraryItemId) {
|
|
||||||
confirmMessage += ' The files on the server and your progress will be unaffected.'
|
|
||||||
}
|
|
||||||
const { value } = await Dialog.confirm({
|
|
||||||
title: 'Confirm',
|
|
||||||
message: confirmMessage
|
|
||||||
})
|
|
||||||
if (value) {
|
|
||||||
const res = await AbsFileSystem.deleteItem(this.localLibraryItem)
|
|
||||||
if (res?.success) {
|
|
||||||
this.$toast.success('Deleted successfully')
|
|
||||||
if (this.isLocal) {
|
|
||||||
// If local then redirect to server version when available
|
|
||||||
if (this.serverLibraryItemId) {
|
|
||||||
this.$router.replace(`/item/${this.serverLibraryItemId}`)
|
|
||||||
} else {
|
|
||||||
this.$router.replace('/bookshelf')
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// Remove localLibraryItem
|
|
||||||
this.$delete(this.libraryItem, 'localLibraryItem')
|
|
||||||
}
|
|
||||||
} else this.$toast.error('Failed to delete')
|
|
||||||
}
|
|
||||||
},
|
|
||||||
async coverImageLoaded(fullCoverUrl) {
|
async coverImageLoaded(fullCoverUrl) {
|
||||||
if (!fullCoverUrl) return
|
if (!fullCoverUrl) return
|
||||||
|
|
||||||
|
@ -545,64 +411,7 @@ export default {
|
||||||
console.log(e)
|
console.log(e)
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
moreMenuAction(action) {
|
|
||||||
this.showMoreMenu = false
|
|
||||||
if (action === 'manageLocal') {
|
|
||||||
this.$nextTick(() => {
|
|
||||||
this.$router.push(`/localMedia/item/${this.localLibraryItemId}`)
|
|
||||||
})
|
|
||||||
} else if (action === 'details') {
|
|
||||||
this.showDetailsModal = true
|
|
||||||
} else if (action === 'playlist') {
|
|
||||||
this.$store.commit('globals/setSelectedPlaylistItems', [{ libraryItem: this.libraryItem, episode: null }])
|
|
||||||
this.$store.commit('globals/setShowPlaylistsAddCreateModal', true)
|
|
||||||
} else if (action === 'markFinished') {
|
|
||||||
if (this.isProcessingReadUpdate) return
|
|
||||||
this.toggleFinished()
|
|
||||||
} else if (action === 'history') {
|
|
||||||
this.$router.push(`/media/${this.mediaId}/history?title=${this.title}`)
|
|
||||||
} else if (action === 'discardProgress') {
|
|
||||||
this.clearProgressClick()
|
|
||||||
} else if (action === 'deleteLocal') {
|
|
||||||
this.deleteLocalItem()
|
|
||||||
} else if (action === 'rssFeed') {
|
|
||||||
this.clickRSSFeed()
|
|
||||||
} else if (action === 'sendEbook') {
|
|
||||||
this.showSendEbookDevicesModal = true
|
|
||||||
}
|
|
||||||
},
|
|
||||||
sendEbookToDeviceAction(deviceName) {
|
|
||||||
this.showSendEbookDevicesModal = false
|
|
||||||
|
|
||||||
const payload = {
|
|
||||||
libraryItemId: this.serverLibraryItemId,
|
|
||||||
deviceName
|
|
||||||
}
|
|
||||||
this.processing = true
|
|
||||||
this.$nativeHttp
|
|
||||||
.post(`/api/emails/send-ebook-to-device`, payload)
|
|
||||||
.then(() => {
|
|
||||||
this.$toast.success('Ebook sent successfully')
|
|
||||||
})
|
|
||||||
.catch((error) => {
|
|
||||||
console.error('Failed to send ebook to device', error)
|
|
||||||
this.$toast.error('Failed to send ebook to device')
|
|
||||||
})
|
|
||||||
.finally(() => {
|
|
||||||
this.processing = false
|
|
||||||
})
|
|
||||||
},
|
|
||||||
clickRSSFeed() {
|
|
||||||
this.$store.commit('globals/setRSSFeedOpenCloseModal', {
|
|
||||||
id: this.serverLibraryItemId,
|
|
||||||
name: this.title,
|
|
||||||
type: 'item',
|
|
||||||
feed: this.rssFeed,
|
|
||||||
hasEpisodesWithoutPubDate: this.episodes.some((ep) => !ep.pubDate)
|
|
||||||
})
|
|
||||||
},
|
|
||||||
moreButtonPress() {
|
moreButtonPress() {
|
||||||
this.showSendEbookDevicesModal = false
|
|
||||||
this.showMoreMenu = true
|
this.showMoreMenu = true
|
||||||
},
|
},
|
||||||
readBook() {
|
readBook() {
|
||||||
|
@ -683,37 +492,6 @@ export default {
|
||||||
this.$eventBus.$emit('play-item', { libraryItemId, serverLibraryItemId: this.serverLibraryItemId, startTime })
|
this.$eventBus.$emit('play-item', { libraryItemId, serverLibraryItemId: this.serverLibraryItemId, startTime })
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
async clearProgressClick() {
|
|
||||||
await this.$hapticsImpact()
|
|
||||||
|
|
||||||
const { value } = await Dialog.confirm({
|
|
||||||
title: 'Confirm',
|
|
||||||
message: 'Are you sure you want to reset your progress?'
|
|
||||||
})
|
|
||||||
if (value) {
|
|
||||||
this.resettingProgress = true
|
|
||||||
const serverMediaProgressId = this.serverItemProgress?.id
|
|
||||||
if (this.localLibraryItemId) {
|
|
||||||
await this.$db.removeLocalMediaProgress(this.localLibraryItemId)
|
|
||||||
this.$store.commit('globals/removeLocalMediaProgressForItem', this.localLibraryItemId)
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.serverLibraryItemId && serverMediaProgressId) {
|
|
||||||
await this.$nativeHttp
|
|
||||||
.delete(`/api/me/progress/${serverMediaProgressId}`)
|
|
||||||
.then(() => {
|
|
||||||
console.log('Progress reset complete')
|
|
||||||
this.$toast.success(`Your progress was reset`)
|
|
||||||
this.$store.commit('user/removeMediaProgress', serverMediaProgressId)
|
|
||||||
})
|
|
||||||
.catch((error) => {
|
|
||||||
console.error('Progress reset failed', error)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
this.resettingProgress = false
|
|
||||||
}
|
|
||||||
},
|
|
||||||
itemUpdated(libraryItem) {
|
itemUpdated(libraryItem) {
|
||||||
if (libraryItem.id === this.serverLibraryItemId) {
|
if (libraryItem.id === this.serverLibraryItemId) {
|
||||||
console.log('Item Updated')
|
console.log('Item Updated')
|
||||||
|
@ -815,48 +593,6 @@ export default {
|
||||||
this.$set(this.libraryItem, 'localLibraryItem', item)
|
this.$set(this.libraryItem, 'localLibraryItem', item)
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
async toggleFinished() {
|
|
||||||
await this.$hapticsImpact()
|
|
||||||
|
|
||||||
// Show confirm if item has progress since it will reset
|
|
||||||
if (this.userItemProgress && this.userItemProgress.progress > 0 && !this.userIsFinished) {
|
|
||||||
const { value } = await Dialog.confirm({
|
|
||||||
title: 'Confirm',
|
|
||||||
message: 'Are you sure you want to mark this item as Finished?'
|
|
||||||
})
|
|
||||||
if (!value) return
|
|
||||||
}
|
|
||||||
|
|
||||||
this.isProcessingReadUpdate = true
|
|
||||||
if (this.isLocal) {
|
|
||||||
const isFinished = !this.userIsFinished
|
|
||||||
const payload = await this.$db.updateLocalMediaProgressFinished({ localLibraryItemId: this.localLibraryItemId, isFinished })
|
|
||||||
console.log('toggleFinished payload', JSON.stringify(payload))
|
|
||||||
if (payload?.error) {
|
|
||||||
this.$toast.error(payload?.error || 'Unknown error')
|
|
||||||
} else {
|
|
||||||
const localMediaProgress = payload.localMediaProgress
|
|
||||||
console.log('toggleFinished localMediaProgress', JSON.stringify(localMediaProgress))
|
|
||||||
if (localMediaProgress) {
|
|
||||||
this.$store.commit('globals/updateLocalMediaProgress', localMediaProgress)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
this.isProcessingReadUpdate = false
|
|
||||||
} else {
|
|
||||||
const updatePayload = {
|
|
||||||
isFinished: !this.userIsFinished
|
|
||||||
}
|
|
||||||
this.$nativeHttp
|
|
||||||
.patch(`/api/me/progress/${this.libraryItemId}`, updatePayload)
|
|
||||||
.catch((error) => {
|
|
||||||
console.error('Failed', error)
|
|
||||||
this.$toast.error(`Failed to mark as ${updatePayload.isFinished ? 'Finished' : 'Not Finished'}`)
|
|
||||||
})
|
|
||||||
.finally(() => {
|
|
||||||
this.isProcessingReadUpdate = false
|
|
||||||
})
|
|
||||||
}
|
|
||||||
},
|
|
||||||
libraryChanged(libraryId) {
|
libraryChanged(libraryId) {
|
||||||
if (this.libraryItem.libraryId !== libraryId) {
|
if (this.libraryItem.libraryId !== libraryId) {
|
||||||
this.$router.replace('/bookshelf')
|
this.$router.replace('/bookshelf')
|
||||||
|
|
|
@ -20,9 +20,14 @@
|
||||||
<p class="text-base text-gray-100">{{ description }}</p>
|
<p class="text-base text-gray-100">{{ description }}</p>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<tables-playlist-items-table :items="playlistItems" :playlist-id="playlist.id" />
|
<tables-playlist-items-table :items="playlistItems" :playlist-id="playlist.id" @showMore="showMore" />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<modals-item-more-menu-modal v-model="showMoreMenu" :library-item="selectedLibraryItem" :episode="selectedEpisode" hide-rss-feed-option :processing.sync="processing" />
|
||||||
|
<div v-show="processing" class="fixed top-0 left-0 w-screen h-screen flex items-center justify-center bg-black/50 z-50">
|
||||||
|
<ui-loading-indicator />
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
@ -67,7 +72,12 @@ export default {
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
data() {
|
data() {
|
||||||
return {}
|
return {
|
||||||
|
showMoreMenu: false,
|
||||||
|
processing: false,
|
||||||
|
selectedLibraryItem: null,
|
||||||
|
selectedEpisode: null
|
||||||
|
}
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
bookCoverAspectRatio() {
|
bookCoverAspectRatio() {
|
||||||
|
@ -101,6 +111,11 @@ export default {
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
|
showMore(playlistItem) {
|
||||||
|
this.selectedLibraryItem = playlistItem.libraryItem
|
||||||
|
this.selectedEpisode = playlistItem.episode
|
||||||
|
this.showMoreMenu = true
|
||||||
|
},
|
||||||
clickPlay() {
|
clickPlay() {
|
||||||
const nextItem = this.playableItems.find((i) => {
|
const nextItem = this.playableItems.find((i) => {
|
||||||
const prog = this.$store.getters['user/getUserMediaProgress'](i.libraryItemId, i.episodeId)
|
const prog = this.$store.getters['user/getUserMediaProgress'](i.libraryItemId, i.episodeId)
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue