import 'package:flutter/material.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import '../../app/views/responsive_dialog.dart'; import '../../app/models.dart'; import '../../app/state.dart'; import '../models.dart'; import '../state.dart'; class ManagePasswordDialog extends ConsumerStatefulWidget { final DevicePath path; final OathState state; const ManagePasswordDialog(this.path, this.state, {Key? key}) : super(key: key); @override ConsumerState createState() => _ManagePasswordDialogState(); } class _ManagePasswordDialogState extends ConsumerState { String _currentPassword = ''; String _newPassword = ''; String _confirmPassword = ''; bool _currentIsWrong = false; @override Widget build(BuildContext context) { // If current device changes, we need to pop back to the main Page. ref.listen(currentDeviceProvider, (previous, next) { Navigator.of(context).pop(); }); return ResponsiveDialog( title: const Text('Manage password'), child: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ if (widget.state.hasKey) ...[ Text( 'Current password', style: Theme.of(context).textTheme.headline6, ), TextField( autofocus: true, obscureText: true, decoration: InputDecoration( border: const OutlineInputBorder(), labelText: 'Current password', errorText: _currentIsWrong ? 'Wrong password' : null), onChanged: (value) { setState(() { _currentPassword = value; }); }, ), Wrap( spacing: 8.0, children: [ OutlinedButton( child: const Text('Remove password'), onPressed: _currentPassword.isNotEmpty ? () async { final result = await ref .read(oathStateProvider(widget.path).notifier) .unsetPassword(_currentPassword); if (result) { Navigator.of(context).pop(); ScaffoldMessenger.of(context).showSnackBar( const SnackBar( content: Text('Password removed'), duration: Duration(seconds: 2), ), ); } else { setState(() { _currentIsWrong = true; }); } } : null, ), if (widget.state.remembered) OutlinedButton( child: const Text('Clear saved password'), onPressed: () async { await ref .read(oathStateProvider(widget.path).notifier) .forgetPassword(); Navigator.of(context).pop(); ScaffoldMessenger.of(context).showSnackBar( const SnackBar( content: Text('Password forgotten'), duration: Duration(seconds: 2), ), ); }, ), ], ), const Divider(), ], Text( 'New password', style: Theme.of(context).textTheme.headline6, ), TextField( autofocus: !widget.state.hasKey, obscureText: true, decoration: InputDecoration( border: const OutlineInputBorder(), labelText: 'New password', enabled: !widget.state.hasKey || _currentPassword.isNotEmpty, ), onChanged: (value) { setState(() { _newPassword = value; }); }, ), TextField( obscureText: true, decoration: InputDecoration( border: const OutlineInputBorder(), labelText: 'Confirm password', enabled: _newPassword.isNotEmpty, ), onChanged: (value) { setState(() { _confirmPassword = value; }); }, ), ] .map((e) => Padding( padding: const EdgeInsets.symmetric(vertical: 8.0), child: e, )) .toList(), ), actions: [ TextButton( onPressed: _newPassword.isNotEmpty && _newPassword == _confirmPassword && (!widget.state.hasKey || _currentPassword.isNotEmpty) ? () async { final result = await ref .read(oathStateProvider(widget.path).notifier) .setPassword(_currentPassword, _newPassword); if (result) { Navigator.of(context).pop(); ScaffoldMessenger.of(context).showSnackBar( const SnackBar( content: Text('Password set'), duration: Duration(seconds: 2), ), ); } else { setState(() { _currentIsWrong = true; }); } } : null, child: const Text('Save'), ) ], ); } } class ManagePasswordDialog2 extends ConsumerWidget { final DeviceNode device; final OathState state; const ManagePasswordDialog2(this.device, this.state, {Key? key}) : super(key: key); @override Widget build(BuildContext context, WidgetRef ref) { // If current device changes, we need to pop back to the main Page. ref.listen(currentDeviceProvider, (previous, next) { Navigator.of(context).pop(); }); return ResponsiveDialog( title: const Text('Manage password'), child: ManagePasswordDialog( device.path, state, // Prevents from losing state on responsive change. key: GlobalKey(), ), ); } }