koel/resources/assets/js/components/song/EditSongForm.vue

369 lines
10 KiB
Vue
Raw Normal View History

2022-04-15 14:24:30 +00:00
<template>
2022-04-21 09:38:24 +00:00
<div class="edit-song" data-testid="edit-song-form" tabindex="0" @keydown.esc="maybeClose">
2022-08-01 08:58:25 +00:00
<SoundBars v-if="loading"/>
2022-04-15 14:24:30 +00:00
<form v-else @submit.prevent="submit">
<header>
2022-07-05 22:28:26 +00:00
<span class="cover" :style="{ backgroundImage: `url(${coverUrl})` }"/>
2022-04-21 09:38:24 +00:00
<div class="meta">
2022-04-15 14:24:30 +00:00
<h1 :class="{ mixed: !editingOnlyOneSong }">{{ displayedTitle }}</h1>
<h2
data-testid="displayed-artist-name"
:class="{ mixed: !allSongsAreFromSameArtist && !formData.artist_name }"
>
{{ displayedArtistName }}
</h2>
<h2
data-testid="displayed-album-name"
:class="{ mixed: !allSongsAreInSameAlbum && !formData.album_name }"
>
{{ displayedAlbumName }}
</h2>
2022-04-21 09:38:24 +00:00
</div>
2022-04-15 14:24:30 +00:00
</header>
2022-07-22 16:15:30 +00:00
<main class="tabs">
2022-04-15 14:24:30 +00:00
<div class="clear" role="tablist">
<button
id="editSongTabDetails"
2022-04-15 14:24:30 +00:00
:aria-selected="currentView === 'details'"
aria-controls="editSongPanelDetails"
role="tab"
2022-06-10 10:47:46 +00:00
type="button"
@click.prevent="currentView = 'details'"
2022-04-15 14:24:30 +00:00
>
Details
</button>
<button
v-if="editingOnlyOneSong"
id="editSongTabLyrics"
2022-04-15 14:24:30 +00:00
:aria-selected="currentView === 'lyrics'"
aria-controls="editSongPanelLyrics"
data-testid="edit-song-lyrics-tab"
role="tab"
2022-06-10 10:47:46 +00:00
type="button"
@click.prevent="currentView = 'lyrics'"
2022-04-15 14:24:30 +00:00
>
Lyrics
</button>
</div>
<div class="panes">
<div
2022-04-21 09:38:24 +00:00
v-show="currentView === 'details'"
2022-04-15 14:24:30 +00:00
id="editSongPanelDetails"
2022-04-21 09:38:24 +00:00
aria-labelledby="editSongTabDetails"
2022-04-15 14:24:30 +00:00
role="tabpanel"
tabindex="0"
>
<div v-if="editingOnlyOneSong" class="form-row">
2022-07-22 16:15:30 +00:00
<label>
Title
<input
v-model="formData.title"
v-koel-focus
data-testid="title-input"
name="title"
title="Title"
type="text"
>
</label>
2022-04-15 14:24:30 +00:00
</div>
<div class="form-row">
<div class="cols">
<label>
Artist
<input
v-model="formData.artist_name"
:placeholder="inputPlaceholder"
data-testid="artist-input"
name="artist"
type="text"
>
</label>
<label>
Album Artist
<input
v-model="formData.album_artist_name"
:placeholder="inputPlaceholder"
data-testid="albumArtist-input"
name="album_artist"
type="text"
>
</label>
</div>
2022-04-15 14:24:30 +00:00
</div>
<div class="form-row">
2022-07-22 16:15:30 +00:00
<label>
Album
<input
v-model="formData.album_name"
:placeholder="inputPlaceholder"
data-testid="album-input"
name="album"
type="text"
>
</label>
2022-04-15 14:24:30 +00:00
</div>
<div class="form-row">
<div class="cols">
<label>
Track
<input
v-model="formData.track"
:placeholder="inputPlaceholder"
data-testid="track-input"
min="1"
name="track"
type="number"
>
</label>
<label>
Disc
<input
v-model="formData.disc"
:placeholder="inputPlaceholder"
data-testid="disc-input"
min="1"
name="disc"
type="number"
>
</label>
</div>
2022-04-15 14:24:30 +00:00
</div>
2022-06-10 10:47:46 +00:00
<div class="form-row">
<div class="cols">
<label>
Genre
<input
v-model="formData.genre"
:placeholder="inputPlaceholder"
data-testid="genre-input"
name="genre"
type="text"
list="genres"
>
<datalist id="genres">
<option v-for="genre in genres" :key="genre" :value="genre"/>
</datalist>
</label>
<label>
Year
<input
v-model="formData.year"
:placeholder="inputPlaceholder"
data-testid="year-input"
name="year"
type="number"
>
</label>
2022-06-10 10:47:46 +00:00
</div>
2022-04-15 14:24:30 +00:00
</div>
</div>
<div
2022-04-21 09:38:24 +00:00
v-if="editingOnlyOneSong"
v-show="currentView === 'lyrics'"
2022-04-15 14:24:30 +00:00
id="editSongPanelLyrics"
2022-04-21 09:38:24 +00:00
aria-labelledby="editSongTabLyrics"
2022-04-15 14:24:30 +00:00
role="tabpanel"
tabindex="0"
>
<div class="form-row">
<textarea
v-model="formData.lyrics"
v-koel-focus
data-testid="lyrics-input"
name="lyrics"
title="Lyrics"
/>
2022-04-15 14:24:30 +00:00
</div>
</div>
</div>
2022-07-22 16:15:30 +00:00
</main>
2022-04-15 14:24:30 +00:00
<footer>
2022-04-15 17:00:08 +00:00
<Btn type="submit">Update</Btn>
2022-04-21 09:38:24 +00:00
<Btn class="btn-cancel" white @click.prevent="maybeClose">Cancel</Btn>
2022-04-15 14:24:30 +00:00
</footer>
</form>
</div>
</template>
2022-04-15 17:00:08 +00:00
<script lang="ts" setup>
2022-07-20 08:00:02 +00:00
import { computed, onMounted, reactive, ref } from 'vue'
2022-06-10 10:47:46 +00:00
import { isEqual } from 'lodash'
import { defaultCover, eventBus, pluralize, requireInjection } from '@/utils'
2022-07-24 11:47:18 +00:00
import { songStore, SongUpdateData } from '@/stores'
import { DialogBoxKey, EditSongFormInitialTabKey, MessageToasterKey, SongsKey } from '@/symbols'
import { genres } from '@/config'
2022-04-15 17:00:08 +00:00
import Btn from '@/components/ui/Btn.vue'
2022-08-01 08:58:25 +00:00
import SoundBars from '@/components/ui/SoundBars.vue'
const toaster = requireInjection(MessageToasterKey)
const dialog = requireInjection(DialogBoxKey)
2022-07-20 08:00:02 +00:00
const [initialTab] = requireInjection(EditSongFormInitialTabKey)
const [songs] = requireInjection(SongsKey)
2022-07-10 15:17:48 +00:00
const currentView = ref<EditSongFormTabName>('details')
2022-06-10 10:47:46 +00:00
const loading = ref(false)
2022-04-15 14:24:30 +00:00
2022-07-20 08:00:02 +00:00
const mutatedSongs = computed(() => songs.value)
2022-04-15 17:00:08 +00:00
/**
* In order not to mess up the original songs, we manually assign and manipulate their attributes.
*/
2022-07-24 11:47:18 +00:00
const formData = reactive<SongUpdateData>({
2022-04-15 17:00:08 +00:00
title: '',
2022-06-10 10:47:46 +00:00
album_name: '',
artist_name: '',
album_artist_name: '',
2022-04-15 17:00:08 +00:00
lyrics: '',
track: null,
disc: null,
year: null,
genre: ''
2022-04-15 17:00:08 +00:00
})
2022-04-15 14:24:30 +00:00
2022-04-30 08:16:28 +00:00
const initialFormData = {}
2022-04-15 17:00:08 +00:00
const editingOnlyOneSong = computed(() => mutatedSongs.value.length === 1)
2022-06-10 10:47:46 +00:00
const inputPlaceholder = computed(() => editingOnlyOneSong.value ? '' : 'Leave unchanged')
2022-04-15 17:00:08 +00:00
2022-06-10 10:47:46 +00:00
const allSongsAreFromSameArtist = computed(() => allSongsShareSameValue('artist_name'))
const allSongsAreInSameAlbum = computed(() => allSongsShareSameValue('album_name'))
2022-07-05 22:28:26 +00:00
const coverUrl = computed(() => allSongsAreInSameAlbum.value
? mutatedSongs.value[0].album_cover || defaultCover
: defaultCover
)
2022-06-10 10:47:46 +00:00
2022-07-24 11:47:18 +00:00
const allSongsShareSameValue = (key: keyof SongUpdateData) => {
2022-06-10 10:47:46 +00:00
if (editingOnlyOneSong.value) return true
return new Set(mutatedSongs.value.map(song => song[key])).size === 1
}
2022-04-15 17:00:08 +00:00
const displayedTitle = computed(() => {
return editingOnlyOneSong.value ? formData.title : `${mutatedSongs.value.length} songs selected`
})
const displayedArtistName = computed(() => {
2022-06-10 10:47:46 +00:00
return allSongsAreFromSameArtist.value || formData.artist_name ? formData.artist_name : 'Mixed Artists'
2022-04-15 17:00:08 +00:00
})
2022-04-15 14:24:30 +00:00
2022-04-15 17:00:08 +00:00
const displayedAlbumName = computed(() => {
2022-06-10 10:47:46 +00:00
return allSongsAreInSameAlbum.value || formData.album_name ? formData.album_name : 'Mixed Albums'
2022-04-15 17:00:08 +00:00
})
2022-04-21 09:38:24 +00:00
const isPristine = computed(() => isEqual(formData, initialFormData))
2022-04-15 17:00:08 +00:00
const open = async () => {
2022-04-21 09:38:24 +00:00
currentView.value = initialTab.value
2022-04-15 17:00:08 +00:00
const firstSong = mutatedSongs.value[0]
2022-04-15 14:24:30 +00:00
2022-06-10 10:47:46 +00:00
formData.title = allSongsShareSameValue('title') ? firstSong.title : ''
formData.album_name = allSongsShareSameValue('album_name') ? firstSong.album_name : ''
formData.artist_name = allSongsShareSameValue('artist_name') ? firstSong.artist_name : ''
// If the album artist(s) is the same as the artist(s), we set the form value as empty to not confuse the user
// and make it less error-prone.
if (
allSongsShareSameValue('artist_name') && allSongsShareSameValue('album_artist_name')
&& firstSong.album_artist_id === firstSong.artist_id
) {
formData.album_artist_name = ''
} else {
formData.album_artist_name = allSongsShareSameValue('album_artist_name') ? firstSong.album_artist_name : ''
}
2022-06-10 10:47:46 +00:00
formData.lyrics = editingOnlyOneSong.value ? firstSong.lyrics : ''
2022-06-10 10:47:46 +00:00
formData.track = allSongsShareSameValue('track') ? firstSong.track : null
formData.track = formData.track || null // if 0, just don't show it
2022-06-10 10:47:46 +00:00
formData.disc = allSongsShareSameValue('disc') ? firstSong.disc : null
formData.disc = formData.disc || null // if 0, just don't show it
2022-04-21 09:38:24 +00:00
formData.year = allSongsShareSameValue('year') ? firstSong.year : null
formData.genre = allSongsShareSameValue('genre') ? firstSong.genre : ''
if (!editingOnlyOneSong.value) {
delete formData.title
delete formData.lyrics
}
2022-04-30 08:16:28 +00:00
Object.assign(initialFormData, formData)
2022-04-15 17:00:08 +00:00
}
const emit = defineEmits(['close'])
const close = () => emit('close')
const maybeClose = async () => {
2022-04-15 17:00:08 +00:00
if (isPristine.value) {
close()
return
2022-04-15 14:24:30 +00:00
}
2022-04-15 17:00:08 +00:00
await dialog.value.confirm('Discard all changes?') && close()
2022-04-15 17:00:08 +00:00
}
const submit = async () => {
loading.value = true
try {
await songStore.update(mutatedSongs.value, formData)
toaster.value.success(`Updated ${pluralize(mutatedSongs.value, 'song')}.`)
eventBus.emit('SONGS_UPDATED')
2022-04-15 17:00:08 +00:00
close()
} finally {
loading.value = false
}
}
2022-07-10 15:17:48 +00:00
onMounted(async () => await open())
2022-04-15 14:24:30 +00:00
</script>
<style lang="scss" scoped>
form {
2022-06-10 10:47:46 +00:00
max-width: 540px;
2022-04-15 14:24:30 +00:00
.tabs {
margin-top: 1.125rem;
2022-04-15 14:24:30 +00:00
padding: 0;
}
> header {
2022-07-05 22:28:26 +00:00
gap: 1.2rem;
.cover {
flex: 0 0 84px;
height: 84px;
background-size: cover;
border-radius: 5px;
2022-04-15 14:24:30 +00:00
}
.meta {
flex: 1;
.mixed {
opacity: .5;
}
}
}
2022-06-10 10:47:46 +00:00
.form-row .cols {
display: flex;
place-content: space-between;
gap: 1rem;
> * {
2022-06-10 10:47:46 +00:00
flex: 1;
}
}
2022-04-15 14:24:30 +00:00
}
</style>