mirror of
https://github.com/koel/koel
synced 2025-01-19 07:54:01 +00:00
245 lines
6.5 KiB
TypeScript
245 lines
6.5 KiB
TypeScript
|
import Vue from 'vue'
|
||
|
import slugify from 'slugify'
|
||
|
import { without, take, remove, orderBy, unionBy } from 'lodash'
|
||
|
import isMobile from 'ismobilejs'
|
||
|
|
||
|
import { secondsToHis, alerts, pluralize, use } from '@/utils'
|
||
|
import { http, auth, ls } from '@/services'
|
||
|
import { sharedStore, favoriteStore, albumStore, artistStore, preferenceStore } from '.'
|
||
|
import stub from '@/stubs/song'
|
||
|
|
||
|
interface BroadcastSongData {
|
||
|
song: {
|
||
|
id: string
|
||
|
title: string
|
||
|
liked: boolean
|
||
|
playbackState: PlaybackState
|
||
|
album: {
|
||
|
name: string
|
||
|
cover: string
|
||
|
}
|
||
|
artist: {
|
||
|
name: string
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
interface SongUpdateResult {
|
||
|
songs: Song[]
|
||
|
artists: Artist[]
|
||
|
albums: Album[]
|
||
|
}
|
||
|
|
||
|
export const songStore = {
|
||
|
stub,
|
||
|
cache: {} as { [key: string]: Song },
|
||
|
|
||
|
state: {
|
||
|
songs: [] as Song[],
|
||
|
recentlyPlayed: [] as Song[]
|
||
|
},
|
||
|
|
||
|
init (songs: Song[]): void {
|
||
|
this.all = songs
|
||
|
this.all.forEach(song => this.setupSong(song))
|
||
|
},
|
||
|
|
||
|
setupSong (song: Song): void {
|
||
|
song.fmtLength = secondsToHis(song.length)
|
||
|
|
||
|
const album = albumStore.byId(song.album_id)!
|
||
|
const artist = artistStore.byId(song.artist_id)!
|
||
|
|
||
|
song.playCount = song.playCount || 0
|
||
|
song.album = album
|
||
|
song.artist = artist
|
||
|
song.liked = song.liked || false
|
||
|
song.lyrics = song.lyrics || ''
|
||
|
song.playbackState = song.playbackState || 'Stopped'
|
||
|
|
||
|
artist.songs = unionBy(artist.songs || [], [song], 'id')
|
||
|
album.songs = unionBy(album.songs || [], [song], 'id')
|
||
|
|
||
|
// now if the song is part of a compilation album, the album must be added
|
||
|
// into its artist as well
|
||
|
if (album.is_compilation) {
|
||
|
artist.albums = unionBy(artist.albums, [album], 'id')
|
||
|
}
|
||
|
|
||
|
// Cache the song, so that byId() is faster
|
||
|
this.cache[song.id] = song
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Initializes the interaction (like/play count) information.
|
||
|
*
|
||
|
* @param {Interaction[]} interactions The array of interactions of the current user
|
||
|
*/
|
||
|
initInteractions (interactions: Interaction[]): void {
|
||
|
favoriteStore.clear()
|
||
|
|
||
|
interactions.forEach(interaction => {
|
||
|
const song = this.byId(interaction.song_id)
|
||
|
|
||
|
if (!song) {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
song.liked = interaction.liked
|
||
|
song.playCount = interaction.play_count
|
||
|
song.album.playCount += song.playCount
|
||
|
song.artist.playCount += song.playCount
|
||
|
|
||
|
song.liked && favoriteStore.add(song)
|
||
|
})
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Get the total duration of some songs.
|
||
|
*
|
||
|
* @param songs
|
||
|
* @param {Boolean} formatted Whether to convert the duration into H:i:s format
|
||
|
*/
|
||
|
getLength: (songs: Song[], formatted: boolean = false): number | string => {
|
||
|
const duration = songs.reduce((length, song) => length + song.length, 0)
|
||
|
|
||
|
return formatted ? secondsToHis(duration) : duration
|
||
|
},
|
||
|
|
||
|
getFormattedLength (songs: Song[]): string {
|
||
|
return <string>this.getLength(songs, true)
|
||
|
},
|
||
|
|
||
|
get all (): Song[] {
|
||
|
return this.state.songs
|
||
|
},
|
||
|
|
||
|
set all (value: Song[]) {
|
||
|
this.state.songs = value
|
||
|
},
|
||
|
|
||
|
byId (id: string): Song | undefined {
|
||
|
return this.cache[id]
|
||
|
},
|
||
|
|
||
|
byIds (ids: string[]): Song[] {
|
||
|
const songs = [] as Song[]
|
||
|
([] as string[]).concat(ids).forEach(id => use(this.byId(id), song => songs.push(song!)))
|
||
|
return songs
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Guess a song by its title and album.
|
||
|
* Forget about Levenshtein distance, this implementation is good enough.
|
||
|
*/
|
||
|
guess: (title: string, album: Album): Song | null => {
|
||
|
title = slugify(title.toLowerCase())
|
||
|
|
||
|
for (const song of album.songs) {
|
||
|
if (slugify(song.title.toLowerCase()) === title) {
|
||
|
return song
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return null
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Increase a play count for a song.
|
||
|
*/
|
||
|
registerPlay: async (song: Song): Promise<void> => {
|
||
|
const oldCount = song.playCount
|
||
|
|
||
|
const interaction = await http.post<Interaction>('interaction/play', { song: song.id })
|
||
|
|
||
|
// Use the data from the server to make sure we don't miss a play from another device.
|
||
|
song.playCount = interaction.play_count
|
||
|
song.album.playCount += song.playCount - oldCount
|
||
|
song.artist.playCount += song.playCount - oldCount
|
||
|
},
|
||
|
|
||
|
scrobble: async (song: Song): Promise<void> => {
|
||
|
await http.post(`${song.id}/scrobble`, { timestamp: song.playStartTime })
|
||
|
},
|
||
|
|
||
|
async update (songsToUpdate: Song[], data: any): Promise<Song[]> {
|
||
|
const { songs, artists, albums } = await http.put<SongUpdateResult>('songs', {
|
||
|
data,
|
||
|
songs: songsToUpdate.map(song => song.id)
|
||
|
})
|
||
|
|
||
|
// Add the artist and album into stores if they're new
|
||
|
artists.forEach(artist => !artistStore.byId(artist.id) && artistStore.add(artist))
|
||
|
albums.forEach(album => !albumStore.byId(album.id) && albumStore.add(album))
|
||
|
|
||
|
songs.forEach(song => {
|
||
|
let originalSong = this.byId(song.id)!
|
||
|
|
||
|
if (originalSong.album_id !== song.album_id) {
|
||
|
// album has been changed. Remove the song from its old album.
|
||
|
originalSong.album.songs = without(originalSong.album.songs, originalSong)
|
||
|
}
|
||
|
|
||
|
if (originalSong.artist_id !== song.artist_id) {
|
||
|
// artist has been changed. Remove the song from its old artist
|
||
|
originalSong.artist.songs = without(originalSong.artist.songs, originalSong)
|
||
|
}
|
||
|
|
||
|
originalSong = Object.assign(originalSong, song)
|
||
|
// re-setup the song
|
||
|
this.setupSong(originalSong)
|
||
|
})
|
||
|
|
||
|
artistStore.compact()
|
||
|
albumStore.compact()
|
||
|
|
||
|
alerts.success(`Updated ${pluralize(songs.length, 'song')}.`)
|
||
|
|
||
|
return songs
|
||
|
},
|
||
|
|
||
|
getSourceUrl: (song: Song): string => {
|
||
|
return isMobile.any && preferenceStore.transcodeOnMobile
|
||
|
? `${sharedStore.state.cdnUrl}play/${song.id}/1/128?api_token=${auth.getToken()}`
|
||
|
: `${sharedStore.state.cdnUrl}play/${song.id}?api_token=${auth.getToken()}`
|
||
|
},
|
||
|
|
||
|
getShareableUrl: (song: Song): string => {
|
||
|
const baseUrl = KOEL_ENV === 'app' ? ls.get<string>('koelHost') : window.BASE_URL
|
||
|
return `${baseUrl}#!/song/${song.id}`
|
||
|
},
|
||
|
|
||
|
get recentlyPlayed (): Song[] {
|
||
|
return this.state.recentlyPlayed
|
||
|
},
|
||
|
|
||
|
getMostPlayed (n = 10): Song[] {
|
||
|
const songs = take(orderBy(this.all, 'playCount', 'desc'), n)
|
||
|
|
||
|
// Remove those with playCount=0
|
||
|
remove(songs, song => !song.playCount)
|
||
|
|
||
|
return songs
|
||
|
},
|
||
|
|
||
|
getRecentlyAdded (n = 10): Song[] {
|
||
|
return take(orderBy(this.all, 'created_at', 'desc'), n)
|
||
|
},
|
||
|
|
||
|
generateDataToBroadcast: (song: Song): BroadcastSongData => ({
|
||
|
song: {
|
||
|
id: song.id,
|
||
|
title: song.title,
|
||
|
liked: song.liked,
|
||
|
playbackState: song.playbackState || 'Stopped',
|
||
|
album: {
|
||
|
name: song.album.name,
|
||
|
cover: song.album.cover
|
||
|
},
|
||
|
artist: {
|
||
|
name: song.artist.name
|
||
|
}
|
||
|
}
|
||
|
})
|
||
|
}
|