koel/resources/assets/js/components/layout/main-wrapper/MainContent.vue

133 lines
5.1 KiB
Vue
Raw Normal View History

2022-04-15 14:24:30 +00:00
<template>
<section id="mainContent">
<!--
Most of the views are render-expensive and have their own UI states (viewport/scroll position), e.g. the song
lists), so we use v-show.
2022-04-15 17:00:08 +00:00
For those that don't need to maintain their own UI state, we use v-if and enjoy some code-splitting juice.
2022-04-15 14:24:30 +00:00
-->
2022-12-02 16:17:37 +00:00
<VisualizerScreen v-if="screen === 'Visualizer'" />
<AlbumArtOverlay v-if="showAlbumArtOverlay && currentSong" :album="currentSong?.album_id" />
2022-04-15 14:24:30 +00:00
2022-12-02 16:17:37 +00:00
<HomeScreen v-show="screen === 'Home'" />
<QueueScreen v-show="screen === 'Queue'" />
<AllSongsScreen v-show="screen === 'Songs'" />
<AlbumListScreen v-show="screen === 'Albums'" />
<ArtistListScreen v-show="screen === 'Artists'" />
<PlaylistScreen v-show="screen === 'Playlist'" />
<FavoritesScreen v-show="screen === 'Favorites'" />
<RecentlyPlayedScreen v-show="screen === 'RecentlyPlayed'" />
<UploadScreen v-show="screen === 'Upload'" />
<SearchExcerptsScreen v-show="screen === 'Search.Excerpt'" />
<GenreScreen v-show="screen === 'Genre'" />
2022-04-15 14:24:30 +00:00
2022-12-02 16:17:37 +00:00
<GenreListScreen v-if="screen === 'Genres'" />
<SearchSongResultsScreen v-if="screen === 'Search.Songs'" />
<AlbumScreen v-if="screen === 'Album'" />
<ArtistScreen v-if="screen === 'Artist'" />
<SettingsScreen v-if="screen === 'Settings'" />
<ProfileScreen v-if="screen === 'Profile'" />
<UserListScreen v-if="screen === 'Users'" />
<YoutubeScreen v-if="useYouTube" v-show="screen === 'YouTube'" />
<NotFoundScreen v-if="screen === '404'" />
2022-04-15 14:24:30 +00:00
</section>
</template>
2022-04-15 17:00:08 +00:00
<script lang="ts" setup>
2022-09-21 08:40:06 +00:00
import { defineAsyncComponent, onMounted, ref, toRef } from 'vue'
2022-11-06 17:09:06 +00:00
import { requireInjection } from '@/utils'
import { preferenceStore } from '@/stores'
2022-11-18 18:44:20 +00:00
import { useRouter, useThirdPartyServices } from '@/composables'
import { CurrentSongKey } from '@/symbols'
2022-07-10 14:33:33 +00:00
2022-04-23 22:01:40 +00:00
import HomeScreen from '@/components/screens/HomeScreen.vue'
import QueueScreen from '@/components/screens/QueueScreen.vue'
2022-04-21 21:51:17 +00:00
import AlbumListScreen from '@/components/screens/AlbumListScreen.vue'
import ArtistListScreen from '@/components/screens/ArtistListScreen.vue'
2022-10-21 20:06:43 +00:00
import GenreListScreen from '@/components/screens/GenreListScreen.vue'
2022-04-21 16:06:45 +00:00
import AllSongsScreen from '@/components/screens/AllSongsScreen.vue'
import PlaylistScreen from '@/components/screens/PlaylistScreen.vue'
import FavoritesScreen from '@/components/screens/FavoritesScreen.vue'
2022-07-10 14:33:33 +00:00
import RecentlyPlayedScreen from '@/components/screens/RecentlyPlayedScreen.vue'
import UploadScreen from '@/components/screens/UploadScreen.vue'
import SearchExcerptsScreen from '@/components/screens/search/SearchExcerptsScreen.vue'
2022-04-15 14:24:30 +00:00
2022-04-23 22:04:16 +00:00
const UserListScreen = defineAsyncComponent(() => import('@/components/screens/UserListScreen.vue'))
2022-04-23 22:47:21 +00:00
const AlbumArtOverlay = defineAsyncComponent(() => import('@/components/ui/AlbumArtOverlay.vue'))
const AlbumScreen = defineAsyncComponent(() => import('@/components/screens/AlbumScreen.vue'))
const ArtistScreen = defineAsyncComponent(() => import('@/components/screens/ArtistScreen.vue'))
2022-10-21 20:06:43 +00:00
const GenreScreen = defineAsyncComponent(() => import('@/components/screens/GenreScreen.vue'))
2022-04-21 22:20:21 +00:00
const SettingsScreen = defineAsyncComponent(() => import('@/components/screens/SettingsScreen.vue'))
2022-04-21 22:51:48 +00:00
const ProfileScreen = defineAsyncComponent(() => import('@/components/screens/ProfileScreen.vue'))
2022-04-23 22:10:46 +00:00
const YoutubeScreen = defineAsyncComponent(() => import('@/components/screens/YouTubeScreen.vue'))
2022-04-23 21:24:02 +00:00
const SearchSongResultsScreen = defineAsyncComponent(() => import('@/components/screens/search/SearchSongResultsScreen.vue'))
2022-09-28 04:28:00 +00:00
const NotFoundScreen = defineAsyncComponent(() => import('@/components/screens/NotFoundScreen.vue'))
2022-11-06 17:09:06 +00:00
const VisualizerScreen = defineAsyncComponent(() => import('@/components/screens/VisualizerScreen.vue'))
2022-04-15 14:24:30 +00:00
const { useYouTube } = useThirdPartyServices()
const { onRouteChanged, getCurrentScreen } = useRouter()
2022-10-13 15:18:47 +00:00
2023-08-20 22:35:58 +00:00
const currentSong = requireInjection(CurrentSongKey, ref(undefined))
const showAlbumArtOverlay = toRef(preferenceStore.state, 'show_album_art_overlay')
const screen = ref<ScreenName>('Home')
2022-04-15 14:24:30 +00:00
2022-11-18 18:44:20 +00:00
onRouteChanged(route => (screen.value = route.screen))
2022-04-15 14:24:30 +00:00
2023-08-20 22:35:58 +00:00
onMounted(async () => {
screen.value = getCurrentScreen()
})
2022-04-15 14:24:30 +00:00
</script>
<style lang="scss">
#mainContent {
flex: 1;
position: relative;
overflow: hidden;
> section {
max-height: 100%;
min-height: 100%;
2022-04-15 14:24:30 +00:00
width: 100%;
display: flex;
flex-direction: column;
backface-visibility: hidden;
-webkit-backface-visibility: hidden;
2022-04-15 14:24:30 +00:00
.main-scroll-wrap {
overflow: scroll;
display: flex;
flex-direction: column;
padding: 1.5rem;
@supports (scrollbar-gutter: stable) {
overflow: auto;
scrollbar-gutter: stable;
2022-07-29 12:12:55 +00:00
@media (hover: none) {
scrollbar-gutter: auto;
}
}
2022-04-15 14:24:30 +00:00
flex: 1;
-ms-overflow-style: -ms-autohiding-scrollbar;
place-content: start;
@media (hover: none) {
// Enable scroll with momentum on touch devices
overflow-y: scroll;
-webkit-overflow-scrolling: touch;
}
}
}
2022-10-13 15:18:47 +00:00
@media screen and (max-width: 768px) {
2022-04-15 14:24:30 +00:00
> section {
// Leave some space for the "Back to top" button
.main-scroll-wrap {
padding-bottom: 64px;
}
}
}
}
</style>