import 'dart:async'; import 'dart:math'; import 'package:awesome_extensions/awesome_extensions.dart'; import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:flutter_translate/flutter_translate.dart'; import 'package:veilid_support/veilid_support.dart'; import '../../account_manager/account_manager.dart'; import '../../tools/tools.dart'; import 'contact_invitation_display.dart'; class SendInviteDialog extends StatefulWidget { const SendInviteDialog({super.key}); @override SendInviteDialogState createState() => SendInviteDialogState(); static Future show(BuildContext context) async { await showStyledDialog( context: context, title: translate('send_invite_dialog.title'), child: const SendInviteDialog()); } } class SendInviteDialogState extends State { final _messageTextController = TextEditingController( text: translate('send_invite_dialog.connect_with_me')); EncryptionKeyType _encryptionKeyType = EncryptionKeyType.none; String _encryptionKey = ''; Timestamp? _expiration; @override void initState() { super.initState(); } Future _onNoneEncryptionSelected(bool selected) async { setState(() { if (selected) { _encryptionKeyType = EncryptionKeyType.none; } }); } Future _onPinEncryptionSelected(bool selected) async { final description = translate('send_invite_dialog.pin_description'); final pin = await showDialog( context: context, builder: (context) => EnterPinDialog(reenter: false, description: description)); if (pin == null) { return; } // ignore: use_build_context_synchronously if (!context.mounted) { return; } final matchpin = await showDialog( context: context, builder: (context) => EnterPinDialog( reenter: true, description: description, )); if (matchpin == null) { return; } else if (pin == matchpin) { setState(() { _encryptionKeyType = EncryptionKeyType.pin; _encryptionKey = pin; }); } else { // ignore: use_build_context_synchronously if (!context.mounted) { return; } showErrorToast( context, translate('send_invite_dialog.pin_does_not_match')); setState(() { _encryptionKeyType = EncryptionKeyType.none; _encryptionKey = ''; }); } } Future _onPasswordEncryptionSelected(bool selected) async { final description = translate('send_invite_dialog.password_description'); final password = await showDialog( context: context, builder: (context) => EnterPasswordDialog(description: description)); if (password == null) { return; } // ignore: use_build_context_synchronously if (!context.mounted) { return; } final matchpass = await showDialog( context: context, builder: (context) => EnterPasswordDialog( matchPass: password, description: description, )); if (matchpass == null) { return; } else if (password == matchpass) { setState(() { _encryptionKeyType = EncryptionKeyType.password; _encryptionKey = password; }); } else { // ignore: use_build_context_synchronously if (!context.mounted) { return; } showErrorToast( context, translate('send_invite_dialog.password_does_not_match')); setState(() { _encryptionKeyType = EncryptionKeyType.none; _encryptionKey = ''; }); } } Future _onGenerateButtonPressed() async { final navigator = Navigator.of(context); // Start generation final activeAccountInfo = await AccountRepository.instance.fetchActiveAccountInfo(); if (activeAccountInfo == null) { navigator.pop(); return; } final generator = ContactInvitationRespositoryxxx.createContactInvitation( activeAccountInfo: activeAccountInfo, encryptionKeyType: _encryptionKeyType, encryptionKey: _encryptionKey, message: _messageTextController.text, expiration: _expiration); // ignore: use_build_context_synchronously if (!context.mounted) { return; } await showDialog( context: context, builder: (context) => ContactInvitationDisplayDialog( name: activeAccountInfo.localAccount.name, message: _messageTextController.text, generator: generator, )); // if (ret == null) { // return; // } ref.invalidate(fetchContactInvitationRecordsProvider); navigator.pop(); } @override // ignore: prefer_expression_function_bodies Widget build(BuildContext context) { final windowSize = MediaQuery.of(context).size; final maxDialogWidth = min(windowSize.width - 64.0, 800.0 - 64.0); final maxDialogHeight = windowSize.height - 64.0; final theme = Theme.of(context); //final scale = theme.extension()!; final textTheme = theme.textTheme; return ConstrainedBox( constraints: BoxConstraints(maxHeight: maxDialogHeight, maxWidth: maxDialogWidth), child: SingleChildScrollView( padding: const EdgeInsets.all(8), child: Column( crossAxisAlignment: CrossAxisAlignment.start, mainAxisSize: MainAxisSize.min, children: [ Text( translate('send_invite_dialog.message_to_contact'), ).paddingAll(8), TextField( controller: _messageTextController, inputFormatters: [ LengthLimitingTextInputFormatter(128), ], decoration: InputDecoration( border: const OutlineInputBorder(), hintText: translate('send_invite_dialog.enter_message_hint'), labelText: translate('send_invite_dialog.message')), ).paddingAll(8), const SizedBox(height: 10), Text(translate('send_invite_dialog.protect_this_invitation'), style: textTheme.labelLarge) .paddingAll(8), Wrap(spacing: 5, children: [ ChoiceChip( label: Text(translate('send_invite_dialog.unlocked')), selected: _encryptionKeyType == EncryptionKeyType.none, onSelected: _onNoneEncryptionSelected, ), ChoiceChip( label: Text(translate('send_invite_dialog.pin')), selected: _encryptionKeyType == EncryptionKeyType.pin, onSelected: _onPinEncryptionSelected, ), ChoiceChip( label: Text(translate('send_invite_dialog.password')), selected: _encryptionKeyType == EncryptionKeyType.password, onSelected: _onPasswordEncryptionSelected, ) ]).paddingAll(8), Container( width: double.infinity, height: 60, padding: const EdgeInsets.all(8), child: ElevatedButton( onPressed: _onGenerateButtonPressed, child: Text( translate('send_invite_dialog.generate'), ), ), ), Text(translate('send_invite_dialog.note')).paddingAll(8), Text( translate('send_invite_dialog.note_text'), style: Theme.of(context).textTheme.bodySmall, ).paddingAll(8), ], ), ), ); } @override void debugFillProperties(DiagnosticPropertiesBuilder properties) { super.debugFillProperties(properties); properties.add(DiagnosticsProperty( 'messageTextController', _messageTextController)); } }