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:provider/provider.dart'; import 'package:veilid_support/veilid_support.dart'; import '../../account_manager/account_manager.dart'; import '../../notifications/notifications.dart'; import '../../theme/theme.dart'; import '../contact_invitation.dart'; class CreateInvitationDialog extends StatefulWidget { const CreateInvitationDialog._({required this.locator}); @override CreateInvitationDialogState createState() => CreateInvitationDialogState(); static Future show(BuildContext context) async { await StyledDialog.show( context: context, title: translate('create_invitation_dialog.title'), child: CreateInvitationDialog._(locator: context.read)); } final Locator locator; @override void debugFillProperties(DiagnosticPropertiesBuilder properties) { super.debugFillProperties(properties); properties.add(DiagnosticsProperty('locator', locator)); } } class CreateInvitationDialogState extends State { late final TextEditingController _messageTextController; EncryptionKeyType _encryptionKeyType = EncryptionKeyType.none; String _encryptionKey = ''; Timestamp? _expiration; @override void initState() { final accountInfo = widget.locator().state; final name = accountInfo.asData?.value.profile.name ?? translate('create_invitation_dialog.me'); _messageTextController = TextEditingController( text: translate('create_invitation_dialog.connect_with_me', args: {'name': name})); super.initState(); } Future _onNoneEncryptionSelected(bool selected) async { setState(() { if (selected) { _encryptionKeyType = EncryptionKeyType.none; } }); } Future _onPinEncryptionSelected(bool selected) async { final description = translate('create_invitation_dialog.pin_description'); final pin = await showDialog( context: context, builder: (context) => EnterPinDialog(reenter: false, description: description)); if (pin == null) { return; } if (!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 { if (!mounted) { return; } context.read().error( text: translate('create_invitation_dialog.pin_does_not_match')); setState(() { _encryptionKeyType = EncryptionKeyType.none; _encryptionKey = ''; }); } } Future _onPasswordEncryptionSelected(bool selected) async { final description = translate('create_invitation_dialog.password_description'); final password = await showDialog( context: context, builder: (context) => EnterPasswordDialog(description: description)); if (password == null) { return; } if (!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 { if (!mounted) { return; } context.read().error( text: translate('create_invitation_dialog.password_does_not_match')); setState(() { _encryptionKeyType = EncryptionKeyType.none; _encryptionKey = ''; }); } } Future _onGenerateButtonPressed() async { final navigator = Navigator.of(context); // Start generation final contactInvitationListCubit = widget.locator(); final profile = widget.locator().state.asData?.value.profile; if (profile == null) { return; } final generator = contactInvitationListCubit.createInvitation( profile: profile, encryptionKeyType: _encryptionKeyType, encryptionKey: _encryptionKey, message: _messageTextController.text, expiration: _expiration); navigator.pop(); await ContactInvitationDisplayDialog.show( context: context, locator: widget.locator, message: _messageTextController.text, create: (context) => InvitationGeneratorCubit(generator)); } @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('create_invitation_dialog.message_to_contact'), ).paddingAll(8), TextField( controller: _messageTextController, inputFormatters: [ LengthLimitingTextInputFormatter(128), ], decoration: InputDecoration( //border: const OutlineInputBorder(), hintText: translate('create_invitation_dialog.enter_message_hint'), labelText: translate('create_invitation_dialog.message')), ).paddingAll(8), const SizedBox(height: 10), Text(translate('create_invitation_dialog.protect_this_invitation'), style: textTheme.labelLarge) .paddingAll(8), Wrap( alignment: WrapAlignment.center, runAlignment: WrapAlignment.center, runSpacing: 8, spacing: 8, children: [ ChoiceChip( label: Text(translate('create_invitation_dialog.unlocked')), selected: _encryptionKeyType == EncryptionKeyType.none, onSelected: _onNoneEncryptionSelected, ), ChoiceChip( label: Text(translate('create_invitation_dialog.pin')), selected: _encryptionKeyType == EncryptionKeyType.pin, onSelected: _onPinEncryptionSelected, ), ChoiceChip( label: Text(translate('create_invitation_dialog.password')), selected: _encryptionKeyType == EncryptionKeyType.password, onSelected: _onPasswordEncryptionSelected, ) ]).paddingAll(8).toCenter(), Container( padding: const EdgeInsets.all(8), child: ElevatedButton( onPressed: _onGenerateButtonPressed, child: Text( translate('create_invitation_dialog.generate'), ).paddingAll(16), ), ).toCenter(), Text(translate('create_invitation_dialog.note')).paddingAll(8), Text( translate('create_invitation_dialog.note_text'), style: Theme.of(context).textTheme.bodySmall, ).paddingAll(8), ], ), ), ); } @override void debugFillProperties(DiagnosticPropertiesBuilder properties) { super.debugFillProperties(properties); properties.add(DiagnosticsProperty( 'messageTextController', _messageTextController)); } }