mirror of
https://github.com/austinried/subtracks.git
synced 2025-12-27 00:59:28 +01:00
* start of music store refactor moving stuff into a state cache better separate it from view logic * added paginated list/album list * reworked fetchAlbumList to remove ui state refactored home screen to use new method i broke playing songs somehow, JS thread goes into a loop * don't reset parts manually, do it all at once * fixed perf issue related to too many rerenders rerenders were caused by strict equality check on object/array picks switched artistInfo to new store updated zustand and fixed deprecation warnings * update typescript and use workspace tsc version for vscode * remove old artistInfo * switched to new playlist w/songs removed more unused stuff * remove unused + (slightly) rework search * refactor star * use only original/large imges for covers/artist fix view artist from context menu add loading indicators to song list and artist views (show info we have right away) * set starred/unstar assuming it works and correct state on error * reorg, remove old music slice files * added back fix for song cover art * sort artists by localCompare name * update licenses * fix now playing background grey bar * update react-native-gesture-handler for node-fetch security alert * fix another gradient height grey bar issue * update licenses again * remove thumbnail cache * rename to remove "Library" from methods * Revert "remove thumbnail cache" This reverts commit e0db4931f11bbf4cd8e73102d06505c6ae85f4a6. * use ids for lists, pull state later * Revert "use only original/large imges for covers/artist" This reverts commit c9aea9065ce6ebe3c8b09c10dd74d4de153d76fd. * deep equal ListItem props for now this needs a bigger refactor * use immer as middleware * refactor api client to use string method hoping to use this for requestKey/deduping next * use thumbnails in list items * Revert "refactor api client to use string method" This reverts commit 234326135b7af96cb91b941e7ca515f45c632556. * rename/cleanup * store servers by id * get rid of settings selectors * renames for clarity remove unused estimateContentLength setting * remove trackplayer selectors * fix migration for library filter settings * fixed shuffle order reporting wrong track/queue * removed the other selectors * don't actually need es6/react for our state * fix slow artist sort on star localeCompare is too slow for large lists
252 lines
6.6 KiB
TypeScript
252 lines
6.6 KiB
TypeScript
import CoverArt from '@app/components/CoverArt'
|
|
import GradientBackground from '@app/components/GradientBackground'
|
|
import HeaderBar from '@app/components/HeaderBar'
|
|
import ImageGradientFlatList from '@app/components/ImageGradientFlatList'
|
|
import ListItem from '@app/components/ListItem'
|
|
import ListPlayerControls from '@app/components/ListPlayerControls'
|
|
import NothingHere from '@app/components/NothingHere'
|
|
import { useCoverArtFile } from '@app/hooks/cache'
|
|
import { Song, Album, Playlist } from '@app/models/library'
|
|
import { useStore, useStoreDeep } from '@app/state/store'
|
|
import colors from '@app/styles/colors'
|
|
import font from '@app/styles/font'
|
|
import React, { useCallback, useEffect, useState } from 'react'
|
|
import { ActivityIndicator, StyleSheet, Text, View } from 'react-native'
|
|
|
|
type SongListType = 'album' | 'playlist'
|
|
|
|
const SongListDetailsFallback = React.memo(() => (
|
|
<GradientBackground style={styles.fallback}>
|
|
<ActivityIndicator size="large" color={colors.accent} />
|
|
</GradientBackground>
|
|
))
|
|
|
|
const SongRenderItem: React.FC<{
|
|
item: {
|
|
song: Song
|
|
contextId?: string
|
|
queueId?: number
|
|
subtitle?: string
|
|
onPress?: () => void
|
|
showArt?: boolean
|
|
}
|
|
}> = ({ item }) => (
|
|
<ListItem
|
|
item={item.song}
|
|
contextId={item.contextId}
|
|
queueId={item.queueId}
|
|
subtitle={item.subtitle}
|
|
onPress={item.onPress}
|
|
showArt={item.showArt}
|
|
style={styles.listItem}
|
|
/>
|
|
)
|
|
|
|
const SongListDetails = React.memo<{
|
|
title: string
|
|
type: SongListType
|
|
songList?: Album | Playlist
|
|
songs?: Song[]
|
|
subtitle?: string
|
|
}>(({ title, songList, songs, subtitle, type }) => {
|
|
const coverArtFile = useCoverArtFile(songList?.coverArt, 'thumbnail')
|
|
const [headerColor, setHeaderColor] = useState<string | undefined>(undefined)
|
|
const setQueue = useStore(store => store.setQueue)
|
|
|
|
if (!songList) {
|
|
return <SongListDetailsFallback />
|
|
}
|
|
|
|
const _songs = [...(songs || [])]
|
|
let typeName = ''
|
|
|
|
if (type === 'album') {
|
|
typeName = 'Album'
|
|
if (_songs.some(s => s.track === undefined)) {
|
|
_songs.sort((a, b) => a.title.localeCompare(b.title))
|
|
} else {
|
|
_songs.sort((a, b) => {
|
|
const aVal = (a.track as number) + (a.discNumber !== undefined ? a.discNumber * 10000 : 0)
|
|
const bVal = (b.track as number) + (b.discNumber !== undefined ? b.discNumber * 10000 : 0)
|
|
return aVal - bVal
|
|
})
|
|
}
|
|
} else {
|
|
typeName = 'Playlist'
|
|
}
|
|
|
|
return (
|
|
<View style={styles.container}>
|
|
<HeaderBar
|
|
headerStyle={{ backgroundColor: headerColor }}
|
|
title={title}
|
|
contextItem={songList.itemType === 'album' ? songList : undefined}
|
|
/>
|
|
<ImageGradientFlatList
|
|
data={_songs.map((s, i) => ({
|
|
song: s,
|
|
contextId: songList.id,
|
|
queueId: i,
|
|
subtitle: s.artist,
|
|
onPress: () => setQueue(_songs, songList.name, type, songList.id, i),
|
|
showArt: songList.itemType === 'playlist',
|
|
}))}
|
|
renderItem={SongRenderItem}
|
|
keyExtractor={(item, i) => i.toString()}
|
|
backgroundProps={{
|
|
imagePath: coverArtFile?.file?.path,
|
|
style: styles.container,
|
|
onGetColor: setHeaderColor,
|
|
}}
|
|
overScrollMode="never"
|
|
windowSize={7}
|
|
contentMarginTop={26}
|
|
ListEmptyComponent={
|
|
songs ? (
|
|
<NothingHere style={styles.nothing} />
|
|
) : (
|
|
<ActivityIndicator size="large" color={colors.accent} style={styles.listLoading} />
|
|
)
|
|
}
|
|
ListHeaderComponent={
|
|
<View style={styles.content}>
|
|
<CoverArt type="cover" size="original" coverArt={songList.coverArt} style={styles.cover} />
|
|
<Text style={styles.title}>{songList.name}</Text>
|
|
{subtitle ? <Text style={styles.subtitle}>{subtitle}</Text> : <></>}
|
|
<ListPlayerControls
|
|
style={styles.controls}
|
|
songs={_songs}
|
|
typeName={typeName}
|
|
queueName={songList.name}
|
|
queueContextId={songList.id}
|
|
queueContextType={type}
|
|
/>
|
|
</View>
|
|
}
|
|
/>
|
|
</View>
|
|
)
|
|
})
|
|
|
|
const PlaylistView = React.memo<{
|
|
id: string
|
|
title: string
|
|
}>(({ id, title }) => {
|
|
const playlist = useStoreDeep(useCallback(store => store.library.playlists[id], [id]))
|
|
const songs = useStoreDeep(
|
|
useCallback(
|
|
store => {
|
|
const ids = store.library.playlistSongs[id]
|
|
return ids ? ids.map(i => store.library.songs[i]) : undefined
|
|
},
|
|
[id],
|
|
),
|
|
)
|
|
|
|
const fetchPlaylist = useStore(store => store.fetchPlaylist)
|
|
|
|
useEffect(() => {
|
|
if (!playlist || !songs) {
|
|
fetchPlaylist(id)
|
|
}
|
|
}, [playlist, fetchPlaylist, id, songs])
|
|
|
|
return (
|
|
<SongListDetails title={title} songList={playlist} songs={songs} subtitle={playlist?.comment} type="playlist" />
|
|
)
|
|
})
|
|
|
|
const AlbumView = React.memo<{
|
|
id: string
|
|
title: string
|
|
}>(({ id, title }) => {
|
|
const album = useStoreDeep(useCallback(store => store.library.albums[id], [id]))
|
|
const songs = useStoreDeep(
|
|
useCallback(
|
|
store => {
|
|
const ids = store.library.albumSongs[id]
|
|
return ids ? ids.map(i => store.library.songs[i]) : undefined
|
|
},
|
|
[id],
|
|
),
|
|
)
|
|
|
|
const fetchAlbum = useStore(store => store.fetchAlbum)
|
|
|
|
useEffect(() => {
|
|
if (!album || !songs) {
|
|
fetchAlbum(id)
|
|
}
|
|
}, [album, fetchAlbum, id, songs])
|
|
|
|
return (
|
|
<SongListDetails
|
|
title={title}
|
|
songList={album}
|
|
songs={songs}
|
|
subtitle={(album?.artist || '') + (album?.year ? ' • ' + album?.year : '')}
|
|
type="album"
|
|
/>
|
|
)
|
|
})
|
|
|
|
const SongListView = React.memo<{
|
|
id: string
|
|
title: string
|
|
type: SongListType
|
|
}>(({ id, title, type }) => {
|
|
return type === 'album' ? <AlbumView id={id} title={title} /> : <PlaylistView id={id} title={title} />
|
|
})
|
|
|
|
const styles = StyleSheet.create({
|
|
container: {
|
|
flex: 1,
|
|
},
|
|
content: {
|
|
alignItems: 'center',
|
|
paddingTop: 10,
|
|
paddingHorizontal: 20,
|
|
},
|
|
controls: {
|
|
marginTop: 20,
|
|
},
|
|
title: {
|
|
fontSize: 24,
|
|
fontFamily: font.bold,
|
|
color: colors.text.primary,
|
|
marginTop: 20,
|
|
textAlign: 'center',
|
|
},
|
|
subtitle: {
|
|
fontFamily: font.regular,
|
|
color: colors.text.secondary,
|
|
fontSize: 14,
|
|
marginTop: 4,
|
|
textAlign: 'center',
|
|
},
|
|
cover: {
|
|
height: 240,
|
|
width: 240,
|
|
},
|
|
songs: {
|
|
marginTop: 26,
|
|
marginBottom: 30,
|
|
width: '100%',
|
|
},
|
|
fallback: {
|
|
alignItems: 'center',
|
|
paddingTop: 100,
|
|
},
|
|
listItem: {
|
|
paddingHorizontal: 20,
|
|
},
|
|
nothing: {
|
|
width: '100%',
|
|
},
|
|
listLoading: {
|
|
marginTop: 10,
|
|
},
|
|
})
|
|
|
|
export default SongListView
|