koel/resources/assets/js/composables/useSongList.ts
2022-07-21 09:54:36 +02:00

128 lines
4 KiB
TypeScript

import { orderBy, sampleSize, take } from 'lodash'
import isMobile from 'ismobilejs'
import { computed, reactive, Ref, ref } from 'vue'
import { playbackService } from '@/services'
import { queueStore, songStore } from '@/stores'
import router from '@/router'
import {
SelectedSongsKey,
SongListConfigKey,
SongListSortFieldKey,
SongListSortOrderKey,
SongListTypeKey,
SongsKey
} from '@/symbols'
import ControlsToggle from '@/components/ui/ScreenControlsToggle.vue'
import SongList from '@/components/song/SongList.vue'
import SongListControls from '@/components/song/SongListControls.vue'
import ThumbnailStack from '@/components/ui/ThumbnailStack.vue'
import { provideReadonly } from '@/utils'
export const useSongList = (
songs: Ref<Song[]>,
type: SongListType,
config: Partial<SongListConfig> = {}
) => {
const songList = ref<InstanceType<typeof SongList>>()
const isPhone = isMobile.phone
const selectedSongs = ref<Song[]>([])
const showingControls = ref(false)
const headerLayout = ref<ScreenHeaderLayout>('expanded')
const onScrollBreakpoint = (direction: 'up' | 'down') => {
headerLayout.value = direction === 'down' ? 'collapsed' : 'expanded'
}
const duration = computed(() => songStore.getFormattedLength(songs.value))
const thumbnails = computed(() => {
const songsWithCover = songs.value.filter(song => song.album_cover)
const sampleCovers = sampleSize(songsWithCover, 20).map(song => song.album_cover)
return take(Array.from(new Set(sampleCovers)), 4)
})
const getSongsToPlay = (): Song[] => songList.value.getAllSongsWithSort()
const playAll = (shuffle: boolean) => playbackService.queueAndPlay(getSongsToPlay(), shuffle)
const playSelected = (shuffle: boolean) => playbackService.queueAndPlay(selectedSongs.value, shuffle)
const toggleControls = () => (showingControls.value = !showingControls.value)
const onPressEnter = async (event: KeyboardEvent) => {
if (selectedSongs.value.length === 1) {
queueStore.queueIfNotQueued(selectedSongs.value[0])
await playbackService.play(selectedSongs.value[0])
return
}
// • Only Enter: Queue songs to bottom
// • Shift+Enter: Queues song to top
// • Cmd/Ctrl+Enter: Queues song to bottom and play the first selected song
// • Cmd/Ctrl+Shift+Enter: Queue songs to top and play the first queued song
event.shiftKey ? queueStore.queueToTop(selectedSongs.value) : queueStore.queue(selectedSongs.value)
if (event.ctrlKey || event.metaKey) {
await playbackService.play(selectedSongs.value[0])
}
router.go('/queue')
}
const sortField = ref<SongListSortField | null>(((): SongListSortField | null => {
if (type === 'album' || type === 'artist') return 'track'
if (type === 'search-results') return null
return config.sortable ? 'title' : null
})())
const sortOrder = ref<SortOrder>('asc')
const sort = (by: SongListSortField | null = sortField.value, order: SortOrder = sortOrder.value) => {
if (!by) return
sortField.value = by
sortOrder.value = order
let sortFields: SongListSortField[] = [by]
if (by === 'track') {
sortFields.push('disc', 'title')
} else if (by === 'album_name') {
sortFields.push('artist_name', 'track', 'disc', 'title')
} else if (by === 'artist_name') {
sortFields.push('album_name', 'track', 'disc', 'title')
}
songs.value = orderBy(songs.value, sortFields, order)
}
provideReadonly(SongListTypeKey, type)
provideReadonly(SongsKey, songs, false)
provideReadonly(SelectedSongsKey, selectedSongs, false)
provideReadonly(SongListConfigKey, reactive(config))
provideReadonly(SongListSortFieldKey, sortField)
provideReadonly(SongListSortOrderKey, sortOrder)
return {
SongList,
SongListControls,
ControlsToggle,
ThumbnailStack,
songs,
headerLayout,
sortField,
sortOrder,
duration,
thumbnails,
songList,
selectedSongs,
showingControls,
isPhone,
onPressEnter,
playAll,
playSelected,
toggleControls,
onScrollBreakpoint,
sort
}
}