import 'dart:async'; import 'package:archive/archive.dart'; import 'package:awesome_extensions/awesome_extensions.dart'; import 'package:basic_utils/basic_utils.dart'; import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:flutter_translate/flutter_translate.dart'; import 'package:qr_flutter/qr_flutter.dart'; import '../tools/tools.dart'; import '../veilid_support/veilid_support.dart'; class ContactInvitationDisplayDialog extends ConsumerStatefulWidget { const ContactInvitationDisplayDialog({ required this.name, required this.message, required this.generator, super.key, }); final String name; final String message; final FutureOr generator; @override ContactInvitationDisplayDialogState createState() => ContactInvitationDisplayDialogState(); @override void debugFillProperties(DiagnosticPropertiesBuilder properties) { super.debugFillProperties(properties); properties ..add(StringProperty('name', name)) ..add(StringProperty('message', message)) ..add(DiagnosticsProperty?>('generator', generator)); } } class ContactInvitationDisplayDialogState extends ConsumerState { final focusNode = FocusNode(); final formKey = GlobalKey(); late final AutoDisposeFutureProvider _generateFutureProvider; @override void initState() { super.initState(); _generateFutureProvider = AutoDisposeFutureProvider((ref) async => widget.generator); } @override void dispose() { focusNode.dispose(); super.dispose(); } String makeTextInvite(String message, Uint8List data) { final invite = StringUtils.addCharAtPosition( base64UrlNoPadEncode(data), '\n', 40, repeat: true); final msg = message.isNotEmpty ? '$message\n' : ''; return '$msg' '--- BEGIN VEILIDCHAT CONTACT INVITE ----\n' '$invite\n' '---- END VEILIDCHAT CONTACT INVITE -----\n'; } @override // ignore: prefer_expression_function_bodies Widget build(BuildContext context) { final theme = Theme.of(context); final scale = theme.extension()!; final textTheme = theme.textTheme; final signedContactInvitationBytesV = ref.watch(_generateFutureProvider); final cardsize = MediaQuery.of(context).size.shortestSide - 24; return Dialog( backgroundColor: Colors.white, child: SizedBox( width: cardsize, height: cardsize, child: signedContactInvitationBytesV.when( loading: () => waitingPage(context), data: (data) { if (data == null) { Navigator.of(context).pop(); return const Text(''); } return Form( key: formKey, child: Column( mainAxisSize: MainAxisSize.min, mainAxisAlignment: MainAxisAlignment.spaceEvenly, children: [ FittedBox( fit: BoxFit.scaleDown, child: Text( translate( 'send_invite_dialog.contact_invitation'), style: textTheme.headlineMedium! .copyWith(color: Colors.black))), ConstrainedBox( constraints: const BoxConstraints( minWidth: 300, minHeight: 300), child: QrImageView.withQr( size: 300, qr: QrCode.fromUint8List( data: data, errorCorrectLevel: QrErrorCorrectLevel.L))), FittedBox( fit: BoxFit.scaleDown, child: Text(widget.message, softWrap: true, style: textTheme.headlineSmall! .copyWith(color: Colors.black))), ElevatedButton.icon( icon: const Icon(Icons.copy), label: Text(translate( 'send_invite_dialog.copy_invitation')), onPressed: () async { showInfoToast( context, translate( 'send_invite_dialog.invitation_copied')); await Clipboard.setData(ClipboardData( text: makeTextInvite(widget.message, data))); }, ), ])); }, error: (e, s) { Navigator.of(context).pop(); showErrorToast(context, translate('send_invite_dialog.failed_to_generate')); return const Text(''); }))); } @override void debugFillProperties(DiagnosticPropertiesBuilder properties) { super.debugFillProperties(properties); properties ..add(DiagnosticsProperty('focusNode', focusNode)) ..add(DiagnosticsProperty>('formKey', formKey)); } }