mirror of
https://github.com/austinried/subtracks.git
synced 2025-12-27 17:19:27 +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
370 lines
9.6 KiB
TypeScript
370 lines
9.6 KiB
TypeScript
import PressableOpacity from '@app/components/PressableOpacity'
|
|
import { useStar } from '@app/hooks/query'
|
|
import { StarrableItemType, Song, Artist, Album } from '@app/models/library'
|
|
import colors from '@app/styles/colors'
|
|
import font from '@app/styles/font'
|
|
import { NavigationProp, useNavigation } from '@react-navigation/native'
|
|
import { ReactComponentLike } from 'prop-types'
|
|
import React from 'react'
|
|
import { ScrollView, StyleProp, StyleSheet, Text, View, ViewStyle } from 'react-native'
|
|
import { Menu, MenuOption, MenuOptions, MenuTrigger, renderers } from 'react-native-popup-menu'
|
|
import IconFA from 'react-native-vector-icons/FontAwesome'
|
|
import IconFA5 from 'react-native-vector-icons/FontAwesome5'
|
|
import CoverArt from './CoverArt'
|
|
import { Star } from './Star'
|
|
|
|
const { SlideInMenu } = renderers
|
|
|
|
type ContextMenuProps = {
|
|
menuStyle?: StyleProp<ViewStyle>
|
|
triggerWrapperStyle?: StyleProp<ViewStyle>
|
|
triggerOuterWrapperStyle?: StyleProp<ViewStyle>
|
|
triggerTouchableStyle?: StyleProp<ViewStyle>
|
|
onPress?: () => any
|
|
triggerOnLongPress?: boolean
|
|
disabled?: boolean
|
|
}
|
|
|
|
type InternalContextMenuProps = ContextMenuProps & {
|
|
menuHeader: React.ReactNode
|
|
menuOptions: React.ReactNode
|
|
}
|
|
|
|
const ContextMenu: React.FC<InternalContextMenuProps> = ({
|
|
menuStyle,
|
|
triggerWrapperStyle,
|
|
triggerOuterWrapperStyle,
|
|
triggerTouchableStyle,
|
|
onPress,
|
|
menuHeader,
|
|
menuOptions,
|
|
children,
|
|
triggerOnLongPress,
|
|
disabled,
|
|
}) => {
|
|
menuStyle = menuStyle || { flex: 1 }
|
|
triggerWrapperStyle = triggerWrapperStyle || { flex: 1 }
|
|
triggerOuterWrapperStyle = triggerOuterWrapperStyle || { flex: 1 }
|
|
triggerTouchableStyle = triggerTouchableStyle || { flex: 1 }
|
|
return (
|
|
<Menu renderer={SlideInMenu} style={menuStyle}>
|
|
<MenuTrigger
|
|
disabled={disabled}
|
|
triggerOnLongPress={triggerOnLongPress === undefined ? true : triggerOnLongPress}
|
|
customStyles={{
|
|
triggerOuterWrapper: triggerOuterWrapperStyle,
|
|
triggerWrapper: triggerWrapperStyle,
|
|
triggerTouchable: { style: triggerTouchableStyle, disabled },
|
|
TriggerTouchableComponent: PressableOpacity,
|
|
}}
|
|
onAlternativeAction={onPress}>
|
|
{children}
|
|
</MenuTrigger>
|
|
<MenuOptions
|
|
customStyles={styles}
|
|
renderOptionsContainer={(options: any) => (
|
|
<ScrollView>
|
|
{menuHeader}
|
|
{options}
|
|
</ScrollView>
|
|
)}>
|
|
{menuOptions}
|
|
</MenuOptions>
|
|
</Menu>
|
|
)
|
|
}
|
|
|
|
type ContextMenuOptionProps = {
|
|
onSelect?: () => any
|
|
}
|
|
|
|
const ContextMenuOption: React.FC<ContextMenuOptionProps> = ({ onSelect, children }) => (
|
|
<MenuOption style={styles.option} onSelect={onSelect}>
|
|
{children}
|
|
</MenuOption>
|
|
)
|
|
|
|
type ContextMenuIconTextOptionProps = ContextMenuOptionProps & {
|
|
IconComponent?: ReactComponentLike
|
|
IconComponentRaw?: React.ReactNode
|
|
name?: string
|
|
size?: number
|
|
color?: string
|
|
text: string
|
|
}
|
|
|
|
const ContextMenuIconTextOption = React.memo<ContextMenuIconTextOptionProps>(
|
|
({ onSelect, IconComponent, IconComponentRaw, name, color, size, text }) => {
|
|
let Icon: React.ReactNode
|
|
if (IconComponentRaw) {
|
|
Icon = IconComponentRaw
|
|
} else if (IconComponent) {
|
|
Icon = <IconComponent name={name} size={size} color={color || colors.text.primary} />
|
|
} else {
|
|
Icon = <></>
|
|
}
|
|
return (
|
|
<ContextMenuOption onSelect={onSelect}>
|
|
<View style={styles.icon}>{Icon}</View>
|
|
<Text style={styles.optionText}>{text}</Text>
|
|
</ContextMenuOption>
|
|
)
|
|
},
|
|
)
|
|
|
|
const MenuHeader = React.memo<{
|
|
coverArt?: string
|
|
artistId?: string
|
|
title: string
|
|
subtitle?: string
|
|
}>(({ coverArt, artistId, title, subtitle }) => (
|
|
<View style={styles.menuHeader}>
|
|
{artistId ? (
|
|
<CoverArt
|
|
type="artist"
|
|
artistId={artistId}
|
|
style={styles.coverArt}
|
|
resizeMode="cover"
|
|
round={true}
|
|
size="thumbnail"
|
|
fadeDuration={0}
|
|
/>
|
|
) : (
|
|
<CoverArt
|
|
type="cover"
|
|
coverArt={coverArt}
|
|
style={styles.coverArt}
|
|
resizeMode="cover"
|
|
size="thumbnail"
|
|
fadeDuration={0}
|
|
/>
|
|
)}
|
|
<View style={styles.menuHeaderText}>
|
|
<Text numberOfLines={1} style={styles.menuTitle}>
|
|
{title}
|
|
</Text>
|
|
{subtitle ? (
|
|
<Text numberOfLines={1} style={styles.menuSubtitle}>
|
|
{subtitle}
|
|
</Text>
|
|
) : (
|
|
<></>
|
|
)}
|
|
</View>
|
|
</View>
|
|
))
|
|
|
|
const OptionStar = React.memo<{
|
|
id: string
|
|
type: StarrableItemType
|
|
additionalText?: string
|
|
}>(({ id, type, additionalText: text }) => {
|
|
const { query, toggle } = useStar(id, type)
|
|
|
|
return (
|
|
<ContextMenuIconTextOption
|
|
IconComponentRaw={<Star starred={!!query.data} size={26} />}
|
|
text={(query.data ? 'Unstar' : 'Star') + (text ? ` ${text}` : '')}
|
|
onSelect={() => toggle.mutate()}
|
|
/>
|
|
)
|
|
})
|
|
|
|
const OptionViewArtist = React.memo<{
|
|
navigation: NavigationProp<any>
|
|
artist?: string
|
|
artistId?: string
|
|
}>(({ navigation, artist, artistId }) => {
|
|
if (!artist || !artistId) {
|
|
return <></>
|
|
}
|
|
|
|
return (
|
|
<ContextMenuIconTextOption
|
|
IconComponent={IconFA}
|
|
name="microphone"
|
|
size={26}
|
|
text="View Artist"
|
|
onSelect={() => navigation.navigate('artist', { id: artistId, title: artist })}
|
|
/>
|
|
)
|
|
})
|
|
|
|
const OptionViewAlbum = React.memo<{
|
|
navigation: NavigationProp<any>
|
|
album?: string
|
|
albumId?: string
|
|
}>(({ navigation, album, albumId }) => {
|
|
if (!album || !albumId) {
|
|
return <></>
|
|
}
|
|
|
|
return (
|
|
<ContextMenuIconTextOption
|
|
IconComponent={IconFA5}
|
|
name="compact-disc"
|
|
size={26}
|
|
text="View Album"
|
|
onSelect={() => navigation.navigate('album', { id: albumId, title: album })}
|
|
/>
|
|
)
|
|
})
|
|
|
|
// const OptionDownload = React.memo<{
|
|
// itemType: string
|
|
// }>(({ itemType }) => (
|
|
// <ContextMenuIconTextOption IconComponent={IconMat} name="file-download" size={26} text={`Download ${itemType}`} />
|
|
// ))
|
|
|
|
export type AlbumContextPressableProps = ContextMenuProps & {
|
|
album: Album
|
|
}
|
|
|
|
export const AlbumContextPressable: React.FC<AlbumContextPressableProps> = props => {
|
|
const navigation = useNavigation()
|
|
const { album, children } = props
|
|
|
|
return (
|
|
<ContextMenu
|
|
{...props}
|
|
menuHeader={<MenuHeader title={album.name} subtitle={album.artist} coverArt={album.coverArt} />}
|
|
menuOptions={
|
|
<>
|
|
<OptionStar id={album.id} type={album.itemType} />
|
|
<OptionViewArtist artist={album.artist} artistId={album.artistId} navigation={navigation} />
|
|
{/* <OptionDownload itemType={album.itemType} /> */}
|
|
</>
|
|
}>
|
|
{children}
|
|
</ContextMenu>
|
|
)
|
|
}
|
|
|
|
export type SongContextPressableProps = ContextMenuProps & {
|
|
song: Song
|
|
}
|
|
|
|
export const SongContextPressable: React.FC<SongContextPressableProps> = props => {
|
|
const navigation = useNavigation()
|
|
const { song, children } = props
|
|
|
|
return (
|
|
<ContextMenu
|
|
{...props}
|
|
menuHeader={<MenuHeader title={song.title} subtitle={song.artist} coverArt={song.coverArt} />}
|
|
menuOptions={
|
|
<>
|
|
<OptionStar id={song.id} type={song.itemType} />
|
|
<OptionViewArtist artist={song.artist} artistId={song.artistId} navigation={navigation} />
|
|
<OptionViewAlbum album={song.album} albumId={song.albumId} navigation={navigation} />
|
|
{/* <OptionDownload itemType={song.itemType} /> */}
|
|
</>
|
|
}>
|
|
{children}
|
|
</ContextMenu>
|
|
)
|
|
}
|
|
|
|
export type ArtistContextPressableProps = ContextMenuProps & {
|
|
artist: Artist
|
|
}
|
|
|
|
export const ArtistContextPressable: React.FC<ArtistContextPressableProps> = props => {
|
|
const { artist, children } = props
|
|
|
|
return (
|
|
<ContextMenu
|
|
{...props}
|
|
menuHeader={<MenuHeader title={artist.name} artistId={artist.id} />}
|
|
menuOptions={
|
|
<>
|
|
<OptionStar id={artist.id} type={artist.itemType} />
|
|
{/* <OptionDownload itemType={artist.itemType} /> */}
|
|
</>
|
|
}>
|
|
{children}
|
|
</ContextMenu>
|
|
)
|
|
}
|
|
|
|
export type NowPlayingContextPressableProps = ContextMenuProps & {
|
|
song: Song
|
|
}
|
|
|
|
export const NowPlayingContextPressable: React.FC<NowPlayingContextPressableProps> = props => {
|
|
const navigation = useNavigation()
|
|
const { song, children } = props
|
|
|
|
return (
|
|
<ContextMenu
|
|
{...props}
|
|
menuHeader={<MenuHeader title={song.title} subtitle={song.artist} coverArt={song.coverArt} />}
|
|
menuOptions={
|
|
<>
|
|
<OptionStar id={song.id} type={song.itemType} />
|
|
<OptionViewArtist artist={song.artist} artistId={song.artistId} navigation={navigation} />
|
|
<OptionViewAlbum album={song.album} albumId={song.albumId} navigation={navigation} />
|
|
</>
|
|
}>
|
|
{children}
|
|
</ContextMenu>
|
|
)
|
|
}
|
|
|
|
const styles = StyleSheet.create({
|
|
optionsContainer: {
|
|
backgroundColor: 'rgba(45, 45, 45, 0.95)',
|
|
maxHeight: 365,
|
|
},
|
|
optionsWrapper: {
|
|
// marginBottom: 10,
|
|
},
|
|
menuHeader: {
|
|
paddingTop: 14,
|
|
paddingBottom: 10,
|
|
paddingHorizontal: 20,
|
|
flex: 1,
|
|
flexDirection: 'row',
|
|
justifyContent: 'center',
|
|
alignItems: 'center',
|
|
},
|
|
coverArt: {
|
|
width: 42,
|
|
height: 42,
|
|
},
|
|
menuHeaderText: {
|
|
flex: 1,
|
|
marginLeft: 10,
|
|
},
|
|
menuTitle: {
|
|
fontFamily: font.semiBold,
|
|
fontSize: 16,
|
|
color: colors.text.primary,
|
|
},
|
|
menuSubtitle: {
|
|
fontFamily: font.regular,
|
|
fontSize: 14,
|
|
color: colors.text.secondary,
|
|
},
|
|
option: {
|
|
paddingVertical: 8,
|
|
paddingHorizontal: 20,
|
|
flexDirection: 'row',
|
|
alignItems: 'center',
|
|
},
|
|
icon: {
|
|
marginRight: 10,
|
|
width: 32,
|
|
height: 32,
|
|
justifyContent: 'center',
|
|
alignItems: 'center',
|
|
// backgroundColor: 'red',
|
|
},
|
|
optionText: {
|
|
fontFamily: font.semiBold,
|
|
fontSize: 16,
|
|
color: colors.text.primary,
|
|
// backgroundColor: 'green',
|
|
},
|
|
})
|