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
249 lines
6.9 KiB
TypeScript
249 lines
6.9 KiB
TypeScript
import { AlbumContextPressable } from '@app/components/ContextMenu'
|
|
import CoverArt from '@app/components/CoverArt'
|
|
import GradientBackground from '@app/components/GradientBackground'
|
|
import GradientScrollView from '@app/components/GradientScrollView'
|
|
import Header from '@app/components/Header'
|
|
import HeaderBar from '@app/components/HeaderBar'
|
|
import ListItem from '@app/components/ListItem'
|
|
import { Album, Song } from '@app/models/library'
|
|
import { useStore, useStoreDeep } from '@app/state/store'
|
|
import colors from '@app/styles/colors'
|
|
import dimensions from '@app/styles/dimensions'
|
|
import font from '@app/styles/font'
|
|
import { mapById } from '@app/util/state'
|
|
import { useLayout } from '@react-native-community/hooks'
|
|
import { useNavigation } from '@react-navigation/native'
|
|
import React, { useCallback, useEffect } from 'react'
|
|
import { ActivityIndicator, StyleSheet, Text, View } from 'react-native'
|
|
import { useAnimatedScrollHandler, useAnimatedStyle, useSharedValue } from 'react-native-reanimated'
|
|
|
|
const AlbumItem = React.memo<{
|
|
album: Album
|
|
height: number
|
|
width: number
|
|
}>(({ album, height, width }) => {
|
|
const navigation = useNavigation()
|
|
|
|
if (height <= 0 || width <= 0) {
|
|
return <></>
|
|
}
|
|
|
|
return (
|
|
<AlbumContextPressable
|
|
album={album}
|
|
onPress={() => navigation.navigate('album', { id: album.id, title: album.name })}
|
|
menuStyle={[styles.albumItem, { width }]}
|
|
triggerOuterWrapperStyle={{ width }}>
|
|
<CoverArt type="cover" coverArt={album.coverArt} style={{ height, width }} resizeMode={'cover'} />
|
|
<Text style={styles.albumTitle}>{album.name}</Text>
|
|
<Text style={styles.albumYear}> {album.year ? album.year : ''}</Text>
|
|
</AlbumContextPressable>
|
|
)
|
|
})
|
|
|
|
const TopSongs = React.memo<{
|
|
songs: Song[]
|
|
name: string
|
|
artistId: string
|
|
}>(({ songs, name, artistId }) => {
|
|
const setQueue = useStore(store => store.setQueue)
|
|
|
|
return (
|
|
<>
|
|
<Header>Top Songs</Header>
|
|
{songs.slice(0, 5).map((s, i) => (
|
|
<ListItem
|
|
key={i}
|
|
item={s}
|
|
contextId={artistId}
|
|
queueId={i}
|
|
showArt={true}
|
|
subtitle={s.album}
|
|
onPress={() => setQueue(songs, name, 'artist', artistId, i)}
|
|
/>
|
|
))}
|
|
</>
|
|
)
|
|
})
|
|
|
|
const ArtistAlbums = React.memo<{
|
|
albums: Album[]
|
|
}>(({ albums }) => {
|
|
const albumsLayout = useLayout()
|
|
|
|
const sortedAlbums = [...albums]
|
|
.sort((a, b) => a.name.localeCompare(b.name))
|
|
.sort((a, b) => (b.year || 0) - (a.year || 0))
|
|
|
|
const albumSize = albumsLayout.width / 2 - styles.contentContainer.paddingHorizontal / 2
|
|
|
|
return (
|
|
<>
|
|
<Header>Albums</Header>
|
|
<View style={styles.albums} onLayout={albumsLayout.onLayout}>
|
|
{sortedAlbums.map(a => (
|
|
<AlbumItem key={a.id} album={a} height={albumSize} width={albumSize} />
|
|
))}
|
|
</View>
|
|
</>
|
|
)
|
|
})
|
|
|
|
const ArtistViewFallback = React.memo(() => (
|
|
<GradientBackground style={styles.fallback}>
|
|
<ActivityIndicator size="large" color={colors.accent} />
|
|
</GradientBackground>
|
|
))
|
|
|
|
const ArtistView = React.memo<{ id: string; title: string }>(({ id, title }) => {
|
|
const artist = useStoreDeep(useCallback(store => store.library.artists[id], [id]))
|
|
const topSongIds = useStoreDeep(useCallback(store => store.library.artistNameTopSongs[artist?.name], [artist?.name]))
|
|
const topSongs = useStoreDeep(
|
|
useCallback(store => (topSongIds ? mapById(store.library.songs, topSongIds) : undefined), [topSongIds]),
|
|
)
|
|
const albumIds = useStoreDeep(useCallback(store => store.library.artistAlbums[id], [id]))
|
|
const albums = useStoreDeep(
|
|
useCallback(store => (albumIds ? mapById(store.library.albums, albumIds) : undefined), [albumIds]),
|
|
)
|
|
|
|
const fetchArtist = useStore(store => store.fetchArtist)
|
|
const fetchTopSongs = useStore(store => store.fetchArtistTopSongs)
|
|
|
|
const coverLayout = useLayout()
|
|
const headerOpacity = useSharedValue(0)
|
|
|
|
const onScroll = useAnimatedScrollHandler({
|
|
onScroll: event => {
|
|
headerOpacity.value = Math.max(0, event.contentOffset.y - 70) / (artistCoverHeight - (70 + dimensions.header))
|
|
},
|
|
})
|
|
|
|
const animatedOpacity = useAnimatedStyle(() => {
|
|
return {
|
|
opacity: headerOpacity.value,
|
|
}
|
|
})
|
|
|
|
useEffect(() => {
|
|
if (!artist || !albumIds) {
|
|
fetchArtist(id)
|
|
}
|
|
}, [artist, albumIds, fetchArtist, id])
|
|
|
|
useEffect(() => {
|
|
if (artist && !topSongIds) {
|
|
fetchTopSongs(artist.name)
|
|
}
|
|
}, [artist, fetchTopSongs, topSongIds])
|
|
|
|
if (!artist) {
|
|
return <ArtistViewFallback />
|
|
}
|
|
|
|
return (
|
|
<View style={styles.container}>
|
|
<HeaderBar title={title} headerStyle={[styles.header, animatedOpacity]} />
|
|
<GradientScrollView
|
|
onLayout={coverLayout.onLayout}
|
|
offset={artistCoverHeight}
|
|
style={styles.scroll}
|
|
contentContainerStyle={styles.scrollContent}
|
|
onScroll={onScroll}>
|
|
<CoverArt type="artist" size="original" artistId={artist.id} style={styles.artistCover} resizeMode={'cover'} />
|
|
<View style={styles.titleContainer}>
|
|
<Text style={styles.title}>{artist.name}</Text>
|
|
</View>
|
|
<View style={styles.contentContainer}>
|
|
{topSongs && albums ? (
|
|
topSongs.length > 0 ? (
|
|
<>
|
|
<TopSongs songs={topSongs} name={artist.name} artistId={artist.id} />
|
|
<ArtistAlbums albums={albums} />
|
|
</>
|
|
) : (
|
|
<ArtistAlbums albums={albums} />
|
|
)
|
|
) : (
|
|
<ActivityIndicator size="large" color={colors.accent} style={styles.loading} />
|
|
)}
|
|
</View>
|
|
</GradientScrollView>
|
|
</View>
|
|
)
|
|
})
|
|
|
|
const artistCoverHeight = 350
|
|
|
|
const styles = StyleSheet.create({
|
|
container: {
|
|
flex: 1,
|
|
},
|
|
header: {
|
|
position: 'absolute',
|
|
zIndex: 1,
|
|
},
|
|
scroll: {
|
|
flex: 1,
|
|
},
|
|
fallback: {
|
|
alignItems: 'center',
|
|
paddingTop: 100,
|
|
},
|
|
scrollContent: {
|
|
alignItems: 'center',
|
|
},
|
|
contentContainer: {
|
|
minHeight: artistCoverHeight * 2,
|
|
width: '100%',
|
|
paddingHorizontal: 20,
|
|
},
|
|
titleContainer: {
|
|
width: '100%',
|
|
height: artistCoverHeight,
|
|
justifyContent: 'flex-end',
|
|
},
|
|
title: {
|
|
fontFamily: font.bold,
|
|
fontSize: 44,
|
|
color: colors.text.primary,
|
|
textAlign: 'center',
|
|
textShadowColor: 'black',
|
|
textShadowOffset: { width: 0, height: 0 },
|
|
textShadowRadius: 16,
|
|
paddingHorizontal: 10,
|
|
marginBottom: 10,
|
|
},
|
|
artistCover: {
|
|
position: 'absolute',
|
|
height: artistCoverHeight,
|
|
width: '100%',
|
|
},
|
|
albums: {
|
|
width: '100%',
|
|
flexDirection: 'row',
|
|
flexWrap: 'wrap',
|
|
alignItems: 'flex-start',
|
|
justifyContent: 'space-between',
|
|
},
|
|
albumItem: {
|
|
marginBottom: 20,
|
|
},
|
|
albumTitle: {
|
|
fontFamily: font.semiBold,
|
|
fontSize: 14,
|
|
color: colors.text.primary,
|
|
marginTop: 4,
|
|
textAlign: 'center',
|
|
},
|
|
albumYear: {
|
|
color: colors.text.secondary,
|
|
fontFamily: font.regular,
|
|
textAlign: 'center',
|
|
},
|
|
loading: {
|
|
marginTop: 30,
|
|
},
|
|
})
|
|
|
|
export default ArtistView
|