repod/src/store/player.ts

112 lines
2.7 KiB
TypeScript
Raw Normal View History

2024-09-14 16:54:56 +02:00
import type { EpisodeActionInterface, EpisodeInterface } from '../utils/types.ts'
2023-08-29 00:47:22 +02:00
import axios from '@nextcloud/axios'
import { defineStore } from 'pinia'
2024-09-14 16:54:56 +02:00
import { formatEpisodeTimestamp } from '../utils/time.ts'
2023-08-29 00:47:22 +02:00
import { generateUrl } from '@nextcloud/router'
const audio = new Audio()
export const usePlayer = defineStore('player', {
state: () => ({
currentTime: null as number | null,
duration: null as number | null,
episode: null as EpisodeInterface | null,
2023-08-29 00:47:22 +02:00
loaded: false,
paused: true,
podcastUrl: null as string | null,
2023-08-29 11:43:17 +02:00
volume: 1,
rate: 1,
2024-01-16 23:13:07 +01:00
started: 0,
}),
actions: {
2024-08-09 22:51:32 +02:00
init() {
audio.ondurationchange = () => (this.duration = audio.duration)
audio.onended = () => this.stop()
audio.onloadeddata = () => (this.loaded = true)
audio.onpause = () => this.pause()
audio.onplay = () => this.play()
audio.onratechange = () => (this.rate = audio.playbackRate)
audio.onseeked = () => (this.currentTime = audio.currentTime)
audio.ontimeupdate = () => (this.currentTime = audio.currentTime)
audio.onvolumechange = () => (this.volume = audio.volume)
},
async load(episode: EpisodeInterface | null, podcastUrl?: string) {
this.episode = episode
this.podcastUrl = podcastUrl || null
2023-08-29 00:47:22 +02:00
if (this.episode?.url) {
audio.src = this.episode.url
2023-08-29 00:47:22 +02:00
audio.load()
try {
const action = await axios.get<EpisodeActionInterface>(
generateUrl('/apps/repod/episodes/action?url={url}', {
url: this.episode.url,
}),
)
this.episode.action = action.data
} catch {}
2023-08-29 12:07:23 +02:00
if (
this.episode.action &&
this.episode.action.position < this.episode.action.total
) {
audio.currentTime = this.episode.action.position
this.started = audio.currentTime
2023-08-29 12:07:23 +02:00
}
audio.play()
2023-08-29 00:47:22 +02:00
} else {
this.loaded = false
this.podcastUrl = null
2023-08-29 00:47:22 +02:00
audio.src = ''
}
},
2024-08-09 22:51:32 +02:00
pause() {
2023-08-29 00:47:22 +02:00
audio.pause()
this.paused = true
this.time()
2023-08-29 00:47:22 +02:00
},
2024-08-09 22:51:32 +02:00
play() {
2024-01-16 22:36:01 +01:00
audio.play()
this.paused = false
this.started = audio.currentTime
2024-01-16 22:36:01 +01:00
},
seek(currentTime: number) {
2023-08-29 00:47:22 +02:00
audio.currentTime = currentTime
this.time()
2023-08-29 00:47:22 +02:00
},
2024-08-09 22:51:32 +02:00
stop() {
this.pause()
this.episode = null
2023-08-29 00:47:22 +02:00
},
2024-08-09 22:51:32 +02:00
time() {
if (!this.podcastUrl || !this.episode?.url) {
return
}
2024-09-14 16:54:56 +02:00
this.episode.action = {
podcast: this.podcastUrl,
episode: this.episode.url,
guid: this.episode.guid,
action: 'play',
timestamp: formatEpisodeTimestamp(new Date()),
started: Math.round(this.started),
position: Math.round(audio.currentTime),
total: Math.round(audio.duration),
}
axios.post(generateUrl('/apps/gpoddersync/episode_action/create'), [
this.episode.action,
])
},
setVolume(volume: number) {
2023-08-29 00:47:22 +02:00
audio.volume = volume
},
setRate(rate: number) {
audio.playbackRate = rate
},
2023-08-29 00:47:22 +02:00
},
})