import { reactive, UnwrapNestedRefs } from 'vue' import { differenceBy, unionBy } from 'lodash' import { cache, http } from '@/services' import { arrayify, logger } from '@/utils' const UNKNOWN_ARTIST_ID = 1 const VARIOUS_ARTISTS_ID = 2 export const artistStore = { vault: new Map>(), state: reactive({ artists: [] as Artist[] }), byId (id: number) { return this.vault.get(id) }, removeByIds (ids: number[]) { this.state.artists = differenceBy(this.state.artists, ids.map(id => this.byId(id)), 'id') ids.forEach(id => this.vault.delete(id)) }, isVarious: (artist: Artist | number) => (typeof artist === 'number') ? artist === VARIOUS_ARTISTS_ID : artist.id === VARIOUS_ARTISTS_ID, isUnknown: (artist: Artist | number) => (typeof artist === 'number') ? artist === UNKNOWN_ARTIST_ID : artist.id === UNKNOWN_ARTIST_ID, isStandard (artist: Artist | number) { return !this.isVarious(artist) && !this.isUnknown(artist) }, async uploadImage (artist: Artist, image: string) { artist.image = (await http.put<{ image_url: string }>(`artists/${artist.id}/image`, { image })).image_url // sync to vault this.byId(artist.id)!.image = artist.image return artist.image }, syncWithVault (artists: Artist | Artist[]) { return arrayify(artists).map(artist => { let local = this.vault.get(artist.id) local = local ? Object.assign(local, artist) : reactive(artist) this.vault.set(artist.id, local) return local }) }, async resolve (id: number) { let artist = this.byId(id) if (!artist) { try { artist = this.syncWithVault( await cache.remember(['artist', id], async () => await http.get(`artists/${id}`)) )[0] } catch (error: unknown) { logger.error(error) } } return artist }, async paginate (page: number) { const resource = await http.get>(`artists?page=${page}`) this.state.artists = unionBy(this.state.artists, this.syncWithVault(resource.data), 'id') return resource.links.next ? ++resource.meta.current_page : null } }