mirror of
https://github.com/Yubico/yubioath-flutter.git
synced 2024-12-24 10:42:47 +03:00
143 lines
4.7 KiB
Dart
143 lines
4.7 KiB
Dart
|
import 'package:flutter/material.dart';
|
||
|
import 'package:flutter_gen/gen_l10n/app_localizations.dart';
|
||
|
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
||
|
|
||
|
import '../../app/message.dart';
|
||
|
import '../../app/shortcuts.dart';
|
||
|
import '../../app/state.dart';
|
||
|
import '../../app/views/fs_dialog.dart';
|
||
|
import '../../widgets/list_title.dart';
|
||
|
import '../models.dart';
|
||
|
import 'delete_fingerprint_dialog.dart';
|
||
|
import 'rename_fingerprint_dialog.dart';
|
||
|
|
||
|
class FingerprintDialog extends ConsumerWidget {
|
||
|
final Fingerprint fingerprint;
|
||
|
const FingerprintDialog(this.fingerprint, {super.key});
|
||
|
|
||
|
@override
|
||
|
Widget build(BuildContext context, WidgetRef ref) {
|
||
|
// TODO: Solve this in a cleaner way
|
||
|
final node = ref.watch(currentDeviceDataProvider).valueOrNull?.node;
|
||
|
if (node == null) {
|
||
|
// The rest of this method assumes there is a device, and will throw an exception if not.
|
||
|
// This will never be shown, as the dialog will be immediately closed
|
||
|
return const SizedBox();
|
||
|
}
|
||
|
|
||
|
return Actions(
|
||
|
actions: {
|
||
|
EditIntent: CallbackAction<EditIntent>(onInvoke: (_) async {
|
||
|
final withContext = ref.read(withContextProvider);
|
||
|
final Fingerprint? renamed =
|
||
|
await withContext((context) async => await showBlurDialog(
|
||
|
context: context,
|
||
|
builder: (context) => RenameFingerprintDialog(
|
||
|
node.path,
|
||
|
fingerprint,
|
||
|
),
|
||
|
));
|
||
|
if (renamed != null) {
|
||
|
// Replace the dialog with the renamed credential
|
||
|
await withContext((context) async {
|
||
|
Navigator.of(context).pop();
|
||
|
await showBlurDialog(
|
||
|
context: context,
|
||
|
builder: (context) {
|
||
|
return FingerprintDialog(renamed);
|
||
|
},
|
||
|
);
|
||
|
});
|
||
|
}
|
||
|
return renamed;
|
||
|
}),
|
||
|
DeleteIntent: CallbackAction<DeleteIntent>(onInvoke: (_) async {
|
||
|
final withContext = ref.read(withContextProvider);
|
||
|
final bool? deleted =
|
||
|
await ref.read(withContextProvider)((context) async =>
|
||
|
await showBlurDialog(
|
||
|
context: context,
|
||
|
builder: (context) => DeleteFingerprintDialog(
|
||
|
node.path,
|
||
|
fingerprint,
|
||
|
),
|
||
|
) ??
|
||
|
false);
|
||
|
|
||
|
// Pop the account dialog if deleted
|
||
|
if (deleted == true) {
|
||
|
await withContext((context) async {
|
||
|
Navigator.of(context).pop();
|
||
|
});
|
||
|
}
|
||
|
return deleted;
|
||
|
}),
|
||
|
},
|
||
|
child: FocusScope(
|
||
|
autofocus: true,
|
||
|
child: FsDialog(
|
||
|
child: Column(
|
||
|
children: [
|
||
|
Padding(
|
||
|
padding: const EdgeInsets.only(top: 48, bottom: 32),
|
||
|
child: Column(
|
||
|
children: [
|
||
|
Text(
|
||
|
fingerprint.label,
|
||
|
style: Theme.of(context).textTheme.headlineSmall,
|
||
|
softWrap: true,
|
||
|
textAlign: TextAlign.center,
|
||
|
),
|
||
|
const SizedBox(height: 16),
|
||
|
const Icon(Icons.fingerprint, size: 72),
|
||
|
],
|
||
|
),
|
||
|
),
|
||
|
ListTitle(AppLocalizations.of(context)!.s_actions,
|
||
|
textStyle: Theme.of(context).textTheme.bodyLarge),
|
||
|
_FingerprintDialogActions(),
|
||
|
],
|
||
|
),
|
||
|
),
|
||
|
),
|
||
|
);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
class _FingerprintDialogActions extends StatelessWidget {
|
||
|
@override
|
||
|
Widget build(BuildContext context) {
|
||
|
final l10n = AppLocalizations.of(context)!;
|
||
|
final theme =
|
||
|
ButtonTheme.of(context).colorScheme ?? Theme.of(context).colorScheme;
|
||
|
return Column(
|
||
|
children: [
|
||
|
ListTile(
|
||
|
leading: CircleAvatar(
|
||
|
backgroundColor: theme.secondary,
|
||
|
foregroundColor: theme.onSecondary,
|
||
|
child: const Icon(Icons.edit),
|
||
|
),
|
||
|
title: Text(l10n.s_rename_fp),
|
||
|
subtitle: Text(l10n.l_rename_fp_desc),
|
||
|
onTap: () {
|
||
|
Actions.invoke(context, const EditIntent());
|
||
|
},
|
||
|
),
|
||
|
ListTile(
|
||
|
leading: CircleAvatar(
|
||
|
backgroundColor: theme.error,
|
||
|
foregroundColor: theme.onError,
|
||
|
child: const Icon(Icons.delete),
|
||
|
),
|
||
|
title: Text(l10n.s_delete_fingerprint),
|
||
|
subtitle: Text(l10n.l_delete_fingerprint_desc),
|
||
|
onTap: () {
|
||
|
Actions.invoke(context, const DeleteIntent());
|
||
|
},
|
||
|
),
|
||
|
],
|
||
|
);
|
||
|
}
|
||
|
}
|