import 'dart:io'; import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:flutter_gen/gen_l10n/app_localizations.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import '../../app/message.dart'; import '../../app/models.dart'; import '../../app/shortcuts.dart'; import '../../app/views/app_failure_page.dart'; import '../../app/views/app_loading_screen.dart'; import '../../app/views/app_page.dart'; import '../../app/views/graphics.dart'; import '../../app/views/message_page.dart'; import '../../widgets/menu_list_tile.dart'; import '../models.dart'; import '../state.dart'; import 'account_list.dart'; import 'add_account_page.dart'; import 'manage_password_dialog.dart'; import 'reset_dialog.dart'; class OathScreen extends ConsumerWidget { final DevicePath devicePath; const OathScreen(this.devicePath, {super.key}); @override Widget build(BuildContext context, WidgetRef ref) { return ref.watch(oathStateProvider(devicePath)).when( loading: () => AppPage( title: Text(AppLocalizations.of(context)!.oath_authenticator), centered: true, child: const AppLoadingScreen(), ), error: (error, _) => AppFailurePage( title: Text(AppLocalizations.of(context)!.oath_authenticator), cause: error, ), data: (oathState) => oathState.locked ? _LockedView(devicePath, oathState) : _UnlockedView(devicePath, oathState), ); } } class _LockedView extends ConsumerWidget { final DevicePath devicePath; final OathState oathState; const _LockedView(this.devicePath, this.oathState); @override Widget build(BuildContext context, WidgetRef ref) => AppPage( title: Text(AppLocalizations.of(context)!.oath_authenticator), keyActions: [ buildMenuItem( title: Text(AppLocalizations.of(context)!.oath_manage_password), leading: const Icon(Icons.password), action: () { showBlurDialog( context: context, builder: (context) => ManagePasswordDialog(devicePath, oathState), ); }, ), buildMenuItem( title: Text(AppLocalizations.of(context)!.oath_reset_oath), leading: const Icon(Icons.delete), action: () { showBlurDialog( context: context, builder: (context) => ResetDialog(devicePath), ); }, ), ], child: Column( children: [ _UnlockForm( devicePath, keystore: oathState.keystore, ), ], ), ); } class _UnlockedView extends ConsumerStatefulWidget { final DevicePath devicePath; final OathState oathState; const _UnlockedView(this.devicePath, this.oathState); @override ConsumerState createState() => _UnlockedViewState(); } class _UnlockedViewState extends ConsumerState<_UnlockedView> { late FocusNode searchFocus; late TextEditingController searchController; @override void initState() { super.initState(); searchFocus = FocusNode(); searchController = TextEditingController(text: ref.read(searchProvider)); } @override void dispose() { searchFocus.dispose(); searchController.dispose(); super.dispose(); } @override Widget build(BuildContext context) { final credentials = ref.watch(credentialsProvider); if (credentials?.isEmpty == true) { return MessagePage( title: Text(AppLocalizations.of(context)!.oath_authenticator), graphic: noAccounts, header: AppLocalizations.of(context)!.oath_no_accounts, keyActions: _buildActions( context, credentials: null, ), ); } return Actions( actions: { SearchIntent: CallbackAction(onInvoke: (_) { searchController.selection = TextSelection( baseOffset: 0, extentOffset: searchController.text.length); searchFocus.requestFocus(); return null; }), }, child: AppPage( title: Focus( canRequestFocus: false, onKeyEvent: (node, event) { if (event.logicalKey == LogicalKeyboardKey.arrowDown) { node.focusInDirection(TraversalDirection.down); return KeyEventResult.handled; } return KeyEventResult.ignored; }, child: Builder(builder: (context) { final textTheme = Theme.of(context).textTheme; return TextFormField( key: const Key('search_accounts'), controller: searchController, focusNode: searchFocus, // Use the default style, but with a smaller font size: style: textTheme.subtitle1 ?.copyWith(fontSize: textTheme.titleSmall?.fontSize), decoration: InputDecoration( hintText: AppLocalizations.of(context)!.oath_search_accounts, isDense: true, prefixIcon: const Icon(Icons.search_outlined), prefixIconConstraints: const BoxConstraints( minHeight: 30, minWidth: 30, ), border: InputBorder.none, ), onChanged: (value) { ref.read(searchProvider.notifier).setFilter(value); }, textInputAction: TextInputAction.next, onFieldSubmitted: (value) { Focus.of(context).focusInDirection(TraversalDirection.down); }, ); }), ), keyActions: _buildActions( context, credentials: credentials, ), child: AccountList(widget.devicePath, widget.oathState), ), ); } List _buildActions( BuildContext context, { required List? credentials, }) { final used = credentials?.length ?? 0; final capacity = widget.oathState.version.isAtLeast(4) ? 32 : null; return [ buildMenuItem( title: Text(AppLocalizations.of(context)!.oath_add_account, key: const Key('add oath account'),), leading: const Icon(Icons.person_add_alt_1), trailing: capacity != null ? '$used/$capacity' : null, action: capacity == null || capacity > used ? () { showBlurDialog( context: context, builder: (context) => OathAddAccountPage( widget.devicePath, widget.oathState, credentials: credentials, openQrScanner: Platform.isAndroid, ), ); } : null, ), buildMenuItem( title: Text(widget.oathState.hasKey ? AppLocalizations.of(context)!.oath_manage_password : AppLocalizations.of(context)!.oath_set_password, key: const Key('set or manage oath password')), leading: const Icon(Icons.password), action: () { showBlurDialog( context: context, builder: (context) => ManagePasswordDialog(widget.devicePath, widget.oathState), ); }), buildMenuItem( title: Text(AppLocalizations.of(context)!.oath_reset_oath, key: const Key('reset oath app')), leading: const Icon(Icons.delete), action: () { showBlurDialog( context: context, builder: (context) => ResetDialog(widget.devicePath), ); }), ]; } } class _UnlockForm extends ConsumerStatefulWidget { final DevicePath _devicePath; final KeystoreState keystore; const _UnlockForm(this._devicePath, {required this.keystore}); @override ConsumerState<_UnlockForm> createState() => _UnlockFormState(); } class _UnlockFormState extends ConsumerState<_UnlockForm> { final _passwordController = TextEditingController(); bool _remember = false; bool _passwordIsWrong = false; bool _isObscure = true; void _submit() async { setState(() { _passwordIsWrong = false; }); final result = await ref .read(oathStateProvider(widget._devicePath).notifier) .unlock(_passwordController.text, remember: _remember); if (!mounted) return; if (!result.first) { setState(() { _passwordIsWrong = true; _passwordController.clear(); }); } else if (_remember && !result.second) { showMessage( context, AppLocalizations.of(context)!.oath_failed_remember_pw); } } @override Widget build(BuildContext context) { final keystoreFailed = widget.keystore == KeystoreState.failed; return Column( children: [ Padding( padding: const EdgeInsets.only(left: 18.0, right: 18, top: 32), child: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Text( AppLocalizations.of(context)!.oath_enter_oath_pw, ), const SizedBox(height: 16.0), TextField( key: const Key('oath password'), controller: _passwordController, autofocus: true, obscureText: _isObscure, decoration: InputDecoration( border: const OutlineInputBorder(), labelText: AppLocalizations.of(context)!.oath_password, errorText: _passwordIsWrong ? AppLocalizations.of(context)!.oath_wrong_password : null, helperText: '', // Prevents resizing when errorText shown prefixIcon: const Icon(Icons.password_outlined), suffixIcon: IconButton( icon: Icon( _isObscure ? Icons.visibility : Icons.visibility_off, color: IconTheme.of(context).color, ), onPressed: () { setState(() { _isObscure = !_isObscure; }); }, ), ), onChanged: (_) => setState(() { _passwordIsWrong = false; }), // Update state on change onSubmitted: (_) => _submit(), ), ], ), ), const SizedBox(height: 8.0), keystoreFailed ? ListTile( leading: const Icon(Icons.warning_amber_rounded), title: Text( AppLocalizations.of(context)!.oath_keystore_unavailable), dense: true, minLeadingWidth: 0, ) : CheckboxListTile( title: Text(AppLocalizations.of(context)!.oath_remember_password), dense: true, controlAffinity: ListTileControlAffinity.leading, value: _remember, onChanged: (value) { setState(() { _remember = value ?? false; }); }, ), Padding( padding: const EdgeInsets.only(top: 12.0, right: 18.0, bottom: 4.0), child: Align( alignment: Alignment.centerRight, child: ElevatedButton.icon( key: const Key('oath unlock'), label: Text(AppLocalizations.of(context)!.oath_unlock), icon: const Icon(Icons.lock_open), onPressed: _passwordController.text.isNotEmpty ? _submit : null, ), ), ), ], ); } }