-
Notifications
You must be signed in to change notification settings - Fork 8
/
Copy pathBookDetails.vue
161 lines (154 loc) · 5.48 KB
/
BookDetails.vue
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
<template>
<div class="flex flex-col overflow-auto py-16 lg:py-0">
<div class="text pt-3 lg:pt-8 p-8 pb-5 flex flex-col justify-center">
<div class="lg:hidden"><BreadCrumbs /></div>
<div class="w-full relative cover-container bg-gray-300 dark:bg-gray-800 justify-center flex items-center rounded-md shadow-inner">
<Cover :book="book" :path="$route.query.folder" />
</div>
</div>
<div class="text-xl flex justify-center px-8 font-bold">{{ name }}</div>
<button @click="listen" class="text-white bg-pink-600 font-normal rounded-lg mx-8 my-4 p-3"><i class="fa-light fa-circle-play"></i> Listen</button>
<template v-if="description">
<div :class="{ active: readmore }" class="text px-8 description-text text-sm readmore" v-html="description"></div>
<div @click="readmore = !readmore" class="px-8 text-pink-600 font-normal text-sm cursor-pointer">{{ moretext }}</div>
</template>
<div class="px-8 mt-3 font-normal text-sm">Progress</div>
<div class="flex px-8 pt-1 items-center">
<div class="relative flex-grow mr-2">
<div class="overflow-hidden h-1 text-xs flex rounded bg-gray-300">
<div :style="{ width: percent + '%'}" class="shadow-none flex flex-col text-center whitespace-nowrap text-white justify-center bg-pink-600"></div>
</div>
</div>
<div class="text-sm">{{ percent }}%</div>
</div>
<div class="px-8 text-xs">{{ remaining }}</div>
<div class="px-8 pt-8 flex items-center justify-between">
<div v-if="cacheavailable" class="cursor-pointer relative">
<span @click="download" v-show="!cached && !downloading" class="fa-layers fa-fw fa-2x">
<i class="fa-solid fa-circle"></i>
<i class="fa-inverse fa-light fa-arrow-down-to-line" data-fa-transform="shrink-8"></i>
</span>
<span v-show="downloading" class="fa-layers fa-fw fa-2x">
<i class="fa-solid fa-circle"></i>
<i class="fa-inverse fa-thin fa-down-to-line fa-beat" data-fa-transform="shrink-8"></i>
</span>
<span v-show="cached" class="fa-layers fa-fw fa-2x">
<i class="fa-solid fa-circle"></i>
<i class="fa-inverse fa-thin fa-trash-can" data-fa-transform="shrink-8"></i>
</span>
<span v-show="downloading" class="fa-layers fa-fw fa-2x absolute inset-0 opacity-70">
<i class="fa-solid fa-circle"></i>
<i class="fa-inverse fa-light fa-spinner-third fa-spin" data-fa-transform="shrink-2"></i>
</span>
</div>
<div @click="loadPlayer" class="cursor-pointer">Show player</div>
</div>
</div>
</template>
<script>
import { mapState } from 'vuex'
export default {
name: 'BookDetails',
props: ['details', 'name', 'server'],
data () {
return {
readmore: false,
cached: false,
downloading: false,
image: null
}
},
computed: {
...mapState('player', ['current', 'currentFile', 'player', 'playing']),
moretext () {
return (this.readmore) ? 'Less..' : 'More..'
},
description () {
return this.$store.state.app.book.description
},
seek () {
return this.$store.state.app.book.seek
},
book () {
return this.$store.state.app.book
},
hash () {
return this.$store.getters['app/hash'](this.$route.query.folder)
},
totalTime () {
let total = 0
this.details.files.forEach((file) => {
total += file.meta.duration
})
return total
},
remaining () {
const remaining = this.totalTime - this.seek
return this.$formatToTime(remaining, 2, false) + ' remaining'
},
percent () {
const remaining = this.totalTime - this.seek
const percent = (remaining / this.totalTime) * 100
return 100 - percent.toFixed(0)
},
cacheavailable () {
return 'caches' in window
}
},
/* async */ mounted () {
// const keys = await caches.keys()
// const path = (this.details && this.details.description) ? this.details.description.path : null
// const cover = (this.details && this.details.cover) ? this.server + 'cover/' + this.details.cover.path : null
/* */
},
methods: {
async listen () {
if (this.playing) {
this.$store.commit('app/player', true)
} else {
await this.loadPlayer()
const autorewind = await this.$store.dispatch('app/autoRewind')
this.player.currentTime = this.current - autorewind
this.$store.dispatch('app/savePlayEvent', this.currentFile.start + this.player.currentTime)
this.player.play()
}
},
async loadPlayer () {
await this.$store.dispatch('player/load')
this.$store.commit('app/player', true)
},
async download () {
this.downloading = true
const isPersisted = await navigator.storage.persist()
console.log(`Persisted storage granted: ${isPersisted}`)
const cacheName = this.$store.state.app.cacheKey + this.hash
await caches.delete(cacheName)
const cacheStorage = await caches.open(cacheName)
await Promise.all(this.details.files.map((file) => {
const req = new Request(this.$store.getters['app/getFileUrl'](file.path),
{
credentials: 'include'
})
return cacheStorage.add(req)
}))
this.downloading = false
this.cached = true
}
}
}
</script>
<style lang="scss">
.description-text {
p {
margin: 15px 0;
}
}
.readmore {
height: 95px;
overflow: hidden;
&.active {
height: auto;
overflow: visible;
}
}
</style>