import 'dart:async'; import 'package:flutter/material.dart'; import 'package:photos/core/configuration.dart'; import 'package:photos/core/event_bus.dart'; import 'package:photos/ente_theme_data.dart'; import 'package:photos/events/two_factor_status_change_event.dart'; import "package:photos/generated/l10n.dart"; import 'package:photos/services/local_authentication_service.dart'; import 'package:photos/services/user_service.dart'; import 'package:photos/theme/ente_theme.dart'; import "package:photos/ui/account/recovery_key_page.dart"; import 'package:photos/ui/account/sessions_page.dart'; import 'package:photos/ui/components/captioned_text_widget.dart'; import 'package:photos/ui/components/expandable_menu_item_widget.dart'; import 'package:photos/ui/components/menu_item_widget/menu_item_widget.dart'; import 'package:photos/ui/components/toggle_switch_widget.dart'; import 'package:photos/ui/settings/common_settings.dart'; import "package:photos/utils/crypto_util.dart"; import "package:photos/utils/dialog_util.dart"; import "package:photos/utils/navigation_util.dart"; import "package:photos/utils/toast_util.dart"; class SecuritySectionWidget extends StatefulWidget { const SecuritySectionWidget({Key? key}) : super(key: key); @override State createState() => _SecuritySectionWidgetState(); } class _SecuritySectionWidgetState extends State { final _config = Configuration.instance; late StreamSubscription _twoFactorStatusChangeEvent; @override void initState() { super.initState(); _twoFactorStatusChangeEvent = Bus.instance.on().listen((event) async { if (mounted) { setState(() {}); } }); } @override void dispose() { _twoFactorStatusChangeEvent.cancel(); super.dispose(); } @override Widget build(BuildContext context) { return ExpandableMenuItemWidget( title: S.of(context).security, selectionOptionsWidget: _getSectionOptions(context), leadingIcon: Icons.local_police_outlined, ); } Widget _getSectionOptions(BuildContext context) { final bool canDisableMFA = UserService.instance.getCachedUserDetails()?.profileData?.canDisableEmailMFA ?? false; final Completer completer = Completer(); final List children = []; if (_config.hasConfiguredAccount()) { children.addAll( [ sectionOptionSpacing, MenuItemWidget( captionedTextWidget: CaptionedTextWidget( title: S.of(context).recoveryKey, ), pressedColor: getEnteColorScheme(context).fillFaint, trailingIcon: Icons.chevron_right_outlined, trailingIconIsMuted: true, showOnlyLoadingState: true, onTap: () async { final hasAuthenticated = await LocalAuthenticationService.instance .requestLocalAuthentication( context, S.of(context).authToViewYourRecoveryKey, ); if (hasAuthenticated) { String recoveryKey; try { recoveryKey = await _getOrCreateRecoveryKey(context); } catch (e) { await showGenericErrorDialog(context: context); return; } unawaited( routeToPage( context, RecoveryKeyPage( recoveryKey, S.of(context).ok, showAppBar: true, onDone: () {}, ), ), ); } }, ), sectionOptionSpacing, MenuItemWidget( captionedTextWidget: CaptionedTextWidget( title: S.of(context).twofactor, ), trailingWidget: ToggleSwitchWidget( value: () => UserService.instance.hasEnabledTwoFactor(), onChanged: () async { final hasAuthenticated = await LocalAuthenticationService .instance .requestLocalAuthentication( context, S.of(context).authToConfigureTwofactorAuthentication, ); final isTwoFactorEnabled = UserService.instance.hasEnabledTwoFactor(); if (hasAuthenticated) { if (isTwoFactorEnabled) { await _disableTwoFactor(); completer.isCompleted ? null : completer.complete(); } else { await UserService.instance .setupTwoFactor(context, completer); } return completer.future; } }, ), ), if (canDisableMFA) sectionOptionSpacing, if (canDisableMFA) MenuItemWidget( captionedTextWidget: CaptionedTextWidget( title: S.of(context).emailVerificationToggle, ), trailingWidget: ToggleSwitchWidget( value: () => UserService.instance.hasEmailMFAEnabled(), onChanged: () async { final hasAuthenticated = await LocalAuthenticationService.instance .requestLocalAuthentication( context, S.of(context).authToChangeEmailVerificationSetting, ); final isEmailMFAEnabled = UserService.instance.hasEmailMFAEnabled(); if (hasAuthenticated) { await updateEmailMFA(!isEmailMFAEnabled); /*if (mounted) { setState(() {}); }*/ } }, ), ), sectionOptionSpacing, ], ); } children.addAll([ MenuItemWidget( captionedTextWidget: CaptionedTextWidget( title: S.of(context).lockscreen, ), trailingWidget: ToggleSwitchWidget( value: () => _config.shouldShowLockScreen(), onChanged: () async { await LocalAuthenticationService.instance .requestLocalAuthForLockScreen( context, !_config.shouldShowLockScreen(), S.of(context).authToChangeLockscreenSetting, S.of(context).lockScreenEnablePreSteps, ); }, ), ), sectionOptionSpacing, MenuItemWidget( captionedTextWidget: CaptionedTextWidget( title: S.of(context).viewActiveSessions, ), pressedColor: getEnteColorScheme(context).fillFaint, trailingIcon: Icons.chevron_right_outlined, trailingIconIsMuted: true, showOnlyLoadingState: true, onTap: () async { final hasAuthenticated = await LocalAuthenticationService.instance .requestLocalAuthentication( context, S.of(context).authToViewYourActiveSessions, ); if (hasAuthenticated) { unawaited( Navigator.of(context).push( MaterialPageRoute( builder: (BuildContext context) { return const SessionsPage(); }, ), ), ); } }, ), sectionOptionSpacing, ]); return Column( children: children, ); } Future _disableTwoFactor() async { final AlertDialog alert = AlertDialog( title: Text(S.of(context).disableTwofactor), content: Text( S.of(context).confirm2FADisable, ), actions: [ TextButton( child: Text( S.of(context).no, style: TextStyle( color: Theme.of(context).colorScheme.greenAlternative, ), ), onPressed: () { Navigator.of(context, rootNavigator: true).pop('dialog'); }, ), TextButton( child: Text( S.of(context).yes, style: const TextStyle( color: Colors.red, ), ), onPressed: () async { await UserService.instance.disableTwoFactor(context); Navigator.of(context, rootNavigator: true).pop('dialog'); }, ), ], ); await showDialog( context: context, builder: (BuildContext context) { return alert; }, ); } Future _getOrCreateRecoveryKey(BuildContext context) async { return CryptoUtil.bin2hex( await UserService.instance.getOrCreateRecoveryKey(context), ); } Future updateEmailMFA(bool isEnabled) async { try { await UserService.instance.updateEmailMFA(isEnabled); } catch (e) { showToast(context, S.of(context).somethingWentWrong); } } }