audiobookshelf/client/components/modals/PlayerSettingsModal.vue
Lunatic d157388680 Add per-chapter auto skip intro/outro for web player
Per-book skip settings stored in browser localStorage. Checks each
chapter's intro/outro zone during playback and automatically seeks
past them, matching the behavior of the Android app implementation.
2026-02-27 14:33:19 +08:00

165 lines
6.8 KiB
Vue

<template>
<modals-modal v-model="show" name="player-settings" :width="500" :height="'unset'">
<div ref="container" class="w-full rounded-lg bg-bg box-shadow-md overflow-y-auto overflow-x-hidden p-4" style="max-height: 80vh; min-height: 40vh">
<h3 class="text-xl font-semibold mb-8">{{ $strings.HeaderPlayerSettings }}</h3>
<div class="flex items-center mb-4">
<ui-toggle-switch v-model="useChapterTrack" @input="setUseChapterTrack" />
<div class="pl-4">
<span>{{ $strings.LabelUseChapterTrack }}</span>
</div>
</div>
<div class="flex items-center mb-4">
<ui-select-input v-model="jumpForwardAmount" :label="$strings.LabelJumpForwardAmount" menuMaxHeight="250px" :items="jumpValues" @input="setJumpForwardAmount" />
</div>
<div class="flex items-center mb-4">
<ui-select-input v-model="jumpBackwardAmount" :label="$strings.LabelJumpBackwardAmount" menuMaxHeight="250px" :items="jumpValues" @input="setJumpBackwardAmount" />
</div>
<div class="flex items-center mb-4">
<ui-select-input v-model="playbackRateIncrementDecrement" :label="$strings.LabelPlaybackRateIncrementDecrement" menuMaxHeight="250px" :items="playbackRateIncrementDecrementValues" @input="setPlaybackRateIncrementDecrementAmount" />
</div>
<!-- 书籍跳过配置 -->
<div class="border-t pt-4 mt-6" v-if="currentLibraryItemId">
<h4 class="text-lg font-medium mb-4">本书跳过设置</h4>
<div class="flex items-center mb-4">
<ui-toggle-switch v-model="skipIntro" @input="setSkipIntro" />
<div class="pl-4 flex-1">
<span>跳过开头</span>
</div>
<ui-text-input v-model="introDuration" type="number" min="0" max="60" @input="setIntroDuration" class="w-20" />
<span class="ml-2 text-sm text-gray-400"></span>
</div>
<div class="flex items-center mb-4">
<ui-toggle-switch v-model="skipOutro" @input="setSkipOutro" />
<div class="pl-4 flex-1">
<span>跳过结尾</span>
</div>
<ui-text-input v-model="outroDuration" type="number" min="0" max="60" @input="setOutroDuration" class="w-20" />
<span class="ml-2 text-sm text-gray-400"></span>
</div>
</div>
</div>
</modals-modal>
</template>
<script>
export default {
props: {
value: Boolean
},
data() {
return {
useChapterTrack: false,
jumpValues: [
{ text: this.$getString('LabelTimeDurationXSeconds', ['10']), value: 10 },
{ text: this.$getString('LabelTimeDurationXSeconds', ['15']), value: 15 },
{ text: this.$getString('LabelTimeDurationXSeconds', ['30']), value: 30 },
{ text: this.$getString('LabelTimeDurationXSeconds', ['60']), value: 60 },
{ text: this.$getString('LabelTimeDurationXMinutes', ['2']), value: 120 },
{ text: this.$getString('LabelTimeDurationXMinutes', ['5']), value: 300 }
],
jumpForwardAmount: 10,
jumpBackwardAmount: 10,
playbackRateIncrementDecrementValues: [0.1, 0.05],
playbackRateIncrementDecrement: 0.1,
// 书籍跳过设置
currentLibraryItemId: null,
skipIntro: false,
introDuration: 10,
skipOutro: false,
outroDuration: 10
}
},
computed: {
show: {
get() {
return this.value
},
set(val) {
this.$emit('input', val)
}
}
},
methods: {
setUseChapterTrack() {
this.$store.dispatch('user/updateUserSettings', { useChapterTrack: this.useChapterTrack })
},
setJumpForwardAmount(val) {
this.jumpForwardAmount = val
this.$store.dispatch('user/updateUserSettings', { jumpForwardAmount: val })
},
setJumpBackwardAmount(val) {
this.jumpBackwardAmount = val
this.$store.dispatch('user/updateUserSettings', { jumpBackwardAmount: val })
},
setPlaybackRateIncrementDecrementAmount(val) {
this.playbackRateIncrementDecrement = val
this.$store.dispatch('user/updateUserSettings', { playbackRateIncrementDecrement: val })
},
// 书籍跳过设置方法
setSkipIntro() {
this.updateBookSkipSetting('skipIntro', this.skipIntro)
},
setIntroDuration() {
this.introDuration = Math.max(0, Math.min(60, parseInt(this.introDuration) || 0))
this.updateBookSkipSetting('introDuration', this.introDuration)
},
setSkipOutro() {
this.updateBookSkipSetting('skipOutro', this.skipOutro)
},
setOutroDuration() {
this.outroDuration = Math.max(0, Math.min(60, parseInt(this.outroDuration) || 0))
this.updateBookSkipSetting('outroDuration', this.outroDuration)
},
updateBookSkipSetting(key, value) {
if (!this.currentLibraryItemId) return
const bookSkipSettings = { ...this.$store.getters['user/getUserSetting']('bookSkipSettings') || {} }
if (!bookSkipSettings[this.currentLibraryItemId]) {
bookSkipSettings[this.currentLibraryItemId] = {}
}
bookSkipSettings[this.currentLibraryItemId][key] = value
this.$store.dispatch('user/updateUserSettings', { bookSkipSettings })
},
settingsUpdated() {
this.useChapterTrack = this.$store.getters['user/getUserSetting']('useChapterTrack')
this.jumpForwardAmount = this.$store.getters['user/getUserSetting']('jumpForwardAmount')
this.jumpBackwardAmount = this.$store.getters['user/getUserSetting']('jumpBackwardAmount')
this.playbackRateIncrementDecrement = this.$store.getters['user/getUserSetting']('playbackRateIncrementDecrement')
// 加载当前书籍的跳过设置
this.loadBookSkipSettings()
},
loadBookSkipSettings() {
// 获取当前播放的书籍ID
const mediaPlayerContainer = this.$root.$refs.mediaPlayerContainer || this.$parent.$refs.mediaPlayerContainer
if (mediaPlayerContainer && mediaPlayerContainer.streamLibraryItem) {
this.currentLibraryItemId = mediaPlayerContainer.streamLibraryItem.id
const bookSkipSettings = this.$store.getters['user/getUserSetting']('bookSkipSettings') || {}
const currentBookSettings = bookSkipSettings[this.currentLibraryItemId] || {}
this.skipIntro = currentBookSettings.skipIntro || false
this.introDuration = currentBookSettings.introDuration || 10
this.skipOutro = currentBookSettings.skipOutro || false
this.outroDuration = currentBookSettings.outroDuration || 10
} else {
this.currentLibraryItemId = null
}
}
},
mounted() {
this.settingsUpdated()
this.$eventBus.$on('user-settings', this.settingsUpdated)
this.$eventBus.$on('playback-session-changed', this.loadBookSkipSettings)
},
beforeDestroy() {
this.$eventBus.$off('user-settings', this.settingsUpdated)
this.$eventBus.$off('playback-session-changed', this.loadBookSkipSettings)
}
}
</script>