mirror of
https://github.com/immich-app/immich.git
synced 2024-12-24 10:37:28 +02:00
4fe535e5e8
This change greatly reduces the chance that a backup is not performed when a new photo/video is made. Instead of combining the change trigger and additonal constraints (wifi or charging) into a single worker, these aspects are now separated. Thus, it is now reliably possible to take pictures while the wifi constraint is not satisfied and upload them hours/days later once connected to wifi without taking a new photo. As a positive side effect, this simplifies the error/retry handling by directly leveraging Android's WorkManager without workarounds. The separation also allows to notify the currently running BackupWorker that new assets were added while backing up other assets to also upload those newly added assets. Further, a new tiny service checks if the app is killed, to reschedule the content change worker and allow to detect the first new photo. Bonus: The home screen now shows backup as enabled if background backup is active. * use separate worker/task for listening on changed/added assets * use separate worker/task for performing the backup * content observer worker enqueues backup worker on each new asset * wifi/charging constraints only apply to backup worker * backupworker is notified of assets added while running to re-run * new service to catch app being killed to workaround WorkManager issue
156 lines
5.4 KiB
Dart
156 lines
5.4 KiB
Dart
import 'package:auto_route/auto_route.dart';
|
|
import 'package:badges/badges.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
|
import 'package:immich_mobile/modules/login/providers/authentication.provider.dart';
|
|
|
|
import 'package:immich_mobile/routing/router.dart';
|
|
import 'package:immich_mobile/modules/backup/models/backup_state.model.dart';
|
|
import 'package:immich_mobile/shared/models/server_info_state.model.dart';
|
|
import 'package:immich_mobile/modules/backup/providers/backup.provider.dart';
|
|
import 'package:immich_mobile/shared/providers/server_info.provider.dart';
|
|
|
|
class ImmichSliverAppBar extends ConsumerWidget {
|
|
const ImmichSliverAppBar({
|
|
Key? key,
|
|
this.onPopBack,
|
|
}) : super(key: key);
|
|
|
|
final Function? onPopBack;
|
|
|
|
@override
|
|
Widget build(BuildContext context, WidgetRef ref) {
|
|
final BackUpState backupState = ref.watch(backupProvider);
|
|
bool isEnableAutoBackup = backupState.backgroundBackup ||
|
|
ref.watch(authenticationProvider).deviceInfo.isAutoBackup;
|
|
final ServerInfoState serverInfoState = ref.watch(serverInfoProvider);
|
|
|
|
return SliverAppBar(
|
|
centerTitle: true,
|
|
floating: true,
|
|
pinned: false,
|
|
snap: false,
|
|
backgroundColor: Theme.of(context).appBarTheme.backgroundColor,
|
|
shape: const RoundedRectangleBorder(
|
|
borderRadius: BorderRadius.all(Radius.circular(5)),
|
|
),
|
|
leading: Builder(
|
|
builder: (BuildContext context) {
|
|
return Stack(
|
|
children: [
|
|
Positioned(
|
|
top: 5,
|
|
child: IconButton(
|
|
splashRadius: 25,
|
|
icon: Icon(
|
|
Icons.face_outlined,
|
|
size: 30,
|
|
color: Theme.of(context).primaryColor,
|
|
),
|
|
onPressed: () {
|
|
Scaffold.of(context).openDrawer();
|
|
},
|
|
),
|
|
),
|
|
if (serverInfoState.isVersionMismatch)
|
|
Positioned(
|
|
bottom: 12,
|
|
right: 12,
|
|
child: GestureDetector(
|
|
onTap: () => Scaffold.of(context).openDrawer(),
|
|
child: Material(
|
|
// color: Colors.grey[200],
|
|
elevation: 1,
|
|
shape: RoundedRectangleBorder(
|
|
borderRadius: BorderRadius.circular(50.0),
|
|
),
|
|
child: const Padding(
|
|
padding: EdgeInsets.all(2.0),
|
|
child: Icon(
|
|
Icons.info,
|
|
color: Color.fromARGB(255, 243, 188, 106),
|
|
size: 15,
|
|
),
|
|
),
|
|
),
|
|
),
|
|
),
|
|
],
|
|
);
|
|
},
|
|
),
|
|
title: const Text(
|
|
'IMMICH',
|
|
style: TextStyle(
|
|
fontFamily: 'SnowburstOne',
|
|
fontWeight: FontWeight.bold,
|
|
fontSize: 22,
|
|
),
|
|
),
|
|
actions: [
|
|
Stack(
|
|
alignment: AlignmentDirectional.center,
|
|
children: [
|
|
if (backupState.backupProgress == BackUpProgressEnum.inProgress)
|
|
Positioned(
|
|
top: 10,
|
|
right: 12,
|
|
child: SizedBox(
|
|
height: 8,
|
|
width: 8,
|
|
child: CircularProgressIndicator(
|
|
strokeWidth: 1,
|
|
valueColor: AlwaysStoppedAnimation<Color>(
|
|
Theme.of(context).primaryColor,
|
|
),
|
|
),
|
|
),
|
|
),
|
|
IconButton(
|
|
splashRadius: 25,
|
|
iconSize: 30,
|
|
icon: isEnableAutoBackup
|
|
? Icon(
|
|
Icons.backup_rounded,
|
|
color: Theme.of(context).primaryColor,
|
|
)
|
|
: Badge(
|
|
padding: const EdgeInsets.all(4),
|
|
elevation: 3,
|
|
position: BadgePosition.bottomEnd(bottom: -4, end: -4),
|
|
badgeColor: Colors.white,
|
|
badgeContent: const Icon(
|
|
Icons.cloud_off_rounded,
|
|
size: 8,
|
|
color: Colors.indigo,
|
|
),
|
|
child: Icon(
|
|
Icons.backup_rounded,
|
|
color: Theme.of(context).primaryColor,
|
|
),
|
|
),
|
|
onPressed: () async {
|
|
var onPop = await AutoRouter.of(context)
|
|
.push(const BackupControllerRoute());
|
|
|
|
if (onPop != null && onPop == true) {
|
|
onPopBack!();
|
|
}
|
|
},
|
|
),
|
|
if (backupState.backupProgress == BackUpProgressEnum.inProgress)
|
|
Positioned(
|
|
bottom: 5,
|
|
child: Text(
|
|
'${backupState.allUniqueAssets.length - backupState.selectedAlbumsBackupAssetsIds.length}',
|
|
style:
|
|
const TextStyle(fontSize: 9, fontWeight: FontWeight.bold),
|
|
),
|
|
),
|
|
],
|
|
),
|
|
],
|
|
);
|
|
}
|
|
}
|