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
203 lines
5.1 KiB
TypeScript
203 lines
5.1 KiB
TypeScript
import { AlbumContextPressable } from '@app/components/ContextMenu'
|
|
import CoverArt from '@app/components/CoverArt'
|
|
import GradientScrollView from '@app/components/GradientScrollView'
|
|
import Header from '@app/components/Header'
|
|
import NothingHere from '@app/components/NothingHere'
|
|
import { useActiveServerRefresh } from '@app/hooks/settings'
|
|
import { useStore, useStoreDeep } from '@app/state/store'
|
|
import colors from '@app/styles/colors'
|
|
import font from '@app/styles/font'
|
|
import { GetAlbumList2TypeBase, GetAlbumListType } from '@app/subsonic/params'
|
|
import { useNavigation } from '@react-navigation/native'
|
|
import equal from 'fast-deep-equal/es6/react'
|
|
import produce from 'immer'
|
|
import React, { useCallback, useState } from 'react'
|
|
import { RefreshControl, ScrollView, StatusBar, StyleSheet, Text, View } from 'react-native'
|
|
import create, { StateSelector } from 'zustand'
|
|
|
|
const titles: { [key in GetAlbumListType]?: string } = {
|
|
recent: 'Recently Played',
|
|
random: 'Random Albums',
|
|
frequent: 'Frequently Played',
|
|
starred: 'Starred Albums',
|
|
}
|
|
|
|
const AlbumItem = React.memo<{
|
|
id: string
|
|
}>(({ id }) => {
|
|
const navigation = useNavigation()
|
|
const album = useStoreDeep(useCallback(store => store.library.albums[id], [id]))
|
|
|
|
if (!album) {
|
|
return <></>
|
|
}
|
|
|
|
return (
|
|
<AlbumContextPressable
|
|
album={album}
|
|
triggerWrapperStyle={styles.item}
|
|
onPress={() => navigation.navigate('album', { id: album.id, title: album.name })}>
|
|
<CoverArt
|
|
type="cover"
|
|
coverArt={album.coverArt}
|
|
style={{ height: styles.item.width, width: styles.item.width }}
|
|
resizeMode={'cover'}
|
|
/>
|
|
<Text style={styles.title} numberOfLines={1}>
|
|
{album.name}
|
|
</Text>
|
|
<Text style={styles.subtitle} numberOfLines={1}>
|
|
{album.artist}
|
|
</Text>
|
|
</AlbumContextPressable>
|
|
)
|
|
})
|
|
|
|
const Category = React.memo<{
|
|
type: string
|
|
}>(({ type }) => {
|
|
const list = useHomeStoreDeep(useCallback(store => store.lists[type] || [], [type]))
|
|
|
|
const Albums = () => (
|
|
<ScrollView
|
|
horizontal={true}
|
|
showsHorizontalScrollIndicator={false}
|
|
overScrollMode={'never'}
|
|
style={styles.artScroll}
|
|
contentContainerStyle={styles.artScrollContent}>
|
|
{list.map(id => (
|
|
<AlbumItem key={id} id={id} />
|
|
))}
|
|
</ScrollView>
|
|
)
|
|
|
|
const Nothing = () => (
|
|
<View style={styles.nothingHereContent}>
|
|
<NothingHere height={160} width={160} />
|
|
</View>
|
|
)
|
|
|
|
return (
|
|
<View style={styles.category}>
|
|
<Header style={styles.header}>{titles[type as GetAlbumListType] || ''}</Header>
|
|
{list.length > 0 ? <Albums /> : <Nothing />}
|
|
</View>
|
|
)
|
|
})
|
|
|
|
interface HomeState {
|
|
lists: { [type: string]: string[] }
|
|
setList: (type: string, list: string[]) => void
|
|
}
|
|
|
|
const useHomeStore = create<HomeState>(set => ({
|
|
lists: {},
|
|
|
|
setList: (type, list) => {
|
|
set(
|
|
produce<HomeState>(state => {
|
|
state.lists[type] = list
|
|
}),
|
|
)
|
|
},
|
|
}))
|
|
|
|
function useHomeStoreDeep<U>(stateSelector: StateSelector<HomeState, U>) {
|
|
return useHomeStore(stateSelector, equal)
|
|
}
|
|
|
|
const Home = () => {
|
|
const [refreshing, setRefreshing] = useState(false)
|
|
const types = useStoreDeep(store => store.settings.screens.home.listTypes)
|
|
const fetchAlbumList = useStore(store => store.fetchAlbumList)
|
|
const setList = useHomeStore(store => store.setList)
|
|
|
|
const refresh = useCallback(async () => {
|
|
setRefreshing(true)
|
|
|
|
await Promise.all(
|
|
types.map(async type => {
|
|
const ids = await fetchAlbumList({ type: type as GetAlbumList2TypeBase, size: 20, offset: 0 })
|
|
setList(type, ids)
|
|
}),
|
|
)
|
|
|
|
setRefreshing(false)
|
|
}, [fetchAlbumList, setList, types])
|
|
|
|
useActiveServerRefresh(
|
|
useCallback(() => {
|
|
types.forEach(type => setList(type, []))
|
|
refresh()
|
|
}, [refresh, setList, types]),
|
|
)
|
|
|
|
return (
|
|
<GradientScrollView
|
|
style={styles.scroll}
|
|
contentContainerStyle={styles.scrollContentContainer}
|
|
refreshControl={
|
|
<RefreshControl
|
|
refreshing={refreshing}
|
|
onRefresh={refresh}
|
|
colors={[colors.accent, colors.accentLow]}
|
|
progressViewOffset={StatusBar.currentHeight}
|
|
/>
|
|
}>
|
|
<View style={styles.content}>
|
|
{types.map(type => (
|
|
<Category key={type} type={type} />
|
|
))}
|
|
</View>
|
|
</GradientScrollView>
|
|
)
|
|
}
|
|
|
|
const styles = StyleSheet.create({
|
|
scroll: {
|
|
flex: 1,
|
|
},
|
|
scrollContentContainer: {
|
|
paddingTop: StatusBar.currentHeight,
|
|
},
|
|
content: {
|
|
paddingBottom: 20,
|
|
},
|
|
header: {
|
|
paddingHorizontal: 20,
|
|
},
|
|
category: {
|
|
// marginTop: 12,
|
|
},
|
|
nothingHereContent: {
|
|
width: '100%',
|
|
height: 190,
|
|
justifyContent: 'center',
|
|
alignItems: 'center',
|
|
},
|
|
artScroll: {
|
|
height: 190,
|
|
},
|
|
artScrollContent: {
|
|
paddingLeft: 20,
|
|
},
|
|
item: {
|
|
flex: 1,
|
|
marginRight: 10,
|
|
width: 150,
|
|
},
|
|
title: {
|
|
fontFamily: font.semiBold,
|
|
fontSize: 13,
|
|
color: colors.text.primary,
|
|
marginTop: 4,
|
|
},
|
|
subtitle: {
|
|
fontFamily: font.regular,
|
|
fontSize: 12,
|
|
color: colors.text.secondary,
|
|
},
|
|
})
|
|
|
|
export default Home
|