2022-04-15 14:24:30 +00:00
|
|
|
<template>
|
|
|
|
<li
|
|
|
|
@dblclick.prevent="makeEditable"
|
|
|
|
:class="['playlist', type, editing ? 'editing' : '', playlist.is_smart ? 'smart' : '']">
|
|
|
|
<a
|
|
|
|
:class="{ active }"
|
|
|
|
:href="url"
|
2022-04-30 11:55:54 +00:00
|
|
|
@contextmenu.prevent="openContextMenu"
|
2022-04-30 13:20:47 +00:00
|
|
|
v-koel-droppable:[contentEditable]="handleDrop"
|
2022-04-15 14:24:30 +00:00
|
|
|
>{{ playlist.name }}</a>
|
|
|
|
|
2022-04-15 17:00:08 +00:00
|
|
|
<NameEditor
|
2022-04-15 14:24:30 +00:00
|
|
|
:playlist="playlist"
|
|
|
|
@cancelled="cancelEditing"
|
|
|
|
@updated="onPlaylistNameUpdated"
|
|
|
|
v-if="nameEditable && editing"
|
|
|
|
/>
|
|
|
|
|
2022-04-15 17:00:08 +00:00
|
|
|
<ContextMenu
|
2022-04-15 14:24:30 +00:00
|
|
|
v-if="hasContextMenu"
|
|
|
|
v-show="showingContextMenu"
|
|
|
|
:playlist="playlist"
|
|
|
|
ref="contextMenu"
|
|
|
|
@edit="makeEditable"
|
|
|
|
/>
|
|
|
|
</li>
|
|
|
|
</template>
|
|
|
|
|
2022-04-15 17:00:08 +00:00
|
|
|
<script lang="ts" setup>
|
|
|
|
import { computed, defineAsyncComponent, nextTick, ref, toRefs } from 'vue'
|
2022-04-24 17:58:12 +00:00
|
|
|
import { alerts, eventBus, pluralize } from '@/utils'
|
2022-04-15 17:00:08 +00:00
|
|
|
import { favoriteStore, playlistStore, songStore } from '@/stores'
|
2022-04-30 13:20:47 +00:00
|
|
|
import router from '@/router'
|
2022-04-15 14:24:30 +00:00
|
|
|
|
2022-04-15 17:00:08 +00:00
|
|
|
type PlaylistType = 'playlist' | 'favorites' | 'recently-played'
|
2022-04-15 14:24:30 +00:00
|
|
|
|
2022-04-20 10:35:36 +00:00
|
|
|
const ContextMenu = defineAsyncComponent(() => import('@/components/playlist/PlaylistContextMenu.vue'))
|
2022-04-24 08:29:14 +00:00
|
|
|
const NameEditor = defineAsyncComponent(() => import('@/components/playlist/PlaylistNameEditor.vue'))
|
2022-04-15 14:24:30 +00:00
|
|
|
|
2022-04-20 10:35:36 +00:00
|
|
|
const contextMenu = ref<InstanceType<typeof ContextMenu>>()
|
2022-04-15 14:24:30 +00:00
|
|
|
|
2022-04-15 17:00:08 +00:00
|
|
|
const props = withDefaults(defineProps<{ playlist: Playlist, type: PlaylistType }>(), { type: 'playlist' })
|
|
|
|
const { playlist, type } = toRefs(props)
|
2022-04-15 14:24:30 +00:00
|
|
|
|
2022-04-15 17:00:08 +00:00
|
|
|
const editing = ref(false)
|
|
|
|
const active = ref(false)
|
|
|
|
const showingContextMenu = ref(false)
|
2022-04-15 14:24:30 +00:00
|
|
|
|
2022-04-15 17:00:08 +00:00
|
|
|
const url = computed(() => {
|
|
|
|
switch (type.value) {
|
|
|
|
case 'playlist':
|
|
|
|
return `#!/playlist/${playlist.value.id}`
|
2022-04-15 14:24:30 +00:00
|
|
|
|
2022-04-15 17:00:08 +00:00
|
|
|
case 'favorites':
|
|
|
|
return '#!/favorites'
|
2022-04-15 14:24:30 +00:00
|
|
|
|
2022-04-15 17:00:08 +00:00
|
|
|
case 'recently-played':
|
|
|
|
return '#!/recently-played'
|
2022-04-15 14:24:30 +00:00
|
|
|
|
2022-04-15 17:00:08 +00:00
|
|
|
default:
|
|
|
|
throw new Error('Invalid playlist type')
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
const nameEditable = computed(() => type.value === 'playlist')
|
|
|
|
const hasContextMenu = computed(() => type.value === 'playlist')
|
|
|
|
|
|
|
|
const contentEditable = computed(() => {
|
|
|
|
if (playlist.value.is_smart) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return type.value === 'playlist' || type.value === 'favorites'
|
|
|
|
})
|
|
|
|
|
|
|
|
const makeEditable = () => {
|
|
|
|
if (!nameEditable.value) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
editing.value = true
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handle songs dropped to our favorite or playlist menu item.
|
|
|
|
*/
|
|
|
|
const handleDrop = (event: DragEvent) => {
|
|
|
|
if (!contentEditable.value) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!event.dataTransfer?.getData('application/x-koel.text+plain')) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
const songs = songStore.byIds(event.dataTransfer.getData('application/x-koel.text+plain').split(','))
|
|
|
|
|
|
|
|
if (!songs.length) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
if (type.value === 'favorites') {
|
|
|
|
favoriteStore.like(songs)
|
|
|
|
} else if (type.value === 'playlist') {
|
|
|
|
playlistStore.addSongs(playlist.value, songs)
|
2022-04-24 17:58:12 +00:00
|
|
|
alerts.success(`Added ${pluralize(songs.length, 'song')} into "${playlist.value.name}."`)
|
2022-04-15 17:00:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
const openContextMenu = async (event: MouseEvent) => {
|
|
|
|
if (hasContextMenu.value) {
|
|
|
|
showingContextMenu.value = true
|
|
|
|
await nextTick()
|
|
|
|
router.go(`/playlist/${playlist.value.id}`)
|
2022-04-20 10:35:36 +00:00
|
|
|
contextMenu.value?.open(event.pageY, event.pageX, { playlist })
|
2022-04-15 17:00:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const cancelEditing = () => (editing.value = false)
|
|
|
|
|
|
|
|
const onPlaylistNameUpdated = (mutatedPlaylist: Playlist) => {
|
|
|
|
playlist.value.name = mutatedPlaylist.name
|
|
|
|
editing.value = false
|
|
|
|
}
|
|
|
|
|
|
|
|
eventBus.on('LOAD_MAIN_CONTENT', (view: MainViewName, _playlist: Playlist): void => {
|
|
|
|
switch (view) {
|
|
|
|
case 'Favorites':
|
|
|
|
active.value = type.value === 'favorites'
|
|
|
|
break
|
|
|
|
|
|
|
|
case 'RecentlyPlayed':
|
|
|
|
active.value = type.value === 'recently-played'
|
|
|
|
|
|
|
|
break
|
|
|
|
case 'Playlist':
|
|
|
|
active.value = playlist.value === _playlist
|
|
|
|
break
|
|
|
|
|
|
|
|
default:
|
|
|
|
active.value = false
|
|
|
|
break
|
2022-04-15 14:24:30 +00:00
|
|
|
}
|
|
|
|
})
|
|
|
|
</script>
|
|
|
|
|
|
|
|
<style lang="scss" scoped>
|
|
|
|
.playlist {
|
|
|
|
user-select: none;
|
|
|
|
|
|
|
|
a {
|
|
|
|
white-space: nowrap;
|
|
|
|
overflow: hidden;
|
|
|
|
text-overflow: ellipsis;
|
|
|
|
|
|
|
|
span {
|
|
|
|
pointer-events: none;
|
|
|
|
}
|
|
|
|
|
|
|
|
&::before {
|
|
|
|
content: "\f0f6";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
&.favorites a::before {
|
|
|
|
content: "\f004";
|
|
|
|
color: var(--color-maroon);
|
|
|
|
}
|
|
|
|
|
|
|
|
&.recently-played a::before {
|
|
|
|
content: "\f1da";
|
|
|
|
color: var(--color-green);
|
|
|
|
}
|
|
|
|
|
|
|
|
&.smart a::before {
|
|
|
|
content: "\f069";
|
|
|
|
}
|
|
|
|
|
|
|
|
input {
|
|
|
|
width: calc(100% - 32px);
|
|
|
|
margin: 5px 16px;
|
|
|
|
}
|
|
|
|
|
|
|
|
&.editing {
|
|
|
|
a {
|
|
|
|
display: none !important;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
</style>
|