Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[Redesign] Downloads Storage Rewrite #568

Merged
merged 90 commits into from
Feb 20, 2024
Merged
Show file tree
Hide file tree
Changes from 18 commits
Commits
Show all changes
90 commits
Select commit Hold shift + click to select a range
e938395
Added isar dependancy.
Komodo5197 Dec 16, 2023
a2bb6ff
Created initial Isar version of the download_helper.
Komodo5197 Dec 17, 2023
e5b0a68
Isar download storage now works. Can add, delete, and playback.
Komodo5197 Dec 18, 2023
2444d93
Most browsing tasks work now. Initial migration code.
Komodo5197 Dec 19, 2023
0d98815
Switched to background_downloader. Fully removed downloads_helper. …
Komodo5197 Dec 20, 2023
4d5ee16
Got numbers on downloads screen working. Got migration mostly functi…
Komodo5197 Dec 20, 2023
b015c33
Fix ios build.
Komodo5197 Dec 20, 2023
642fa0c
Merge branch 'main' of https://github.com/Komodo5197/finamp into main…
Komodo5197 Dec 20, 2023
0a2b55a
Addressed various TODOs.
Komodo5197 Dec 21, 2023
00a1b16
Fixed getting collectionInfo.
Komodo5197 Dec 21, 2023
1bfea88
Minor fixes.
Komodo5197 Dec 22, 2023
db142f5
Minor fixes.
Komodo5197 Dec 23, 2023
b934a86
Merge remote-tracking branch 'upstream/redesign' into main-downloads-…
Komodo5197 Dec 23, 2023
5e4960a
Merged into redesign. Needs more integration work.
Komodo5197 Dec 23, 2023
e8b4c7f
Improved metadata API. Added implementation of saving all songs for …
Komodo5197 Dec 24, 2023
b303dc0
Improved download/delete buttons and statusing. Got greyed-out songs…
Komodo5197 Dec 25, 2023
d7d6419
Switched to using info links instead of info types. This complicates…
Komodo5197 Dec 27, 2023
06e26cd
Changed download error screen into active downloads screen.
Komodo5197 Dec 28, 2023
7afa5c8
Upgraded dependancies. Got external download locations working, hope…
Komodo5197 Dec 30, 2023
97522b7
Fixed missing download completions. Added download library buttons. …
Komodo5197 Dec 31, 2023
2ba8813
Added secondary isar collection for download task data. Added backgr…
Komodo5197 Jan 1, 2024
0dd08a9
Added filtering by library to downloads display. Download BaseItemDt…
Komodo5197 Jan 2, 2024
c0fb4aa
Fixed migration breaking. Fixed artists library filtering.
Komodo5197 Jan 2, 2024
1a6c0f5
Various minor fixes. Added a bunch of comments.
Komodo5197 Jan 4, 2024
e08d2e5
Merge branch 'redesign' of https://github.com/jmshrv/finamp into main…
Komodo5197 Jan 4, 2024
f925264
Merged latest redesign changes.
Komodo5197 Jan 4, 2024
7d86016
Allowed offline deletes. Added download settings screen with configu…
Komodo5197 Jan 5, 2024
0770114
Added offline artist stuff. Created Isar queue for syncs to match the…
Komodo5197 Jan 6, 2024
12da8da
Improved downloads overview screen with syncing node count and song/i…
Komodo5197 Jan 6, 2024
495420f
Attempted to improve performance by reducing unneeded deserialization.
Komodo5197 Jan 7, 2024
9b98386
Various tweaks to hopefully improve performance and clean up some pos…
Komodo5197 Jan 8, 2024
e8ebb1e
Removed some throttling. Reduced database writes. Fixed a scenario …
Komodo5197 Jan 10, 2024
ed26b1b
Switched database writes to synchronous. Messed with throttling. De…
Komodo5197 Jan 11, 2024
036bce8
Moved some methods from isar_downloads to isar_downloads_backend. Im…
Komodo5197 Jan 12, 2024
b2290fc
Allowed falling back from blurhashes to imageIds - untested. Added s…
Komodo5197 Jan 13, 2024
5be8295
Improved item filesize display. Fixed sync error handling and retrie…
Komodo5197 Jan 14, 2024
6ca63b7
Threw some more finampCollection types in. Playlists and favorites a…
Komodo5197 Jan 15, 2024
549cd5f
Fixed light mode color issues. Fixed syncing while offline caching c…
Komodo5197 Jan 16, 2024
62ba887
Show requiring item in tooltip on incidental download button. Block …
Komodo5197 Jan 17, 2024
c758ae6
Added a way to localize names of finampCollections like favorites. A…
Komodo5197 Jan 18, 2024
833ed63
Moved getItems into worker isolate. Moved user store to Isar to assi…
Komodo5197 Jan 19, 2024
7f441a2
Tweaked throttling a bit. Added downloads repair indicator.
Komodo5197 Jan 20, 2024
9b295ff
Switched background_downloader to pull wifi github branch instead of …
Komodo5197 Jan 22, 2024
0ed1d5d
Merge branch 'redesign' into pr/Komodo5197/568
Chaphasilor Jan 23, 2024
3e1f118
Added mention of low-priority pause setting to background pause notice.
Komodo5197 Jan 24, 2024
3eec2ad
Removed max concurrent download limit on iOS.
Komodo5197 Jan 25, 2024
e9873ee
Added persistent logging to file.
Komodo5197 Jan 25, 2024
69ff2c6
Disabled logging to file.
Komodo5197 Jan 26, 2024
040ad03
Merge branch 'redesign' of https://github.com/jmshrv/finamp into main…
Komodo5197 Jan 26, 2024
3e2ae73
Merged latest redesign and cleaned up issues.
Komodo5197 Jan 26, 2024
ee6fe98
Improved offline sort order.
Komodo5197 Jan 26, 2024
a1179f1
Brand new throttling strategy. Fixed bug with music screen refresh. …
Komodo5197 Jan 26, 2024
2e2d7b2
Merge branch 'redesign' of https://github.com/jmshrv/finamp into main…
Komodo5197 Jan 27, 2024
9a34fe6
Correctly show playing downloaded songs.
Komodo5197 Jan 27, 2024
7702f2f
Re-enabled logging to file. Tweaked sync throttling.
Komodo5197 Jan 28, 2024
4ed4495
Cache current user data.
Komodo5197 Jan 28, 2024
1920a6f
Merge branch 'redesign' of https://github.com/jmshrv/finamp into main…
Komodo5197 Jan 28, 2024
6a28e66
Enabled offline functionality on new long press go to artist.
Komodo5197 Jan 28, 2024
bbce8d1
Handle paused downloads better. Delay startup for download queue ini…
Komodo5197 Feb 5, 2024
2055a2f
Avoid changing item path if not in notdownloaded state.
Komodo5197 Feb 6, 2024
71f3a14
Merge branch 'redesign' of https://github.com/jmshrv/finamp into main…
Komodo5197 Feb 6, 2024
458738d
Corrected errors with merge. Cleaned up authorization headers. Igno…
Komodo5197 Feb 7, 2024
84c21a5
Added transcoded downloads. This is mostly just a port of the code i…
Komodo5197 Feb 9, 2024
c2511f4
Merge branch 'redesign' of https://github.com/jmshrv/finamp into main…
Komodo5197 Feb 9, 2024
b579f50
Download settings are now properly updated by syncs so that items don…
Komodo5197 Feb 10, 2024
d3243aa
Merge branch 'redesign' of https://github.com/jmshrv/finamp into main…
Komodo5197 Feb 10, 2024
127e0c1
Merge fix.
Komodo5197 Feb 10, 2024
7443fe0
reduce padding of DownloadedItems list expansion tiles
Chaphasilor Feb 10, 2024
7d336f0
Fixed transcode settings propogation issues caused by use of a single…
Komodo5197 Feb 11, 2024
3fe5fff
Merge branch 'main-downloads-rework' of https://github.com/Komodo5197…
Komodo5197 Feb 11, 2024
d7e4de4
Downloads screen padding tweak.
Komodo5197 Feb 11, 2024
6931024
Merge branch 'redesign' of https://github.com/jmshrv/finamp into main…
Komodo5197 Feb 11, 2024
caea382
Merge cleanup.
Komodo5197 Feb 11, 2024
27465cf
Minor UI fixes.
Komodo5197 Feb 11, 2024
75f57ce
update active downloads button icon
Chaphasilor Feb 11, 2024
43e4e7b
Fixed possible issue where multiple download locations could result i…
Komodo5197 Feb 11, 2024
8472202
Merge branch 'redesign' of https://github.com/jmshrv/finamp into main…
Komodo5197 Feb 11, 2024
1def01d
Merge fixes.
Komodo5197 Feb 11, 2024
32ecb1b
Merge branch 'redesign' of https://github.com/jmshrv/finamp into main…
Komodo5197 Feb 14, 2024
7ca0f29
Updated background_downloader to stable.
Komodo5197 Feb 14, 2024
d0b539e
Merge branch 'redesign' of https://github.com/jmshrv/finamp into main…
Komodo5197 Feb 15, 2024
59d9ccd
Got extendbody working without cutting things off. Added shadow to i…
Komodo5197 Feb 15, 2024
659fafa
Removed rectangular playing bar setting.
Komodo5197 Feb 15, 2024
95fa529
Merge branch 'redesign' of https://github.com/jmshrv/finamp into main…
Komodo5197 Feb 16, 2024
116e699
Merge fixes. Upgraded gradle+dependancies after switch to flutter 3.19.
Komodo5197 Feb 16, 2024
255d2a7
code review fixes. Fixed some bugs with offline sorting.
Komodo5197 Feb 17, 2024
7c80734
Code review fixes. Copied playlist ordering code for albums.
Komodo5197 Feb 18, 2024
62b4df4
Code review fixes.
Komodo5197 Feb 18, 2024
eaa9d80
Code review fixes.
Komodo5197 Feb 20, 2024
b07aeca
Exclude Application Support from backups
jmshrv Feb 20, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 4 additions & 1 deletion analysis_options.yaml
Original file line number Diff line number Diff line change
@@ -1 +1,4 @@
include: package:flutter_lints/flutter.yaml
include: package:flutter_lints/flutter.yaml
linter:
rules:
- unawaited_futures
2 changes: 1 addition & 1 deletion ios/Flutter/AppFrameworkInfo.plist
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,6 @@
<key>CFBundleVersion</key>
<string>1.0</string>
<key>MinimumOSVersion</key>
<string>11.0</string>
<string>13.0</string>
</dict>
</plist>
4 changes: 2 additions & 2 deletions ios/Podfile
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
# Uncomment this line to define a global platform for your project
platform :ios, '11.0'
platform :ios, '13.0'

