2022-04-15 14:24:30 +00:00
|
|
|
<template>
|
2022-12-02 16:17:37 +00:00
|
|
|
<Overlay ref="overlay" />
|
|
|
|
<DialogBox ref="dialog" />
|
|
|
|
<MessageToaster ref="toaster" />
|
|
|
|
<GlobalEventListeners />
|
2024-03-16 18:11:08 +00:00
|
|
|
<OfflineNotification v-if="!online" />
|
2022-07-10 15:59:26 +00:00
|
|
|
|
2024-03-23 08:29:03 +00:00
|
|
|
<div v-if="layout === 'main' && initialized" id="main" @dragend="onDragEnd" @dragover="onDragOver" @drop="onDrop">
|
2022-12-02 16:17:37 +00:00
|
|
|
<Hotkeys />
|
|
|
|
<MainWrapper />
|
|
|
|
<AppFooter />
|
|
|
|
<SupportKoel />
|
|
|
|
<SongContextMenu />
|
|
|
|
<AlbumContextMenu />
|
|
|
|
<ArtistContextMenu />
|
|
|
|
<PlaylistContextMenu />
|
|
|
|
<PlaylistFolderContextMenu />
|
|
|
|
<CreateNewPlaylistContextMenu />
|
2024-02-24 15:37:01 +00:00
|
|
|
<DropZone v-show="showDropZone" @close="showDropZone = false" />
|
2022-04-15 14:24:30 +00:00
|
|
|
</div>
|
|
|
|
|
2023-08-20 22:35:58 +00:00
|
|
|
<LoginForm v-if="layout === 'auth'" @loggedin="onUserLoggedIn" />
|
|
|
|
|
|
|
|
<AcceptInvitation v-if="layout === 'invitation'" />
|
2024-02-25 19:32:53 +00:00
|
|
|
<ResetPasswordForm v-if="layout === 'reset-password'" />
|
2022-04-15 14:24:30 +00:00
|
|
|
</template>
|
|
|
|
|
|
|
|
<script lang="ts" setup>
|
2022-10-13 15:18:47 +00:00
|
|
|
import { defineAsyncComponent, nextTick, onMounted, provide, ref, watch } from 'vue'
|
2024-03-16 18:11:08 +00:00
|
|
|
import { useOnline } from '@vueuse/core'
|
2022-10-13 15:18:47 +00:00
|
|
|
import { commonStore, preferenceStore as preferences, queueStore } from '@/stores'
|
2022-10-30 23:13:57 +00:00
|
|
|
import { authService, socketListener, socketService, uploadService } from '@/services'
|
2022-11-19 18:04:21 +00:00
|
|
|
import { CurrentSongKey, DialogBoxKey, MessageToasterKey, OverlayKey } from '@/symbols'
|
2024-03-16 18:11:08 +00:00
|
|
|
import { useRouter } from '@/composables'
|
2022-04-15 14:24:30 +00:00
|
|
|
|
2022-07-26 09:51:19 +00:00
|
|
|
import DialogBox from '@/components/ui/DialogBox.vue'
|
|
|
|
import MessageToaster from '@/components/ui/MessageToaster.vue'
|
2022-09-08 05:06:49 +00:00
|
|
|
import Overlay from '@/components/ui/Overlay.vue'
|
2022-10-27 13:44:40 +00:00
|
|
|
import OfflineNotification from '@/components/ui/OfflineNotification.vue'
|
2022-09-08 05:06:49 +00:00
|
|
|
|
|
|
|
// Do not dynamic-import app footer, as it contains the <audio> element
|
|
|
|
// that is necessary to properly initialize the playService and equalizer.
|
|
|
|
import AppFooter from '@/components/layout/app-footer/index.vue'
|
2022-04-15 14:24:30 +00:00
|
|
|
|
2022-10-07 14:33:50 +00:00
|
|
|
// GlobalEventListener must NOT be lazy-loaded, so that it can handle LOG_OUT event properly.
|
2024-03-16 18:11:08 +00:00
|
|
|
import GlobalEventListeners from '@/components/utils/GlobalEventListeners.vue'
|
2022-10-07 14:33:50 +00:00
|
|
|
|
2022-09-08 05:06:49 +00:00
|
|
|
const Hotkeys = defineAsyncComponent(() => import('@/components/utils/HotkeyListener.vue'))
|
|
|
|
const LoginForm = defineAsyncComponent(() => import('@/components/auth/LoginForm.vue'))
|
|
|
|
const MainWrapper = defineAsyncComponent(() => import('@/components/layout/main-wrapper/index.vue'))
|
|
|
|
const AlbumContextMenu = defineAsyncComponent(() => import('@/components/album/AlbumContextMenu.vue'))
|
|
|
|
const ArtistContextMenu = defineAsyncComponent(() => import('@/components/artist/ArtistContextMenu.vue'))
|
|
|
|
const PlaylistContextMenu = defineAsyncComponent(() => import('@/components/playlist/PlaylistContextMenu.vue'))
|
|
|
|
const PlaylistFolderContextMenu = defineAsyncComponent(() => import('@/components/playlist/PlaylistFolderContextMenu.vue'))
|
|
|
|
const SongContextMenu = defineAsyncComponent(() => import('@/components/song/SongContextMenu.vue'))
|
|
|
|
const CreateNewPlaylistContextMenu = defineAsyncComponent(() => import('@/components/playlist/CreateNewPlaylistContextMenu.vue'))
|
2022-04-24 08:29:14 +00:00
|
|
|
const SupportKoel = defineAsyncComponent(() => import('@/components/meta/SupportKoel.vue'))
|
2022-09-12 11:11:56 +00:00
|
|
|
const DropZone = defineAsyncComponent(() => import('@/components/ui/upload/DropZone.vue'))
|
2023-08-20 22:35:58 +00:00
|
|
|
const AcceptInvitation = defineAsyncComponent(() => import('@/components/invitation/AcceptInvitation.vue'))
|
2024-02-25 19:32:53 +00:00
|
|
|
const ResetPasswordForm = defineAsyncComponent(() => import('@/components/auth/ResetPasswordForm.vue'))
|
2022-04-15 14:24:30 +00:00
|
|
|
|
2022-11-19 18:04:21 +00:00
|
|
|
const overlay = ref<InstanceType<typeof Overlay>>()
|
2022-07-26 09:51:19 +00:00
|
|
|
const dialog = ref<InstanceType<typeof DialogBox>>()
|
2022-09-08 05:06:49 +00:00
|
|
|
const toaster = ref<InstanceType<typeof MessageToaster>>()
|
2022-12-19 15:44:48 +00:00
|
|
|
const currentSong = ref<Song>()
|
2022-09-12 11:11:56 +00:00
|
|
|
const showDropZone = ref(false)
|
2022-04-20 10:27:10 +00:00
|
|
|
|
2024-02-25 19:32:53 +00:00
|
|
|
const layout = ref<'main' | 'auth' | 'invitation' | 'reset-password'>()
|
2023-08-20 22:35:58 +00:00
|
|
|
|
2024-02-25 19:32:53 +00:00
|
|
|
const { isCurrentScreen, getCurrentScreen, resolveRoute } = useRouter()
|
2024-03-16 18:11:08 +00:00
|
|
|
const online = useOnline()
|
2022-10-27 13:44:40 +00:00
|
|
|
|
2022-04-15 14:24:30 +00:00
|
|
|
/**
|
|
|
|
* Request for notification permission if it's not provided and the user is OK with notifications.
|
|
|
|
*/
|
|
|
|
const requestNotificationPermission = async () => {
|
2024-01-23 22:50:50 +00:00
|
|
|
if (preferences.show_now_playing_notification && window.Notification && window.Notification.permission !== 'granted') {
|
|
|
|
preferences.show_now_playing_notification = await window.Notification.requestPermission() === 'denied'
|
2022-04-15 14:24:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-08 05:06:49 +00:00
|
|
|
const onUserLoggedIn = async () => {
|
2023-08-20 22:35:58 +00:00
|
|
|
layout.value = 'main'
|
2022-09-08 05:06:49 +00:00
|
|
|
await init()
|
2022-04-15 14:24:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
onMounted(async () => {
|
2024-03-31 17:19:03 +00:00
|
|
|
// If the user is authenticated via a proxy, we have the token in the window object.
|
|
|
|
// Simply forward it to the authService and continue with the normal flow.
|
|
|
|
if (window.AUTH_TOKEN) {
|
|
|
|
authService.setTokensUsingCompositeToken(window.AUTH_TOKEN)
|
|
|
|
}
|
|
|
|
|
2022-04-15 14:24:30 +00:00
|
|
|
// The app has just been initialized, check if we can get the user data with an already existing token
|
2022-11-16 17:57:38 +00:00
|
|
|
if (authService.hasApiToken()) {
|
2022-04-15 14:24:30 +00:00
|
|
|
await init()
|
2023-08-20 22:35:58 +00:00
|
|
|
|
|
|
|
// call resolveRoute() after init() so that the onResolve hooks can use the stores
|
|
|
|
await resolveRoute()
|
|
|
|
layout.value = 'main'
|
|
|
|
} else {
|
|
|
|
await resolveRoute()
|
2024-02-25 19:32:53 +00:00
|
|
|
|
|
|
|
switch (getCurrentScreen()) {
|
|
|
|
case 'Invitation.Accept':
|
|
|
|
layout.value = 'invitation'
|
|
|
|
break
|
|
|
|
case 'Password.Reset':
|
|
|
|
layout.value = 'reset-password'
|
|
|
|
break
|
|
|
|
default:
|
|
|
|
layout.value = 'auth'
|
|
|
|
}
|
2022-04-15 14:24:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Add an ugly mac/non-mac class for OS-targeting styles.
|
|
|
|
// I'm crying inside.
|
2022-07-25 12:57:58 +00:00
|
|
|
document.documentElement.classList.add(navigator.userAgent.includes('Mac') ? 'mac' : 'non-mac')
|
2022-04-15 14:24:30 +00:00
|
|
|
})
|
|
|
|
|
2024-03-23 08:29:03 +00:00
|
|
|
const initialized = ref(false)
|
|
|
|
|
2022-04-15 14:24:30 +00:00
|
|
|
const init = async () => {
|
2022-12-02 16:17:37 +00:00
|
|
|
overlay.value!.show({ message: 'Just a little patience…' })
|
2022-04-15 14:24:30 +00:00
|
|
|
|
|
|
|
try {
|
2022-04-24 08:50:45 +00:00
|
|
|
await commonStore.init()
|
2024-03-23 08:29:03 +00:00
|
|
|
initialized.value = true
|
2022-04-15 14:24:30 +00:00
|
|
|
|
2022-07-25 08:35:15 +00:00
|
|
|
await requestNotificationPermission()
|
2022-04-15 14:24:30 +00:00
|
|
|
|
2022-08-04 10:39:03 +00:00
|
|
|
window.addEventListener('beforeunload', (e: BeforeUnloadEvent) => {
|
2024-01-23 22:50:50 +00:00
|
|
|
if (uploadService.shouldWarnUponWindowUnload() || preferences.confirm_before_closing) {
|
2022-08-04 10:39:03 +00:00
|
|
|
e.preventDefault()
|
|
|
|
e.returnValue = ''
|
2022-07-25 08:35:15 +00:00
|
|
|
}
|
|
|
|
})
|
2022-07-24 10:53:49 +00:00
|
|
|
|
|
|
|
await socketService.init() && socketListener.listen()
|
2022-04-15 14:24:30 +00:00
|
|
|
} catch (err) {
|
2023-08-20 22:35:58 +00:00
|
|
|
layout.value = 'auth'
|
2022-04-15 14:24:30 +00:00
|
|
|
throw err
|
2023-12-22 13:46:53 +00:00
|
|
|
} finally {
|
|
|
|
overlay.value!.hide()
|
2022-04-15 14:24:30 +00:00
|
|
|
}
|
|
|
|
}
|
2022-07-26 09:51:19 +00:00
|
|
|
|
2022-09-12 11:11:56 +00:00
|
|
|
const onDragOver = (e: DragEvent) => {
|
2022-11-18 18:44:20 +00:00
|
|
|
showDropZone.value = Boolean(e.dataTransfer?.types.includes('Files')) && !isCurrentScreen('Upload')
|
2022-09-12 11:11:56 +00:00
|
|
|
}
|
|
|
|
|
2022-12-19 15:44:48 +00:00
|
|
|
watch(() => queueStore.current, song => (currentSong.value = song))
|
2022-10-13 15:18:47 +00:00
|
|
|
|
2022-09-12 11:11:56 +00:00
|
|
|
const onDragEnd = () => (showDropZone.value = false)
|
|
|
|
const onDrop = () => (showDropZone.value = false)
|
|
|
|
|
2022-11-19 18:04:21 +00:00
|
|
|
provide(OverlayKey, overlay)
|
2022-07-26 09:51:19 +00:00
|
|
|
provide(DialogBoxKey, dialog)
|
|
|
|
provide(MessageToasterKey, toaster)
|
2022-10-13 15:18:47 +00:00
|
|
|
provide(CurrentSongKey, currentSong)
|
2022-04-15 14:24:30 +00:00
|
|
|
</script>
|
|
|
|
|
|
|
|
<style lang="scss">
|
2022-04-15 17:00:08 +00:00
|
|
|
@import "#/app.scss";
|
2022-04-15 14:24:30 +00:00
|
|
|
|
|
|
|
#dragGhost {
|
|
|
|
display: inline-block;
|
|
|
|
background: var(--color-green);
|
|
|
|
padding: .8rem;
|
2022-07-26 09:51:19 +00:00
|
|
|
border-radius: .3rem;
|
2022-04-15 14:24:30 +00:00
|
|
|
color: var(--color-text-primary);
|
|
|
|
font-family: var(--font-family);
|
|
|
|
font-size: 1rem;
|
|
|
|
font-weight: var(--font-weight-light);
|
|
|
|
position: fixed;
|
|
|
|
top: 0;
|
|
|
|
left: 0;
|
|
|
|
z-index: -1;
|
|
|
|
|
|
|
|
@media (hover: none) {
|
|
|
|
display: none;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#copyArea {
|
|
|
|
position: absolute;
|
|
|
|
left: -9999px;
|
|
|
|
width: 1px;
|
|
|
|
height: 1px;
|
|
|
|
bottom: 1px;
|
|
|
|
|
|
|
|
@media (hover: none) {
|
|
|
|
display: none;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-20 22:35:58 +00:00
|
|
|
#main {
|
2022-04-15 14:24:30 +00:00
|
|
|
display: flex;
|
|
|
|
height: 100vh;
|
|
|
|
flex-direction: column;
|
2022-09-08 05:06:49 +00:00
|
|
|
justify-content: flex-end;
|
2022-04-15 14:24:30 +00:00
|
|
|
}
|
|
|
|
|
2022-10-13 15:18:47 +00:00
|
|
|
#main {
|
|
|
|
@media screen and (max-width: 768px) {
|
2022-10-22 07:29:51 +00:00
|
|
|
position: absolute;
|
|
|
|
height: 100%;
|
|
|
|
width: 100%;
|
|
|
|
top: 0;
|
2022-10-13 15:18:47 +00:00
|
|
|
padding-top: var(--header-height);
|
|
|
|
}
|
|
|
|
}
|
2022-04-15 14:24:30 +00:00
|
|
|
</style>
|