mirror of
https://github.com/koel/koel
synced 2024-12-24 11:33:05 +00:00
217 lines
4.6 KiB
Vue
217 lines
4.6 KiB
Vue
<template>
|
|
<div
|
|
class="add-to"
|
|
v-show="showing"
|
|
tabindex="0"
|
|
v-koel-clickaway="close"
|
|
v-koel-focus
|
|
@keydown.esc="close"
|
|
data-test="add-to-menu"
|
|
>
|
|
<section class="existing-playlists">
|
|
<p>Add {{ songs.length | pluralize('song') }} to</p>
|
|
|
|
<ul>
|
|
<template v-if="config.queue">
|
|
<li class="after-current" @click="queueSongsAfterCurrent" role="button" tabindex="0">After Current Song</li>
|
|
<li class="bottom-queue" @click="queueSongsToBottom" role="button" tabindex="0">Bottom of Queue</li>
|
|
<li class="top-queue" @click="queueSongsToTop" role="button" tabindex="0">Top of Queue</li>
|
|
</template>
|
|
|
|
<li
|
|
@click="addSongsToFavorite"
|
|
class="favorites"
|
|
tabindex="0"
|
|
v-if="config.favorites"
|
|
>
|
|
Favorites
|
|
</li>
|
|
|
|
<template v-if="config.playlists">
|
|
<li
|
|
:key="playlist.id"
|
|
@click="addSongsToExistingPlaylist(playlist)"
|
|
class="playlist"
|
|
tabindex="0"
|
|
v-for="playlist in playlists"
|
|
>
|
|
{{ playlist.name }}
|
|
</li>
|
|
</template>
|
|
</ul>
|
|
</section>
|
|
|
|
<section class="new-playlist" v-if="config.newPlaylist">
|
|
<p>or create a new playlist</p>
|
|
|
|
<form class="form-save form-simple form-new-playlist" @submit.prevent="createNewPlaylistFromSongs">
|
|
<input
|
|
@keyup.esc.prevent="close"
|
|
required
|
|
type="text"
|
|
placeholder="Playlist name"
|
|
v-model="newPlaylistName"
|
|
data-test="new-playlist-name"
|
|
>
|
|
<btn type="submit" title="Save">⏎</btn>
|
|
</form>
|
|
</section>
|
|
</div>
|
|
</template>
|
|
|
|
<script lang="ts">
|
|
import mixins from 'vue-typed-mixins'
|
|
import { pluralize } from '@/utils'
|
|
import { playlistStore } from '@/stores'
|
|
import router from '@/router'
|
|
import songMenuMethods from '@/mixins/song-menu-methods.ts'
|
|
import { PropOptions } from 'vue'
|
|
|
|
export default mixins(songMenuMethods).extend({
|
|
components: {
|
|
Btn: () => import('@/components/ui/btn.vue')
|
|
},
|
|
|
|
props: {
|
|
showing: {
|
|
type: Boolean,
|
|
default: false
|
|
},
|
|
config: {
|
|
type: Object
|
|
} as PropOptions<AddToMenuConfig>
|
|
},
|
|
|
|
filters: { pluralize },
|
|
|
|
data: () => ({
|
|
newPlaylistName: '',
|
|
playlistState: playlistStore.state
|
|
}),
|
|
|
|
watch: {
|
|
songs (): void {
|
|
if (!this.songs.length) {
|
|
this.close()
|
|
}
|
|
}
|
|
},
|
|
|
|
computed: {
|
|
playlists (): Playlist[] {
|
|
return this.playlistState.playlists.filter(playlist => !playlist.is_smart)
|
|
}
|
|
},
|
|
|
|
methods: {
|
|
/**
|
|
* Save the selected songs as a playlist.
|
|
* As of current we don't have selective save.
|
|
*/
|
|
async createNewPlaylistFromSongs (): Promise<void> {
|
|
this.newPlaylistName = this.newPlaylistName.trim()
|
|
|
|
if (!this.newPlaylistName) {
|
|
return
|
|
}
|
|
|
|
const playlist = await playlistStore.store(this.newPlaylistName, this.songs)
|
|
this.newPlaylistName = ''
|
|
// Activate the new playlist right away
|
|
this.$nextTick((): void => router.go(`playlist/${playlist.id}`))
|
|
|
|
this.close()
|
|
},
|
|
|
|
close (): void {
|
|
this.$emit('closing')
|
|
}
|
|
}
|
|
})
|
|
</script>
|
|
|
|
<style lang="scss" scoped>
|
|
.add-to {
|
|
@include context-menu();
|
|
|
|
width: 100%;
|
|
max-width: 225px;
|
|
padding: .75rem;
|
|
|
|
> * + * {
|
|
margin-top: 1rem;
|
|
}
|
|
|
|
p {
|
|
margin-bottom: .5rem;
|
|
font-size: .9rem;
|
|
}
|
|
|
|
.new-playlist {
|
|
margin-top: .5rem;
|
|
}
|
|
|
|
ul {
|
|
max-height: 12rem;
|
|
overflow-y: scroll;
|
|
-webkit-overflow-scrolling: touch;
|
|
|
|
> li + li {
|
|
margin-top: .3rem;
|
|
}
|
|
}
|
|
|
|
li {
|
|
height: 2.25rem;
|
|
line-height: 2.25rem;
|
|
padding: 0 .75rem;
|
|
white-space: nowrap;
|
|
overflow: hidden;
|
|
text-overflow: ellipsis;
|
|
border-radius: 3px;
|
|
background: rgba(255, 255, 255, .05);
|
|
cursor: pointer;
|
|
|
|
&:hover {
|
|
background: var(--color-highlight);
|
|
color: var(--color-text-primary);
|
|
}
|
|
}
|
|
|
|
&::before {
|
|
display: block;
|
|
content: " ";
|
|
width: 0;
|
|
height: 0;
|
|
border-left: 10px solid transparent;
|
|
border-right: 10px solid transparent;
|
|
border-bottom: 10px solid var(--color-bg-secondary);
|
|
position: absolute;
|
|
top: -7px;
|
|
left: calc(50% - 10px);
|
|
}
|
|
|
|
form {
|
|
width: 100%;
|
|
display: flex;
|
|
justify-content: center;
|
|
align-items: center;
|
|
|
|
input[type="text"] {
|
|
width: 100%;
|
|
border-radius: 5px 0 0 5px;
|
|
height: 28px;
|
|
}
|
|
|
|
button[type="submit"] {
|
|
margin-top: 0;
|
|
border-radius: 0 5px 5px 0 !important;
|
|
height: 28px;
|
|
line-height: 28px;
|
|
padding-top: 0;
|
|
padding-bottom: 0;
|
|
margin-left: -2px !important;
|
|
}
|
|
}
|
|
}
|
|
</style>
|