Files
jellyfin-audio-player/src/screens/Downloads/index.tsx

164 lines
6.3 KiB
TypeScript
Raw Normal View History

2022-01-02 02:28:52 +01:00
import useDefaultStyles from 'components/Colors';
import React, { useCallback, useMemo } from 'react';
2022-05-17 23:34:25 +02:00
import { FlatListProps, View } from 'react-native';
2022-01-02 02:28:52 +01:00
import { FlatList } from 'react-native-gesture-handler';
import { SafeAreaView } from 'react-native-safe-area-context';
2022-05-18 22:10:06 +02:00
import { useAppDispatch, useTypedSelector } from 'store';
2022-01-02 02:28:52 +01:00
import formatBytes from 'utility/formatBytes';
2022-02-11 17:44:04 +02:00
import TrashIcon from 'assets/icons/trash.svg';
import ArrowClockwise from 'assets/icons/arrow-clockwise.svg';
2022-01-02 02:28:52 +01:00
import { EntityId } from '@reduxjs/toolkit';
import { queueTrackForDownload, removeDownloadedTrack } from 'store/downloads/actions';
2022-01-02 02:28:52 +01:00
import Button from 'components/Button';
import { t } from 'i18n-js';
2022-01-02 18:11:30 +01:00
import DownloadIcon from 'components/DownloadIcon';
2022-01-02 19:16:12 +01:00
import styled from 'styled-components/native';
2022-05-17 23:34:25 +02:00
import { Text } from 'components/Typography';
import FastImage from 'react-native-fast-image';
import { useGetImage } from 'utility/JellyfinApi';
import { ShadowWrapper } from 'components/Shadow';
2022-01-02 19:16:12 +01:00
const DownloadedTrack = styled.View`
flex: 1 0 auto;
flex-direction: row;
padding: 8px 0;
align-items: center;
margin: 0 20px;
`;
2022-01-02 02:28:52 +01:00
2022-05-17 23:34:25 +02:00
const AlbumImage = styled(FastImage)`
height: 32px;
width: 32px;
border-radius: 4px;
`;
2022-01-02 02:28:52 +01:00
function Downloads() {
const defaultStyles = useDefaultStyles();
2022-05-18 22:10:06 +02:00
const dispatch = useAppDispatch();
2022-05-17 23:34:25 +02:00
const getImage = useGetImage();
2022-01-02 02:28:52 +01:00
const { entities, ids } = useTypedSelector((state) => state.downloads);
const tracks = useTypedSelector((state) => state.music.tracks.entities);
// Calculate the total download size
const totalDownloadSize = useMemo(() => (
ids?.reduce<number>((sum, id) => sum + (entities[id]?.size || 0), 0)
), [ids, entities]);
2022-01-02 19:16:12 +01:00
/**
* Handlers for actions in this components
*/
// Delete a single downloaded track
2022-01-02 02:28:52 +01:00
const handleDelete = useCallback((id: EntityId) => {
dispatch(removeDownloadedTrack(id));
}, [dispatch]);
2022-01-02 19:16:12 +01:00
// Delete all downloaded tracks
const handleDeleteAllTracks = useCallback(() => ids.forEach(handleDelete), [handleDelete, ids]);
// Retry a single failed track
2022-01-02 18:11:30 +01:00
const retryTrack = useCallback((id: EntityId) => {
dispatch(queueTrackForDownload(id));
2022-01-02 18:11:30 +01:00
}, [dispatch]);
2022-01-02 02:28:52 +01:00
2022-01-02 19:16:12 +01:00
// Retry all failed tracks
const failedIds = useMemo(() => ids.filter((id) => !entities[id]?.isComplete), [ids, entities]);
const handleRetryFailed = useCallback(() => (
failedIds.forEach(retryTrack)
), [failedIds, retryTrack]);
/**
* Render section
*/
const ListHeaderComponent = useMemo(() => (
2022-05-17 23:34:25 +02:00
<View style={[{ paddingHorizontal: 20, paddingBottom: 12, borderBottomWidth: 0.5 }, defaultStyles.border]}>
<View style={{ flexDirection: 'row', alignItems: 'center' }}>
<Text
style={[
defaultStyles.textHalfOpacity,
{ marginRight: 8, flex: 1, fontSize: 12 },
]}
numberOfLines={1}
>
{t('total-download-size')}: {formatBytes(totalDownloadSize)}
</Text>
<Button
icon={TrashIcon}
title={t('delete-all-tracks')}
onPress={handleDeleteAllTracks}
disabled={!ids.length}
size="small"
/>
</View>
{failedIds.length > 0 && (
<Button
icon={ArrowClockwise}
title={t('retry-failed-downloads')}
onPress={handleRetryFailed}
disabled={failedIds.length === 0}
style={{ marginTop: 4 }}
/>
)}
2022-01-02 19:16:12 +01:00
</View>
), [totalDownloadSize, defaultStyles, failedIds.length, handleRetryFailed, handleDeleteAllTracks, ids.length]);
const renderItem = useCallback<NonNullable<FlatListProps<EntityId>['renderItem']>>(({ item }) => (
2022-05-11 23:57:30 +02:00
<DownloadedTrack>
2022-01-02 19:16:12 +01:00
<View style={{ marginRight: 12 }}>
2022-05-17 23:34:25 +02:00
<ShadowWrapper size="small">
<AlbumImage source={{ uri: getImage(item as string) }} style={defaultStyles.imageBackground} />
</ShadowWrapper>
2022-01-02 19:16:12 +01:00
</View>
<View style={{ flexShrink: 1, marginRight: 8 }}>
2022-01-02 23:02:54 +01:00
<Text style={[{ fontSize: 16, marginBottom: 4 }, defaultStyles.text]} numberOfLines={1}>
2022-01-02 19:16:12 +01:00
{tracks[item]?.Name}
</Text>
<Text style={[{ flexShrink: 1, fontSize: 11 }, defaultStyles.textHalfOpacity]} numberOfLines={1}>
2022-05-17 23:34:25 +02:00
{tracks[item]?.AlbumArtist} {tracks[item]?.Album ? `${tracks[item]?.Album}` : ''}
2022-01-02 19:16:12 +01:00
</Text>
</View>
<View style={{ marginLeft: 'auto', flexDirection: 'row', alignItems: 'center' }}>
{entities[item]?.isComplete && entities[item]?.size ? (
2022-05-17 23:34:25 +02:00
<Text style={[defaultStyles.textQuarterOpacity, { marginRight: 12, fontSize: 12 }]}>
2022-01-02 19:16:12 +01:00
{formatBytes(entities[item]?.size || 0)}
</Text>
) : null}
2022-05-17 23:34:25 +02:00
<View style={{ marginRight: 12 }}>
<DownloadIcon trackId={item} />
</View>
<Button onPress={() => handleDelete(item)} size="small" icon={TrashIcon} />
2022-01-02 19:16:12 +01:00
{!entities[item]?.isComplete && (
2022-05-17 23:34:25 +02:00
<Button onPress={() => retryTrack(item)} size="small" icon={ArrowClockwise} style={{ marginLeft: 4 }} />
2022-01-02 19:16:12 +01:00
)}
</View>
</DownloadedTrack>
2022-05-17 23:34:25 +02:00
), [entities, retryTrack, handleDelete, defaultStyles, tracks, getImage]);
2022-01-02 19:16:12 +01:00
// If no tracks have been downloaded, show a short message describing this
2022-01-02 02:28:52 +01:00
if (!ids.length) {
return (
<View style={{ margin: 24, flex: 1, alignItems: 'center', justifyContent: 'center' }}>
<Text style={[{ textAlign: 'center'}, defaultStyles.textHalfOpacity]}>
{t('no-downloads')}
</Text>
</View>
);
}
return (
2022-01-02 18:11:30 +01:00
<SafeAreaView style={{ flex: 1 }}>
2022-05-17 23:34:25 +02:00
{ListHeaderComponent}
2022-01-02 02:28:52 +01:00
<FlatList
2022-01-02 19:16:12 +01:00
data={ids}
2022-05-17 23:34:25 +02:00
style={{ flex: 1, paddingTop: 12 }}
2022-01-02 18:11:30 +01:00
contentContainerStyle={{ flexGrow: 1 }}
2022-01-02 19:16:12 +01:00
renderItem={renderItem}
2022-01-02 02:28:52 +01:00
/>
</SafeAreaView>
);
}
export default Downloads;