import 'package:easy_localization/easy_localization.dart'; import 'package:flutter/material.dart'; import 'package:flutter_svg/flutter_svg.dart'; import 'package:selfprivacy/logic/bloc/backups/backups_bloc.dart'; import 'package:selfprivacy/logic/bloc/server_jobs/server_jobs_bloc.dart'; import 'package:selfprivacy/logic/bloc/volumes/volumes_bloc.dart'; import 'package:selfprivacy/logic/models/service.dart'; 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.busyServices; selectedServices.addAll( widget.services .where((final Service service) => !busyServices.contains(service.id)), ); } @override Widget build(final BuildContext context) { final List busyServices = context.watch().state.busyServices; 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.adaptive( 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); final List descriptionWidgets = []; if (busy) { descriptionWidgets.add(Text('backup.service_busy'.tr())); } else { descriptionWidgets.add( Text( 'service_page.uses'.tr( namedArgs: { 'usage': service.storageUsage.used.toString(), 'volume': context .read() .state .getVolume(service.storageUsage.volume ?? '') .displayName, }, ), style: Theme.of(context).textTheme.labelMedium, ), ); descriptionWidgets.add( Text(service.backupDescription), ); } return CheckboxListTile.adaptive( onChanged: !busy ? (final bool? value) { setState(() { if (value ?? true) { setState(() { selectedServices.add(service); }); } else { setState(() { selectedServices.remove(service); }); } }); } : null, title: Text( service.displayName, ), subtitle: Column( crossAxisAlignment: CrossAxisAlignment.start, children: descriptionWidgets, ), secondary: SvgPicture.string( service.svgIcon, height: 24, width: 24, colorFilter: ColorFilter.mode( busy ? Theme.of(context).colorScheme.outlineVariant : Theme.of(context).colorScheme.onSurface, BlendMode.srcIn, ), ), value: selectedServices.contains(service), ); }, ), const SizedBox(height: 16), // Create backup button FilledButton( onPressed: selectedServices.isEmpty ? null : () { context .read() .add(CreateBackups(selectedServices)); Navigator.of(context).pop(); }, child: Text( 'backup.start'.tr(), ), ), ], ); } }