koel/resources/assets/js/components/playlist/PlaylistSidebarItem.vue

139 lines
3.7 KiB
Vue

<template>
<li
ref="el"
:class="{ droppable }"
class="playlist"
draggable="true"
@contextmenu="onContextMenu"
@dragleave="onDragLeave"
@dragover="onDragOver"
@dragstart="onDragStart"
@drop="onDrop"
>
<a :class="{ active }" :href="url">
<icon v-if="isRecentlyPlayedList(list)" :icon="faClockRotateLeft" class="text-green" fixed-width />
<icon v-else-if="isFavoriteList(list)" :icon="faHeart" class="text-maroon" fixed-width />
<icon v-else-if="list.is_smart" :icon="faWandMagicSparkles" fixed-width />
<icon v-else :icon="faFileLines" fixed-width />
{{ list.name }}
</a>
</li>
</template>
<script lang="ts" setup>
import { faClockRotateLeft, faFileLines, faHeart, faWandMagicSparkles } from '@fortawesome/free-solid-svg-icons'
import { computed, ref, toRefs } from 'vue'
import { eventBus } from '@/utils'
import { favoriteStore } from '@/stores'
import { useDraggable, useDroppable, usePlaylistManagement, useRouter } from '@/composables'
const { onRouteChanged } = useRouter()
const { startDragging } = useDraggable('playlist')
const { acceptsDrop, resolveDroppedSongs } = useDroppable(['songs', 'album', 'artist'])
const droppable = ref(false)
const { addSongsToPlaylist } = usePlaylistManagement()
const props = defineProps<{ list: PlaylistLike }>()
const { list } = toRefs(props)
const isPlaylist = (list: PlaylistLike): list is Playlist => 'id' in list
const isFavoriteList = (list: PlaylistLike): list is FavoriteList => list.name === 'Favorites'
const isRecentlyPlayedList = (list: PlaylistLike): list is RecentlyPlayedList => list.name === 'Recently Played'
const active = ref(false)
const url = computed(() => {
if (isPlaylist(list.value)) return `#/playlist/${list.value.id}`
if (isFavoriteList(list.value)) return '#/favorites'
if (isRecentlyPlayedList(list.value)) return '#/recently-played'
throw new Error('Invalid playlist-like type.')
})
const contentEditable = computed(() => {
if (isRecentlyPlayedList(list.value)) return false
if (isFavoriteList(list.value)) return true
return !list.value.is_smart
})
const onContextMenu = (event: MouseEvent) => {
if (isPlaylist(list.value)) {
event.preventDefault()
eventBus.emit('PLAYLIST_CONTEXT_MENU_REQUESTED', event, list.value)
}
}
const onDragStart = (event: DragEvent) => isPlaylist(list.value) && startDragging(event, list.value)
const onDragOver = (event: DragEvent) => {
if (!contentEditable.value) return false
if (!acceptsDrop(event)) return false
event.preventDefault()
droppable.value = true
return false
}
const onDragLeave = () => (droppable.value = false)
const onDrop = async (event: DragEvent) => {
droppable.value = false
if (!contentEditable.value) return false
if (!acceptsDrop(event)) return false
const songs = await resolveDroppedSongs(event)
if (!songs?.length) return false
if (isFavoriteList(list.value)) {
await favoriteStore.like(songs)
} else if (isPlaylist(list.value)) {
await addSongsToPlaylist(list.value, songs)
}
return false
}
onRouteChanged(route => {
switch (route.screen) {
case 'Favorites':
active.value = isFavoriteList(list.value)
break
case 'RecentlyPlayed':
active.value = isRecentlyPlayedList(list.value)
break
case 'Playlist':
active.value = (list.value as Playlist).id === parseInt(route.params!.id)
break
default:
active.value = false
break
}
})
</script>
<style lang="scss" scoped>
.playlist {
user-select: none;
&.droppable {
box-shadow: inset 0 0 0 1px var(--color-accent);
border-radius: 4px;
cursor: copy;
}
:deep(a) {
span {
pointer-events: none;
}
}
}
</style>