mirror of
https://github.com/owntone/owntone-server.git
synced 2025-04-04 11:50:44 -04:00
This reverts commit bb2a778b46afca0fcd56bda6f02857fad78b14d8, reversing changes made to f8e2298b677476f46eb67cc4331c18124f92a791.
349 lines
9.6 KiB
Vue
349 lines
9.6 KiB
Vue
<template>
|
|
<div id="app">
|
|
<navbar-top />
|
|
<vue-progress-bar class="has-background-info" />
|
|
<router-view v-slot="{ Component }">
|
|
<component :is="Component" />
|
|
</router-view>
|
|
|
|
<modal-dialog-remote-pairing
|
|
:show="pairing_active"
|
|
@close="pairing_active = false"
|
|
/>
|
|
<modal-dialog-update
|
|
:show="show_update_dialog"
|
|
@close="show_update_dialog = false"
|
|
/>
|
|
<notification-list v-show="!show_burger_menu" />
|
|
<navbar-bottom />
|
|
<div
|
|
v-show="show_burger_menu || show_player_menu"
|
|
class="fd-overlay-fullscreen"
|
|
@click="show_burger_menu = show_player_menu = false"
|
|
/>
|
|
</div>
|
|
</template>
|
|
|
|
<script>
|
|
import ModalDialogRemotePairing from '@/components/ModalDialogRemotePairing.vue'
|
|
import ModalDialogUpdate from '@/components/ModalDialogUpdate.vue'
|
|
import NavbarBottom from '@/components/NavbarBottom.vue'
|
|
import NavbarTop from '@/components/NavbarTop.vue'
|
|
import NotificationList from '@/components/NotificationList.vue'
|
|
import ReconnectingWebSocket from 'reconnectingwebsocket'
|
|
import { useConfigurationStore } from '@/stores/configuration'
|
|
import { useLibraryStore } from '@/stores/library'
|
|
import { useLyricsStore } from '@/stores/lyrics'
|
|
import { useNotificationsStore } from '@/stores/notifications'
|
|
import { useOutputsStore } from './stores/outputs'
|
|
import { usePlayerStore } from '@/stores/player'
|
|
import { useQueueStore } from '@/stores/queue'
|
|
import { useRemotesStore } from './stores/remotes'
|
|
import { useServicesStore } from '@/stores/services'
|
|
import { useSettingsStore } from '@/stores/settings'
|
|
import { useUIStore } from './stores/ui'
|
|
import webapi from '@/webapi'
|
|
|
|
export default {
|
|
name: 'App',
|
|
components: {
|
|
ModalDialogRemotePairing,
|
|
ModalDialogUpdate,
|
|
NavbarBottom,
|
|
NavbarTop,
|
|
NotificationList
|
|
},
|
|
|
|
setup() {
|
|
return {
|
|
configurationStore: useConfigurationStore(),
|
|
libraryStore: useLibraryStore(),
|
|
lyricsStore: useLyricsStore(),
|
|
notificationsStore: useNotificationsStore(),
|
|
outputsStore: useOutputsStore(),
|
|
playerStore: usePlayerStore(),
|
|
queueStore: useQueueStore(),
|
|
remotesStore: useRemotesStore(),
|
|
servicesStore: useServicesStore(),
|
|
settingsStore: useSettingsStore(),
|
|
uiStore: useUIStore()
|
|
}
|
|
},
|
|
|
|
data() {
|
|
return {
|
|
pairing_active: false,
|
|
reconnect_attempts: 0,
|
|
token_timer_id: 0
|
|
}
|
|
},
|
|
|
|
computed: {
|
|
show_burger_menu: {
|
|
get() {
|
|
return this.uiStore.show_burger_menu
|
|
},
|
|
set(value) {
|
|
this.uiStore.show_burger_menu = value
|
|
}
|
|
},
|
|
show_player_menu: {
|
|
get() {
|
|
return this.uiStore.show_player_menu
|
|
},
|
|
set(value) {
|
|
this.uiStore.show_player_menu = value
|
|
}
|
|
},
|
|
show_update_dialog: {
|
|
get() {
|
|
return this.uiStore.show_update_dialog
|
|
},
|
|
set(value) {
|
|
this.uiStore.show_update_dialog = value
|
|
}
|
|
}
|
|
},
|
|
|
|
watch: {
|
|
show_burger_menu() {
|
|
this.update_is_clipped()
|
|
},
|
|
show_player_menu() {
|
|
this.update_is_clipped()
|
|
}
|
|
},
|
|
|
|
created() {
|
|
this.connect()
|
|
// Start the progress bar on app start
|
|
this.$Progress.start()
|
|
// Hook the progress bar to start before we move router-view
|
|
this.$router.beforeEach((to, from, next) => {
|
|
if (to.meta.show_progress && !(to.path === from.path && to.hash)) {
|
|
if (to.meta.progress) {
|
|
this.$Progress.parseMeta(to.meta.progress)
|
|
}
|
|
this.$Progress.start()
|
|
}
|
|
next()
|
|
})
|
|
// Hook the progress bar to finish after we've finished moving router-view
|
|
this.$router.afterEach((to, from) => {
|
|
if (to.meta.show_progress) {
|
|
this.$Progress.finish()
|
|
}
|
|
})
|
|
},
|
|
|
|
methods: {
|
|
connect() {
|
|
webapi
|
|
.config()
|
|
.then(({ data }) => {
|
|
this.configurationStore.$state = data
|
|
this.uiStore.hide_singles = data.hide_singles
|
|
document.title = data.library_name
|
|
this.open_websocket()
|
|
this.$Progress.finish()
|
|
})
|
|
.catch(() => {
|
|
this.notificationsStore.add({
|
|
text: this.$t('server.connection-failed'),
|
|
topic: 'connection',
|
|
type: 'danger'
|
|
})
|
|
})
|
|
},
|
|
open_websocket() {
|
|
const socket = this.create_websocket()
|
|
const vm = this
|
|
socket.onopen = () => {
|
|
vm.reconnect_attempts = 0
|
|
socket.send(
|
|
JSON.stringify({
|
|
notify: [
|
|
'update',
|
|
'database',
|
|
'player',
|
|
'options',
|
|
'outputs',
|
|
'volume',
|
|
'queue',
|
|
'spotify',
|
|
'lastfm',
|
|
'pairing'
|
|
]
|
|
})
|
|
)
|
|
vm.update_outputs()
|
|
vm.update_player_status()
|
|
vm.update_library_stats()
|
|
vm.update_settings()
|
|
vm.update_queue()
|
|
vm.update_spotify()
|
|
vm.update_lastfm()
|
|
vm.update_pairing()
|
|
}
|
|
|
|
/*
|
|
* When the app becomes active, force an update of all information,
|
|
* because we may have missed notifications while the app was inactive.
|
|
* There are two relevant events - focus and visibilitychange, so we
|
|
* throttle the updates to avoid multiple redundant updates.
|
|
*/
|
|
let update_throttled = false
|
|
|
|
const update_info = () => {
|
|
if (update_throttled) {
|
|
return
|
|
}
|
|
vm.update_outputs()
|
|
vm.update_player_status()
|
|
vm.update_library_stats()
|
|
vm.update_settings()
|
|
vm.update_queue()
|
|
vm.update_spotify()
|
|
vm.update_lastfm()
|
|
vm.update_pairing()
|
|
update_throttled = true
|
|
setTimeout(() => {
|
|
update_throttled = false
|
|
}, 500)
|
|
}
|
|
|
|
/*
|
|
* These events are fired when the window becomes active in different
|
|
* ways. When this happens, we should update 'now playing' info, etc.
|
|
*/
|
|
window.addEventListener('focus', update_info)
|
|
document.addEventListener('visibilitychange', () => {
|
|
if (document.visibilityState === 'visible') {
|
|
update_info()
|
|
}
|
|
})
|
|
|
|
socket.onmessage = (response) => {
|
|
const data = JSON.parse(response.data)
|
|
if (
|
|
data.notify.includes('update') ||
|
|
data.notify.includes('database')
|
|
) {
|
|
vm.update_library_stats()
|
|
}
|
|
if (
|
|
data.notify.includes('player') ||
|
|
data.notify.includes('options') ||
|
|
data.notify.includes('volume')
|
|
) {
|
|
vm.update_player_status()
|
|
}
|
|
if (data.notify.includes('outputs') || data.notify.includes('volume')) {
|
|
vm.update_outputs()
|
|
}
|
|
if (data.notify.includes('queue')) {
|
|
vm.update_queue()
|
|
}
|
|
if (data.notify.includes('spotify')) {
|
|
vm.update_spotify()
|
|
}
|
|
if (data.notify.includes('lastfm')) {
|
|
vm.update_lastfm()
|
|
}
|
|
if (data.notify.includes('pairing')) {
|
|
vm.update_pairing()
|
|
}
|
|
}
|
|
},
|
|
create_websocket() {
|
|
const protocol = window.location.protocol.replace('http', 'ws')
|
|
const hostname =
|
|
(import.meta.env.DEV &&
|
|
URL.parse(import.meta.env.VITE_OWNTONE_URL)?.hostname) ||
|
|
window.location.hostname
|
|
const suffix =
|
|
this.configurationStore.websocket_port || `${window.location.port}/ws`
|
|
const url = `${protocol}${hostname}:${suffix}`
|
|
return new ReconnectingWebSocket(url, 'notify', {
|
|
maxReconnectInterval: 2000,
|
|
reconnectInterval: 1000
|
|
})
|
|
},
|
|
update_is_clipped() {
|
|
if (this.show_burger_menu || this.show_player_menu) {
|
|
document.querySelector('html').classList.add('is-clipped')
|
|
} else {
|
|
document.querySelector('html').classList.remove('is-clipped')
|
|
}
|
|
},
|
|
update_lastfm() {
|
|
webapi.lastfm().then(({ data }) => {
|
|
this.servicesStore.lastfm = data
|
|
})
|
|
},
|
|
update_library_stats() {
|
|
webapi.library_stats().then(({ data }) => {
|
|
this.libraryStore.$state = data
|
|
})
|
|
webapi.library_count('scan_kind is rss').then(({ data }) => {
|
|
this.libraryStore.rss = data
|
|
})
|
|
},
|
|
update_lyrics() {
|
|
const track = this.queueStore.current
|
|
if (track?.track_id) {
|
|
webapi.library_track(track.track_id).then(({ data }) => {
|
|
this.lyricsStore.lyrics = data.lyrics
|
|
})
|
|
} else {
|
|
this.lyricsStore.$reset()
|
|
}
|
|
},
|
|
update_outputs() {
|
|
webapi.outputs().then(({ data }) => {
|
|
this.outputsStore.outputs = data.outputs
|
|
})
|
|
},
|
|
update_pairing() {
|
|
webapi.pairing().then(({ data }) => {
|
|
this.remotesStore.$state = data
|
|
this.pairing_active = data.active
|
|
})
|
|
},
|
|
update_player_status() {
|
|
webapi.player_status().then(({ data }) => {
|
|
this.playerStore.$state = data
|
|
this.update_lyrics()
|
|
})
|
|
},
|
|
update_queue() {
|
|
webapi.queue().then(({ data }) => {
|
|
this.queueStore.$state = data
|
|
this.update_lyrics()
|
|
})
|
|
},
|
|
update_settings() {
|
|
webapi.settings().then(({ data }) => {
|
|
this.settingsStore.$state = data
|
|
})
|
|
},
|
|
update_spotify() {
|
|
webapi.spotify().then(({ data }) => {
|
|
this.servicesStore.spotify = data
|
|
if (this.token_timer_id > 0) {
|
|
window.clearTimeout(this.token_timer_id)
|
|
this.token_timer_id = 0
|
|
}
|
|
if (data.webapi_token_expires_in > 0 && data.webapi_token) {
|
|
this.token_timer_id = window.setTimeout(
|
|
this.update_spotify,
|
|
1000 * data.webapi_token_expires_in
|
|
)
|
|
}
|
|
})
|
|
}
|
|
},
|
|
template: '<App/>'
|
|
}
|
|
</script>
|