resources/assets/js/stores/favoriteStore.ts

Summary

Maintainability
A
3 hrs
Test Coverage
import { reactive } from 'vue'
import { differenceBy, unionBy } from 'lodash'
import { http } from '@/services'
import { arrayify } from '@/utils'
import { songStore } from '@/stores'

export const favoriteStore = {
  state: reactive({
    songs: [] as Song[]
  }),

  async toggleOne (song: Song) {
    // Don't wait for the HTTP response to update the status, just toggle right away.
    // This may cause a minor problem if the request fails somehow, but do we care?
    song.liked = !song.liked
    song.liked ? this.add(song) : this.remove(song)

    await http.post<Song>('interaction/like', { song: song.id })
  },

  add (songs: Song | Song[]) {
    this.state.songs = unionBy(this.state.songs, arrayify(songs), 'id')
  },

  remove (songs: Song | Song[]) {
    this.state.songs = differenceBy(this.state.songs, arrayify(songs), 'id')
  },

  async like (songs: Song[]) {
    // Don't wait for the HTTP response to update the status, just set them to Liked right away.
    // This may cause a minor problem if the request fails somehow, but do we care?
    songs.forEach(song => (song.liked = true))
    this.add(songs)

    await http.post('interaction/batch/like', { songs: songs.map(song => song.id) })
  },

  async unlike (songs: Song[]) {
    songs.forEach(song => (song.liked = false))
    this.remove(songs)

    await http.post('interaction/batch/unlike', { songs: songs.map(song => song.id) })
  },

  async fetch () {
    this.state.songs = songStore.syncWithVault(await http.get<Song[]>('songs/favorite'))
    return this.state.songs
  }
}