import 'package:auto_route/auto_route.dart'; import 'package:easy_localization/easy_localization.dart'; import 'package:flutter/material.dart'; import 'package:flutter_svg/flutter_svg.dart'; import 'package:selfprivacy/logic/cubit/server_installation/server_installation_cubit.dart'; import 'package:selfprivacy/logic/cubit/backups/backups_cubit.dart'; import 'package:selfprivacy/logic/cubit/server_jobs/server_jobs_cubit.dart'; import 'package:selfprivacy/logic/cubit/services/services_cubit.dart'; import 'package:selfprivacy/logic/models/backup.dart'; import 'package:selfprivacy/logic/models/json/server_job.dart'; import 'package:selfprivacy/logic/models/service.dart'; import 'package:selfprivacy/logic/models/state_types.dart'; import 'package:selfprivacy/ui/components/buttons/brand_button.dart'; import 'package:selfprivacy/ui/components/cards/outlined_card.dart'; import 'package:selfprivacy/ui/layouts/brand_hero_screen.dart'; import 'package:selfprivacy/ui/components/brand_icons/brand_icons.dart'; import 'package:selfprivacy/ui/helpers/modals.dart'; GlobalKey navigatorKey = GlobalKey(); @RoutePage() class BackupDetailsPage extends StatefulWidget { const BackupDetailsPage({super.key}); @override State createState() => _BackupDetailsPageState(); } class _BackupDetailsPageState extends State with SingleTickerProviderStateMixin { @override Widget build(final BuildContext context) { final bool isReady = context.watch().state is ServerInstallationFinished; final bool isBackupInitialized = context.watch().state.isInitialized; final StateType providerState = isReady && isBackupInitialized ? StateType.stable : StateType.uninitialized; final bool preventActions = context.watch().state.preventActions; final List backups = context.watch().state.backups; final bool refreshing = context.watch().state.refreshing; final List services = context.watch().state.servicesThatCanBeBackedUp; return BrandHeroScreen( heroIcon: BrandIcons.save, heroTitle: 'backup.card_title'.tr(), heroSubtitle: 'backup.description'.tr(), children: [ if (isReady && !isBackupInitialized) BrandButton.rised( onPressed: preventActions ? null : () async { await context.read().initializeBackups(); }, text: 'backup.initialize'.tr(), ), ListTile( onTap: preventActions ? null : () { // await context.read().createBackup(); showModalBottomSheet( useRootNavigator: true, context: context, isScrollControlled: true, builder: (final BuildContext context) => DraggableScrollableSheet( expand: false, maxChildSize: 0.9, minChildSize: 0.4, initialChildSize: 0.6, builder: (final context, final scrollController) => CreateBackupsModal( services: services, scrollController: scrollController, ), ), ); }, leading: const Icon( Icons.add_circle_outline_rounded, ), title: Text( 'backup.create_new'.tr(), ), ), const SizedBox(height: 16), // Card with a list of existing backups // Each list item has a date // When clicked, starts the restore action if (isBackupInitialized) Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ ListTile( title: Text( 'backups.latest_snapshots'.tr(), style: Theme.of(context).textTheme.headlineSmall!.copyWith( color: Theme.of(context).colorScheme.secondary, ), ), subtitle: Text( 'backups.latest_snapshots_subtitle'.tr(), style: Theme.of(context).textTheme.labelMedium, ), ), if (backups.isEmpty) ListTile( leading: const Icon( Icons.error_outline, ), title: Text('backup.no_backups'.tr()), ), if (backups.isNotEmpty) Column( children: backups.take(20).map( (final Backup backup) { final service = context .read() .state .getServiceById(backup.serviceId); return ListTile( onTap: preventActions ? null : () { showPopUpAlert( alertTitle: 'backup.restoring'.tr(), description: 'backup.restore_alert'.tr( args: [backup.time.toString()], ), actionButtonTitle: 'modals.yes'.tr(), actionButtonOnPressed: () => { context .read() .restoreBackup(backup.id) }, ); }, title: Text( '${MaterialLocalizations.of(context).formatShortDate(backup.time)} ${TimeOfDay.fromDateTime(backup.time).format(context)}', ), subtitle: Text( service?.displayName ?? backup.fallbackServiceName, ), leading: service != null ? SvgPicture.string( service.svgIcon, height: 24, width: 24, colorFilter: ColorFilter.mode( Theme.of(context).colorScheme.onBackground, BlendMode.srcIn, ), ) : const Icon( Icons.question_mark_outlined, ), ); }, ).toList(), ), if (backups.isNotEmpty && backups.length > 20) ListTile( title: Text( 'backups.show_more'.tr(), style: Theme.of(context).textTheme.labelMedium, ), leading: const Icon( Icons.arrow_drop_down, ), onTap: null, ) ], ), const SizedBox(height: 16), OutlinedCard( child: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ ListTile( title: Text( 'backup.refresh'.tr(), ), onTap: refreshing ? null : () => {context.read().updateBackups()}, enabled: !refreshing, ), if (providerState != StateType.uninitialized) Column( children: [ const Divider( height: 1.0, ), ListTile( title: Text( 'backup.refetch_backups'.tr(), ), onTap: preventActions ? null : () => { context .read() .forceUpdateBackups() }, ), const Divider( height: 1.0, ), ListTile( title: Text( 'backup.reupload_key'.tr(), ), onTap: preventActions ? null : () => {context.read().reuploadKey()}, ), ], ), ], ), ), ], ); } } class CreateBackupsModal extends StatefulWidget { const CreateBackupsModal({ required this.services, required this.scrollController, super.key, }); final List services; final ScrollController scrollController; @override State createState() => _CreateBackupsModalState(); } class _CreateBackupsModalState extends State { // Store in state the selected services to backup List selectedServices = []; // Select all services on modal open @override void initState() { super.initState(); final List busyServices = context .read() .state .backupJobList .where( (final ServerJob job) => job.status == JobStatusEnum.running || job.status == JobStatusEnum.created, ) .map((final ServerJob job) => job.typeId.split('.')[1]) .toList(); selectedServices.addAll( widget.services .where((final Service service) => !busyServices.contains(service.id)), ); } @override Widget build(final BuildContext context) { final List busyServices = context .watch() .state .backupJobList .where( (final ServerJob job) => job.status == JobStatusEnum.running || job.status == JobStatusEnum.created, ) .map((final ServerJob job) => job.typeId.split('.')[1]) .toList(); return ListView( controller: widget.scrollController, padding: const EdgeInsets.all(16), children: [ const SizedBox(height: 16), Text( 'backup.create_new_select_heading'.tr(), style: Theme.of(context).textTheme.headlineSmall, textAlign: TextAlign.center, ), const SizedBox(height: 16), // Select all services tile CheckboxListTile( onChanged: (final bool? value) { setState(() { if (value ?? true) { setState(() { selectedServices.clear(); selectedServices.addAll( widget.services.where( (final service) => !busyServices.contains(service.id), ), ); }); } else { selectedServices.clear(); } }); }, title: Text( 'backup.select_all'.tr(), ), secondary: const Icon( Icons.checklist_outlined, ), value: selectedServices.length >= widget.services.length - busyServices.length, ), const Divider( height: 1.0, ), ...widget.services.map( (final Service service) { final bool busy = busyServices.contains(service.id); return CheckboxListTile( onChanged: !busy ? (final bool? value) { setState(() { if (value ?? true) { setState(() { selectedServices.add(service); }); } else { setState(() { selectedServices.remove(service); }); } }); } : null, title: Text( service.displayName, ), subtitle: Text( busy ? 'backup.service_busy'.tr() : service.backupDescription, ), secondary: SvgPicture.string( service.svgIcon, height: 24, width: 24, colorFilter: ColorFilter.mode( busy ? Theme.of(context).colorScheme.outlineVariant : Theme.of(context).colorScheme.onBackground, BlendMode.srcIn, ), ), value: selectedServices.contains(service), ); }, ), const SizedBox(height: 16), // Create backup button FilledButton( onPressed: selectedServices.isEmpty ? null : () { context .read() .createMultipleBackups(selectedServices); Navigator.of(context).pop(); }, child: Text( 'backup.start'.tr(), ), ), ], ); } }