mirror of
https://github.com/austinried/subtracks.git
synced 2025-12-27 09:09:29 +01:00
* initial react-query experiments * use queries for item screens send the data we do have over routing to prepopulate (album/playlist) use number for starred because sending Date freaks out react-navigation * add in equiv. song cover art fix * reorg, switch artistview over start mapping song cover art when any are available * refactor useStar to queries fix caching for starred items and album cover art * add hook to reset queries on server change * refactor search to use query * fix song cover art setting * use query for artistInfo * remove last bits of library state * cleanup * use query key factory already fixed one wrong key... * require coverart size * let's try no promise queues on these for now * image cache uses query * perf fix for playlist parsing also use placeholder data so we don't have to deal with staleness * drill that disabled also list controls doesn't need its own songs hook/copy * switch to react-native-blob-util for downloads slightly slower but allows us to use DownloadManager, which backgrounds downloads so they are no longer corrupted when the app suspends * add a fake "top songs" based on artist search then sorted by play count/ratings artistview should load now even if topSongs fails * try not to swap between topSongs/search on refetch set queueContext by song list so the index isn't off if the list changes * add content type validation for file fetching also try to speed up existing file return by limiting fs ops * if the HEAD fails, don't queue the download * clean up params * reimpl clear image cache * precompute contextId prevents wrong "is playing" when any mismatch between queue and list * clear images from all servers use external files dir instead of cache * fix pressable disabled flicker don't retry topsongs on failure try to optimize setqueue and fixcoverart a bit * wait for queries during clear * break out fetchExistingFile from fetchFile allows to tell if file is coming from disk or not only show placeholder/loading spinner if actually fetching image * forgot these wouldn't do anything with objects * remove query cache when switching servers * add content-disposition extension gathering add support for progress hook (needs native support still) * added custom RNBU pkg with progress changes * fully unmount tabs when server changes prevents unwanted requests, gives fresh start on switch fix fixCoverArt not re-rendering in certain cases on search * use serverId from fetch deps * fix lint * update licenses * just use the whole lodash package * make using cache buster optional
230 lines
6.3 KiB
TypeScript
230 lines
6.3 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 { useQueryArtist, useQueryArtistTopSongs } from '@app/hooks/query'
|
|
import { useSetQueue } from '@app/hooks/trackplayer'
|
|
import { Album, Song } from '@app/models/library'
|
|
import colors from '@app/styles/colors'
|
|
import dimensions from '@app/styles/dimensions'
|
|
import font from '@app/styles/font'
|
|
import { useLayout } from '@react-native-community/hooks'
|
|
import { useNavigation } from '@react-navigation/native'
|
|
import equal from 'fast-deep-equal/es6/react'
|
|
import React 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, album })}
|
|
menuStyle={[styles.albumItem, { width }]}
|
|
triggerOuterWrapperStyle={{ width }}>
|
|
<CoverArt type="cover" coverArt={album.coverArt} style={{ height, width }} resizeMode="cover" size="thumbnail" />
|
|
<Text style={styles.albumTitle}>{album.name}</Text>
|
|
<Text style={styles.albumYear}> {album.year ? album.year : ''}</Text>
|
|
</AlbumContextPressable>
|
|
)
|
|
}, equal)
|
|
|
|
const TopSongs = React.memo<{
|
|
songs: Song[]
|
|
name: string
|
|
}>(({ songs, name }) => {
|
|
const { setQueue, isReady, contextId } = useSetQueue('artist', songs)
|
|
|
|
return (
|
|
<>
|
|
<Header>Top Songs</Header>
|
|
{songs.slice(0, 5).map((s, i) => (
|
|
<ListItem
|
|
key={i}
|
|
item={s}
|
|
contextId={contextId}
|
|
queueId={i}
|
|
showArt={true}
|
|
subtitle={s.album}
|
|
onPress={() => setQueue({ title: name, playTrack: i })}
|
|
disabled={!isReady}
|
|
/>
|
|
))}
|
|
</>
|
|
)
|
|
}, equal)
|
|
|
|
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>
|
|
</>
|
|
)
|
|
}, equal)
|
|
|
|
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 { data: artistData } = useQueryArtist(id)
|
|
const { data: topSongs, isError } = useQueryArtistTopSongs(artistData?.artist?.name)
|
|
|
|
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,
|
|
}
|
|
})
|
|
|
|
if (!artistData) {
|
|
return <ArtistViewFallback />
|
|
}
|
|
|
|
const { artist, albums } = artistData
|
|
|
|
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 || isError) && artist ? (
|
|
topSongs && topSongs.length > 0 ? (
|
|
<>
|
|
<TopSongs songs={topSongs} name={artist.name} />
|
|
<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
|