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

253 lines
6.2 KiB
Vue
Raw Normal View History

2022-04-15 14:24:30 +00:00
<template>
2022-12-02 16:17:37 +00:00
<nav id="sidebar" v-koel-clickaway="closeIfMobile" :class="{ showing: mobileShowing }" class="side side-nav">
<SearchForm />
2022-04-15 14:24:30 +00:00
<section class="music">
<h1>Your Music</h1>
<ul class="menu">
<li>
<a :class="['home', activeScreen === 'Home' ? 'active' : '']" href="#/home">
2022-12-02 16:17:37 +00:00
<icon :icon="faHome" fixed-width />
2022-07-15 07:23:55 +00:00
Home
</a>
2022-04-15 14:24:30 +00:00
</li>
<li
:class="droppableToQueue && 'droppable'"
@dragleave="onQueueDragLeave"
@dragover="onQueueDragOver"
@drop="onQueueDrop"
>
<a :class="['queue', activeScreen === 'Queue' ? 'active' : '']" href="#/queue">
2022-12-02 16:17:37 +00:00
<icon :icon="faListOl" fixed-width />
2022-04-24 08:29:14 +00:00
Current Queue
</a>
2022-04-15 14:24:30 +00:00
</li>
<li>
<a :class="['songs', activeScreen === 'Songs' ? 'active' : '']" href="#/songs">
2022-12-02 16:17:37 +00:00
<icon :icon="faMusic" fixed-width />
2022-07-15 07:23:55 +00:00
All Songs
</a>
2022-04-15 14:24:30 +00:00
</li>
<li>
<a :class="['albums', activeScreen === 'Albums' ? 'active' : '']" href="#/albums">
2022-12-02 16:17:37 +00:00
<icon :icon="faCompactDisc" fixed-width />
2022-07-15 07:23:55 +00:00
Albums
</a>
2022-04-15 14:24:30 +00:00
</li>
<li>
<a :class="['artists', activeScreen === 'Artists' ? 'active' : '']" href="#/artists">
2022-12-02 16:17:37 +00:00
<icon :icon="faMicrophone" fixed-width />
2022-07-15 07:23:55 +00:00
Artists
</a>
2022-04-15 14:24:30 +00:00
</li>
2022-10-21 20:06:43 +00:00
<li>
<a :class="['genres', activeScreen === 'Genres' ? 'active' : '']" href="#/genres">
2022-12-02 16:17:37 +00:00
<icon :icon="faTags" fixed-width />
2022-10-21 20:06:43 +00:00
Genres
</a>
</li>
2022-04-24 08:29:14 +00:00
<li v-if="useYouTube">
<a :class="['youtube', activeScreen === 'YouTube' ? 'active' : '']" href="#/youtube">
2022-12-02 16:17:37 +00:00
<icon :icon="faYoutube" fixed-width />
2022-07-15 07:23:55 +00:00
YouTube Video
</a>
2022-04-15 14:24:30 +00:00
</li>
</ul>
</section>
2022-12-02 16:17:37 +00:00
<PlaylistList />
2022-04-15 14:24:30 +00:00
<section v-if="isAdmin" class="manage">
2022-04-15 14:24:30 +00:00
<h1>Manage</h1>
<ul class="menu">
<li>
<a :class="['settings', activeScreen === 'Settings' ? 'active' : '']" href="#/settings">
2022-12-02 16:17:37 +00:00
<icon :icon="faTools" fixed-width />
2022-07-15 07:23:55 +00:00
Settings
</a>
2022-04-15 14:24:30 +00:00
</li>
<li>
<a :class="['upload', activeScreen === 'Upload' ? 'active' : '']" href="#/upload">
2022-12-02 16:17:37 +00:00
<icon :icon="faUpload" fixed-width />
2022-07-15 07:23:55 +00:00
Upload
</a>
2022-04-15 14:24:30 +00:00
</li>
<li>
<a :class="['users', activeScreen === 'Users' ? 'active' : '']" href="#/users">
2022-12-02 16:17:37 +00:00
<icon :icon="faUsers" fixed-width />
2022-07-15 07:23:55 +00:00
Users
</a>
2022-04-15 14:24:30 +00:00
</li>
</ul>
</section>
</nav>
</template>
2022-04-15 17:00:08 +00:00
<script lang="ts" setup>
2022-07-15 07:23:55 +00:00
import {
faCompactDisc,
faHome,
faListOl,
faMicrophone,
faMusic,
2022-10-21 20:06:43 +00:00
faTags,
2022-07-15 07:23:55 +00:00
faTools,
faUpload,
faUsers
} from '@fortawesome/free-solid-svg-icons'
import { faYoutube } from '@fortawesome/free-brands-svg-icons'
import { ref } from 'vue'
2022-11-18 18:44:20 +00:00
import { eventBus } from '@/utils'
2022-06-10 10:47:46 +00:00
import { queueStore } from '@/stores'
2022-11-18 18:44:20 +00:00
import { useAuthorization, useDroppable, useRouter, useThirdPartyServices } from '@/composables'
2022-04-15 14:24:30 +00:00
import PlaylistList from '@/components/playlist/PlaylistSidebarList.vue'
2022-10-13 15:18:47 +00:00
import SearchForm from '@/components/ui/SearchForm.vue'
2022-04-15 14:24:30 +00:00
2022-10-13 15:18:47 +00:00
const mobileShowing = ref(false)
const activeScreen = ref<ScreenName>()
const droppableToQueue = ref(false)
2022-11-18 18:44:20 +00:00
const { onRouteChanged } = useRouter()
const { acceptsDrop, resolveDroppedSongs } = useDroppable(['songs', 'album', 'artist', 'playlist'])
const { useYouTube } = useThirdPartyServices()
const { isAdmin } = useAuthorization()
2022-04-15 14:24:30 +00:00
const onQueueDragOver = (event: DragEvent) => {
if (!acceptsDrop(event)) return false
event.preventDefault()
event.dataTransfer!.dropEffect = 'move'
droppableToQueue.value = true
}
const onQueueDragLeave = () => (droppableToQueue.value = false)
const onQueueDrop = async (event: DragEvent) => {
droppableToQueue.value = false
if (!acceptsDrop(event)) return false
event.preventDefault()
const songs = await resolveDroppedSongs(event) || []
2022-04-15 17:00:08 +00:00
songs.length && queueStore.queue(songs)
2022-04-15 14:24:30 +00:00
2022-04-15 17:00:08 +00:00
return false
}
2022-04-15 14:24:30 +00:00
2022-10-13 15:18:47 +00:00
const closeIfMobile = () => (mobileShowing.value = false)
2022-11-18 18:44:20 +00:00
onRouteChanged(route => {
2022-10-13 15:18:47 +00:00
mobileShowing.value = false
activeScreen.value = route.screen
})
/**
* Listen to toggle sidebar event to show or hide the sidebar.
* This should only be triggered on a mobile device.
*/
eventBus.on('TOGGLE_SIDEBAR', () => (mobileShowing.value = !mobileShowing.value))
2022-04-15 14:24:30 +00:00
</script>
<style lang="scss" scoped>
nav {
2022-10-13 15:18:47 +00:00
width: var(--sidebar-width);
2022-04-15 14:24:30 +00:00
background-color: var(--color-bg-secondary);
2022-10-13 15:18:47 +00:00
padding: 2.05rem 1.5rem;
2022-04-15 14:24:30 +00:00
overflow: auto;
overflow-x: hidden;
-ms-overflow-style: -ms-autohiding-scrollbar;
box-shadow: 0 0 5px 0 rgba(0, 0, 0, 0.1);
2022-04-15 14:24:30 +00:00
> * + * {
margin-top: 2.25rem;
}
@media (hover: none) {
// Enable scroll with momentum on touch devices
overflow-y: scroll;
-webkit-overflow-scrolling: touch;
}
.droppable {
box-shadow: inset 0 0 0 1px var(--color-accent);
border-radius: 4px;
cursor: copy;
2022-04-15 14:24:30 +00:00
}
2022-08-01 08:58:25 +00:00
.queue > span {
display: flex;
align-items: baseline;
justify-content: space-between;
flex: 1;
}
2022-10-27 17:06:49 +00:00
:deep(h1) {
text-transform: uppercase;
letter-spacing: 1px;
margin-bottom: 12px;
}
2022-10-27 17:06:49 +00:00
:deep(a svg) {
2022-10-13 15:18:47 +00:00
opacity: .7;
}
2022-10-27 17:06:49 +00:00
:deep(a) {
display: flex;
align-items: center;
gap: .7rem;
height: 36px;
line-height: 36px;
white-space: nowrap;
text-overflow: ellipsis;
2022-10-13 15:18:47 +00:00
position: relative;
2022-10-13 15:18:47 +00:00
&:active {
padding: 2px 0 0 2px;
2022-04-15 14:24:30 +00:00
}
2022-10-13 15:18:47 +00:00
&.active, &:hover {
color: var(--color-text-primary);
2022-04-15 14:24:30 +00:00
}
2022-10-13 15:18:47 +00:00
&.active {
&::before {
content: '';
position: absolute;
top: 25%;
right: -1.5rem;
width: 4px;
height: 50%;
background-color: var(--color-highlight);
box-shadow: 0 0 40px 10px var(--color-highlight);
border-radius: 9999rem;
}
}
}
2022-10-27 17:06:49 +00:00
:deep(li li a) { // submenu items
2022-10-13 15:18:47 +00:00
padding-left: 11px;
&:active {
padding: 2px 0 0 13px;
}
2022-04-15 14:24:30 +00:00
}
2022-10-13 15:18:47 +00:00
@media screen and (max-width: 768px) {
2022-04-15 14:24:30 +00:00
@include themed-background();
2022-10-13 15:18:47 +00:00
transform: translateX(-100vw);
transition: transform .2s ease-in-out;
2022-04-15 14:24:30 +00:00
position: fixed;
width: 100%;
z-index: 99;
2022-10-13 15:18:47 +00:00
height: calc(100vh - var(--header-height));
2022-04-15 14:24:30 +00:00
&.showing {
2022-10-13 15:18:47 +00:00
transform: translateX(0);
2022-04-15 14:24:30 +00:00
}
}
}
</style>