# Fixing DKImagePickerController Bug
use_modular_headers!
Expand Down Expand Up @@ -43,7 +43,7 @@ post_install do |installer|
flutter_additional_ios_build_settings(target)

target.build_configurations.each do |config|
config.build_settings['IPHONEOS_DEPLOYMENT_TARGET'] = '11.0'
config.build_settings['IPHONEOS_DEPLOYMENT_TARGET'] = '13.0'
end

target.build_configurations.each do |config|
Expand Down
9 changes: 0 additions & 9 deletions ios/Runner/AppDelegate.swift
Original file line number Diff line number Diff line change
@@ -1,6 +1,5 @@
import UIKit
import Flutter
import flutter_downloader

@UIApplicationMain
@objc class AppDelegate: FlutterAppDelegate {
Expand All @@ -9,7 +8,6 @@ import flutter_downloader
didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?
) -> Bool {
GeneratedPluginRegistrant.register(with: self)
FlutterDownloaderPlugin.setPluginRegistrantCallback(registerPlugins)

// Exclude the documents folder from iCloud backup since we keep songs there.
try! setExcludeFromiCloudBackup(isExcluded: true)
Expand All @@ -24,10 +22,3 @@ private func setExcludeFromiCloudBackup(isExcluded: Bool) throws {
values.isExcludedFromBackup = isExcluded
try fileOrDirectoryURL.setResourceValues(values)
}


private func registerPlugins(registry: FlutterPluginRegistry) {
if (!registry.hasPlugin("FlutterDownloaderPlugin")) {
FlutterDownloaderPlugin.register(with: registry.registrar(forPlugin: "FlutterDownloaderPlugin")!)
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ import 'package:get_it/get_it.dart';
import '../../models/jellyfin_models.dart';
import '../../services/jellyfin_api_helper.dart';
import '../MusicScreen/album_item.dart';
import '../error_snackbar.dart';
import '../global_snackbar.dart';

class AddToPlaylistList extends StatefulWidget {
const AddToPlaylistList({
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@ import 'package:get_it/get_it.dart';

import '../../services/finamp_user_helper.dart';
import '../../services/jellyfin_api_helper.dart';
import '../error_snackbar.dart';
import '../global_snackbar.dart';

class NewPlaylistDialog extends StatefulWidget {
const NewPlaylistDialog({
Expand Down
28 changes: 16 additions & 12 deletions lib/components/AlbumScreen/album_screen_content.dart
Original file line number Diff line number Diff line change
@@ -1,10 +1,9 @@
import 'dart:async';

import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:flutter_gen/gen_l10n/app_localizations.dart';
import 'package:flutter_sticky_header/flutter_sticky_header.dart';

import '../../models/finamp_models.dart';
import '../../models/jellyfin_models.dart';
import '../../services/finamp_settings_helper.dart';
import '../../components/favourite_button.dart';
Expand All @@ -20,10 +19,12 @@ class AlbumScreenContent extends StatefulWidget {
Key? key,
required this.parent,
required this.children,
required this.playableChildren,
}) : super(key: key);

final BaseItemDto parent;
final List<BaseItemDto> children;
final List<BaseItemDto> playableChildren;

Komodo5197 marked this conversation as resolved.
Show resolved Hide resolved
@override
State<AlbumScreenContent> createState() => _AlbumScreenContentState();
Expand All @@ -38,6 +39,7 @@ class _AlbumScreenContentState extends State<AlbumScreenContent> {
// handle multi-disc albums and it's 00:35 so I can't be bothered to get
// it to return an index
setState(() {
widget.playableChildren.remove(item);
widget.children.remove(item);
});
}
Expand All @@ -46,6 +48,7 @@ class _AlbumScreenContentState extends State<AlbumScreenContent> {
// if not in playlist, try splitting up tracks by disc numbers
// if first track has a disc number, let's assume the rest has it too
if (widget.parent.type != "Playlist" &&
widget.children.isNotEmpty &&
widget.children[0].parentIndexNumber != null) {
int? lastDiscNumber;
for (var child in widget.children) {
Expand Down Expand Up @@ -73,14 +76,17 @@ class _AlbumScreenContentState extends State<AlbumScreenContent> {
flexibleSpace: AlbumScreenContentFlexibleSpaceBar(
parentItem: widget.parent,
isPlaylist: widget.parent.type == "Playlist",
items: widget.children,
items: widget.playableChildren,
),
actions: [
if (widget.parent.type == "Playlist" &&
!FinampSettingsHelper.finampSettings.isOffline)
PlaylistNameEditButton(playlist: widget.parent),
FavoriteButton(item: widget.parent),
DownloadButton(parent: widget.parent, items: widget.children)
DownloadButton(
item: DownloadStub.fromItem(
type: DownloadItemType.collection, item: widget.parent),
children: widget.children.length)
],
),
if (widget.children.length > 1 &&
Expand All @@ -102,15 +108,15 @@ class _AlbumScreenContentState extends State<AlbumScreenContent> {
),
sliver: SongsSliverList(
childrenForList: childrenOfThisDisc,
childrenForQueue: Future.value(widget.children),
childrenForQueue: Future.value(widget.playableChildren),
parent: widget.parent,
onRemoveFromList: onDelete,
),
)
else if (widget.children.isNotEmpty)
SongsSliverList(
childrenForList: widget.children,
childrenForQueue: Future.value(widget.children),
childrenForQueue: Future.value(widget.playableChildren),
parent: widget.parent,
onRemoveFromList: onDelete,
),
Expand Down Expand Up @@ -159,10 +165,8 @@ class _SongsSliverListState extends State<SongsSliverList> {
// When user selects song from disc other than first, index number is
// incorrect and song with the same index on first disc is played instead.
// Adding this offset ensures playback starts for nth song on correct disc.
final indexOffset = widget.parent.type == "MusicAlbum"
? widget.childrenForQueue.then((childrenForQueue) =>
childrenForQueue.indexOf(widget.childrenForList[0]))
: Future.value(0);
final indexOffset = widget.childrenForQueue.then((childrenForQueue) =>
childrenForQueue.indexWhere((element) => element.id == widget.childrenForList[index].id));

final BaseItemDto item = widget.childrenForList[index];

Expand All @@ -178,8 +182,8 @@ class _SongsSliverListState extends State<SongsSliverList> {

return SongListTile(
item: item,
childrenFuture: widget.childrenForQueue,
indexFuture: indexOffset.then((offset) => offset + index),
children: widget.childrenForQueue,
index: indexOffset,
parentItem: widget.parent,
onRemoveFromList: () {
final item = removeItem();
Expand Down
176 changes: 82 additions & 94 deletions lib/components/AlbumScreen/download_button.dart
Original file line number Diff line number Diff line change
@@ -1,125 +1,113 @@
import 'package:flutter/material.dart';
import 'package:flutter_gen/gen_l10n/app_localizations.dart';
import 'package:flutter_riverpod/flutter_riverpod.dart';
import 'package:get_it/get_it.dart';
import 'package:hive/hive.dart';

import '../../services/downloads_helper.dart';
import '../../services/finamp_settings_helper.dart';
import '../../services/finamp_user_helper.dart';
import '../../models/jellyfin_models.dart';
import '../../models/finamp_models.dart';
import '../error_snackbar.dart';
import '../../services/isar_downloads.dart';
import '../global_snackbar.dart';
import 'download_dialog.dart';
import '../confirmation_prompt_dialog.dart';

class DownloadButton extends StatefulWidget {
class DownloadButton extends ConsumerWidget {
const DownloadButton({
Key? key,
required this.parent,
required this.items,
required this.item,
this.children,
}) : super(key: key);

final BaseItemDto parent;
final List<BaseItemDto> items;
final DownloadStub item;
final int? children;

@override
State<DownloadButton> createState() => _DownloadButtonState();
}

class _DownloadButtonState extends State<DownloadButton> {
final _downloadsHelper = GetIt.instance<DownloadsHelper>();
final _finampUserHelper = GetIt.instance<FinampUserHelper>();
late bool isDownloaded;

@override
void initState() {
super.initState();
isDownloaded = _downloadsHelper.isAlbumDownloaded(widget.parent.id);
}

@override
Widget build(BuildContext context) {
void checkIfDownloaded() {
if (!mounted) return;
setState(() {
isDownloaded = _downloadsHelper.isAlbumDownloaded(widget.parent.id);
});
}
Widget build(BuildContext context, WidgetRef ref) {
final isarDownloads = GetIt.instance<IsarDownloads>();
var status =
ref.watch(isarDownloads.statusProvider((item, children))).value;

return ValueListenableBuilder<Box<FinampSettings>>(
valueListenable: FinampSettingsHelper.finampSettingsListener,
builder: (context, box, child) {
bool? isOffline = box.get("FinampSettings")?.isOffline;
if (FinampSettingsHelper.finampSettings.isOffline) {
return const SizedBox.shrink();
}

return IconButton(
icon: isDownloaded
? const Icon(Icons.delete)
: const Icon(Icons.file_download),
var downloadButton = IconButton(
icon: status == DownloadItemStatus.notNeeded
? const Icon(Icons.file_download)
: const Icon(Icons.lock), //TODO get better icon
onPressed: () {
if (FinampSettingsHelper
.finampSettings.downloadLocationsMap.length ==
1) {
isarDownloads.addDownload(
stub: item,
downloadLocation: FinampSettingsHelper
.finampSettings.downloadLocationsMap.values.first);
} else {
showDialog(
context: context,
builder: (context) => DownloadDialog(
item: item,
),
);
}
},
);
var deleteButton = IconButton(
icon: const Icon(Icons.delete),
// If offline, we don't allow the user to delete items.
// If we did, we'd have to implement listeners for MusicScreenTabView so that the user can't delete a parent, go back, and select the same parent.
// If they did, AlbumScreen would show an error since the item no longer exists.
// Also, the user could delete the parent and immediately redownload it, which will either cause unwanted network usage or cause more errors because the user is offline.
onPressed: isOffline ?? false
? null
: () {
if (isDownloaded) {
showDialog(
context: context,
builder: (context) => ConfirmationPromptDialog(
promptText: AppLocalizations.of(context)!
.deleteDownloadsPrompt(
widget.parent.name ?? "",
widget.parent.type == "Playlist"
? "playlist"
: "album"),
confirmButtonText: AppLocalizations.of(context)!
.deleteDownloadsConfirmButtonText,
abortButtonText: AppLocalizations.of(context)!
.deleteDownloadsAbortButtonText,
onConfirmed: () async {
final messenger = ScaffoldMessenger.of(context);
try {
await _downloadsHelper.deleteDownloads(
jellyfinItemIds:
widget.items.map((e) => e.id).toList(),
deletedFor: widget.parent.id);
checkIfDownloaded();
messenger.showSnackBar(SnackBar(
content: Text(AppLocalizations.of(context)!
.downloadsDeleted)));
} catch (error) {
errorSnackbar(error, context);
}
},
onAborted: () {},
),
);
// .whenComplete(() => checkIfDownloaded());
} else {
if (FinampSettingsHelper
.finampSettings.downloadLocationsMap.length ==
1) {
checkedAddDownloads(
context,
downloadLocation: FinampSettingsHelper
.finampSettings.downloadLocationsMap.values.first,
parents: [widget.parent],
items: [widget.items],
viewId: _finampUserHelper.currentUser!.currentViewId!,
).whenComplete(() => checkIfDownloaded());
} else {
showDialog(
context: context,
builder: (context) => DownloadDialog(
parents: [widget.parent],
items: [widget.items],
viewId: _finampUserHelper.currentUser!.currentViewId!,
),
).whenComplete(() => checkIfDownloaded());
}
onPressed: () {
showDialog(
context: context,
builder: (context) => ConfirmationPromptDialog(
promptText: AppLocalizations.of(context)!.deleteDownloadsPrompt(
item.baseItem?.name ?? "", item.type.name),
confirmButtonText: AppLocalizations.of(context)!
.deleteDownloadsConfirmButtonText,
abortButtonText: AppLocalizations.of(context)!
.deleteDownloadsAbortButtonText,
onConfirmed: () async {
final messenger = ScaffoldMessenger.of(context);
final text = AppLocalizations.of(context)!.downloadsDeleted;
try {
await isarDownloads.deleteDownload(stub: item);
messenger.showSnackBar(SnackBar(content: Text(text)));
} catch (error) {
GlobalSnackbar.error(error);
}
},
onAborted: () {},
),
);
// .whenComplete(() => checkIfDownloaded());
},
);
var syncButton = IconButton(
icon: const Icon(Icons.sync),
onPressed: () {
isarDownloads.resync(item);
},
color: status?.outdated ?? false ? Colors.yellow : null,
);
var coreButton =
status?.isRequired ?? true ? deleteButton : downloadButton;
if (status != null) {
if (status == DownloadItemStatus.notNeeded ||
item.baseItemType == BaseItemDtoType.album ||
item.baseItemType == BaseItemDtoType.song) {
return coreButton;
} else {
return Row(children: [syncButton, coreButton]);
}
} else {
return const SizedBox.shrink();
}
},
);
}
Expand Down
Loading
Loading