ente/lib/ui/home/status_bar_widget.dart

278 lines
8.2 KiB
Dart
Raw Normal View History

import 'dart:async';
import 'package:flutter/material.dart';
import 'package:photos/core/event_bus.dart';
2022-07-12 06:30:02 +00:00
import 'package:photos/ente_theme_data.dart';
import 'package:photos/events/notification_event.dart';
2020-11-16 16:35:16 +00:00
import 'package:photos/events/sync_status_update_event.dart';
2020-10-03 17:58:26 +00:00
import 'package:photos/services/sync_service.dart';
2022-09-20 06:08:12 +00:00
import 'package:photos/services/user_remote_flag_service.dart';
import 'package:photos/theme/text_style.dart';
2022-09-20 06:08:12 +00:00
import 'package:photos/ui/account/verify_recovery_page.dart';
import 'package:photos/ui/components/home_header_widget.dart';
2022-09-20 06:08:12 +00:00
import 'package:photos/ui/components/notification_warning_widget.dart';
2022-10-20 05:48:47 +00:00
import 'package:photos/ui/home/header_error_widget.dart';
2022-09-20 06:08:12 +00:00
import 'package:photos/utils/navigation_util.dart';
2022-06-13 07:16:11 +00:00
const double kContainerHeight = 36;
2022-06-12 16:10:38 +00:00
2022-06-12 14:50:00 +00:00
class StatusBarWidget extends StatefulWidget {
const StatusBarWidget({Key? key}) : super(key: key);
@override
2022-06-12 16:10:38 +00:00
State<StatusBarWidget> createState() => _StatusBarWidgetState();
}
2022-06-12 14:50:00 +00:00
class _StatusBarWidgetState extends State<StatusBarWidget> {
late StreamSubscription<SyncStatusUpdate> _subscription;
late StreamSubscription<NotificationEvent> _notificationSubscription;
2022-06-12 16:10:38 +00:00
bool _showStatus = false;
bool _showErrorBanner = false;
Error? _syncError;
2022-06-12 16:10:38 +00:00
@override
void initState() {
_subscription = Bus.instance.on<SyncStatusUpdate>().listen((event) {
if (event.status == SyncStatus.error) {
setState(() {
_syncError = event.error;
_showErrorBanner = true;
});
} else {
setState(() {
_syncError = null;
_showErrorBanner = false;
});
}
2022-07-03 07:47:15 +00:00
if (event.status == SyncStatus.completedFirstGalleryImport ||
event.status == SyncStatus.completedBackup) {
2022-07-04 06:02:17 +00:00
Future.delayed(const Duration(milliseconds: 2000), () {
2022-06-12 16:10:38 +00:00
if (mounted) {
setState(() {
_showStatus = false;
});
}
});
} else {
setState(() {
_showStatus = true;
});
}
});
_notificationSubscription =
Bus.instance.on<NotificationEvent>().listen((event) {
if (mounted) {
setState(() {});
}
});
2022-06-12 16:10:38 +00:00
super.initState();
}
@override
void dispose() {
_subscription.cancel();
_notificationSubscription.cancel();
2022-06-12 16:10:38 +00:00
super.dispose();
}
@override
Widget build(BuildContext context) {
2022-08-24 11:16:48 +00:00
return Column(
children: [
HomeHeaderWidget(
centerWidget: _showStatus
? _showErrorBanner
? const Text("ente", style: brandStyleMedium)
: const SyncStatusWidget()
: const Text("ente", style: brandStyleMedium),
2022-08-24 11:16:48 +00:00
),
AnimatedOpacity(
opacity: _showErrorBanner ? 1 : 0,
duration: const Duration(milliseconds: 200),
child: const Divider(
height: 8,
),
2022-08-24 11:16:48 +00:00
),
_showErrorBanner
? HeaderErrorWidget(error: _syncError)
: const SizedBox.shrink(),
2022-09-20 06:08:12 +00:00
UserRemoteFlagService.instance.shouldShowRecoveryVerification()
? NotificationWarningWidget(
warningIcon: Icons.error_outline,
2022-09-20 06:08:12 +00:00
actionIcon: Icons.arrow_forward,
text: "Confirm your recovery key",
2022-09-20 06:08:12 +00:00
onTap: () async => {
await routeToPage(
context,
const VerifyRecoveryPage(),
forceCustomPageRoute: true,
)
},
)
: const SizedBox.shrink()
2022-08-24 11:16:48 +00:00
],
2022-06-12 16:10:38 +00:00
);
}
}
class SyncStatusWidget extends StatefulWidget {
const SyncStatusWidget({Key? key}) : super(key: key);
2022-06-12 16:10:38 +00:00
@override
2022-07-03 09:45:00 +00:00
State<SyncStatusWidget> createState() => _SyncStatusWidgetState();
2022-06-12 16:10:38 +00:00
}
class _SyncStatusWidgetState extends State<SyncStatusWidget> {
static const Duration kSleepDuration = Duration(milliseconds: 3000);
SyncStatusUpdate? _event;
late StreamSubscription<SyncStatusUpdate> _subscription;
2022-06-12 16:10:38 +00:00
@override
void initState() {
2020-11-12 16:32:10 +00:00
_subscription = Bus.instance.on<SyncStatusUpdate>().listen((event) {
setState(() {
_event = event;
});
});
2020-11-16 16:35:16 +00:00
_event = SyncService.instance.getLastSyncStatusEvent();
super.initState();
}
@override
void dispose() {
_subscription.cancel();
super.dispose();
}
@override
Widget build(BuildContext context) {
2022-08-29 14:43:31 +00:00
final bool isNotOutdatedEvent = _event != null &&
(_event!.status == SyncStatus.completedBackup ||
_event!.status == SyncStatus.completedFirstGalleryImport) &&
(DateTime.now().microsecondsSinceEpoch - _event!.timestamp >
2022-07-03 09:45:00 +00:00
kSleepDuration.inMicroseconds);
if (_event == null ||
isNotOutdatedEvent ||
2022-10-14 09:53:45 +00:00
//sync error cases are handled in StatusBarWidget
_event!.status == SyncStatus.error) {
return const SizedBox.shrink();
}
if (_event!.status == SyncStatus.completedBackup) {
2022-07-04 06:02:17 +00:00
return const SyncStatusCompletedWidget();
}
2022-06-12 16:28:50 +00:00
return RefreshIndicatorWidget(_event);
}
}
class RefreshIndicatorWidget extends StatelessWidget {
static const _inProgressIcon = CircularProgressIndicator(
strokeWidth: 2,
valueColor: AlwaysStoppedAnimation<Color>(Color.fromRGBO(45, 194, 98, 1.0)),
);
final SyncStatusUpdate? event;
2022-06-12 16:28:50 +00:00
const RefreshIndicatorWidget(this.event, {Key? key}) : super(key: key);
2022-06-12 16:28:50 +00:00
@override
Widget build(BuildContext context) {
2022-06-12 16:10:38 +00:00
return Container(
height: kContainerHeight,
alignment: Alignment.center,
child: SingleChildScrollView(
2022-07-04 06:02:17 +00:00
physics: const NeverScrollableScrollPhysics(),
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
crossAxisAlignment: CrossAxisAlignment.center,
children: [
Row(
mainAxisAlignment: MainAxisAlignment.center,
crossAxisAlignment: CrossAxisAlignment.center,
children: [
Container(
2022-07-04 06:02:17 +00:00
padding: const EdgeInsets.all(2),
width: 22,
height: 22,
2022-06-12 16:10:38 +00:00
child: _inProgressIcon,
),
Padding(
padding: const EdgeInsets.fromLTRB(12, 4, 0, 0),
child: Text(_getRefreshingText()),
),
],
),
],
),
),
);
}
2022-06-12 14:36:11 +00:00
String _getRefreshingText() {
if (event!.status == SyncStatus.startedFirstGalleryImport ||
event!.status == SyncStatus.completedFirstGalleryImport) {
2022-06-12 14:36:11 +00:00
return "Loading gallery...";
}
if (event!.status == SyncStatus.applyingRemoteDiff) {
2022-06-12 14:36:11 +00:00
return "Syncing...";
}
if (event!.status == SyncStatus.preparingForUpload) {
2022-06-12 14:36:11 +00:00
return "Encrypting backup...";
}
if (event!.status == SyncStatus.inProgress) {
return event!.completed.toString() +
2022-07-03 09:45:00 +00:00
"/" +
event!.total.toString() +
2022-07-03 09:45:00 +00:00
" memories preserved";
2022-06-12 14:36:11 +00:00
}
if (event!.status == SyncStatus.paused) {
return event!.reason;
2022-06-12 14:36:11 +00:00
}
if (event!.status == SyncStatus.error) {
2022-12-30 15:42:03 +00:00
return event!.reason;
2022-06-12 16:14:55 +00:00
}
if (event!.status == SyncStatus.completedBackup) {
if (event!.wasStopped) {
2022-06-12 14:36:11 +00:00
return "Sync stopped";
}
}
2022-06-12 16:14:55 +00:00
return "All memories preserved";
2022-06-12 14:36:11 +00:00
}
}
2022-06-12 16:10:38 +00:00
class SyncStatusCompletedWidget extends StatelessWidget {
const SyncStatusCompletedWidget({Key? key}) : super(key: key);
2022-06-12 16:10:38 +00:00
@override
Widget build(BuildContext context) {
return Container(
color: Theme.of(context).colorScheme.defaultBackgroundColor,
height: kContainerHeight,
2022-06-13 07:16:11 +00:00
child: Align(
alignment: Alignment.center,
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
crossAxisAlignment: CrossAxisAlignment.center,
children: [
Row(
mainAxisAlignment: MainAxisAlignment.center,
crossAxisAlignment: CrossAxisAlignment.center,
children: [
Icon(
Icons.cloud_done_outlined,
2022-07-12 06:30:02 +00:00
color: Theme.of(context).colorScheme.greenAlternative,
2022-06-13 07:16:11 +00:00
size: 22,
),
2022-07-04 06:02:17 +00:00
const Padding(
padding: EdgeInsets.only(left: 12),
2022-06-13 07:16:11 +00:00
child: Text("All memories preserved"),
),
],
),
],
),
),
2022-06-12 16:10:38 +00:00
);
}
}