import 'dart:async'; import 'package:awesome_extensions/awesome_extensions.dart'; import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; import 'package:flutter_translate/flutter_translate.dart'; import 'package:provider/provider.dart'; import 'package:veilid_support/veilid_support.dart'; import '../../theme/theme.dart'; import '../../tools/tools.dart'; import 'invitation_dialog.dart'; class PasteInvitationDialog extends StatefulWidget { const PasteInvitationDialog({required Locator locator, super.key}) : _locator = locator; @override PasteInvitationDialogState createState() => PasteInvitationDialogState(); static Future show(BuildContext context) async { final locator = context.read; await showPopControlDialog( context: context, builder: (context) => StyledDialog( title: translate('paste_invitation_dialog.title'), child: PasteInvitationDialog(locator: locator))); } final Locator _locator; } class PasteInvitationDialogState extends State { final _pasteTextController = TextEditingController(); @override void initState() { super.initState(); } Future _onPasteChanged( String text, Future Function({ required Uint8List inviteData, }) validateInviteData) async { final lines = text.split('\n'); if (lines.isEmpty) { return; } var firstline = lines.indexWhere((element) => element.contains('BEGIN VEILIDCHAT')); firstline += 1; var lastline = lines.indexWhere((element) => element.contains('END VEILIDCHAT')); if (lastline == -1) { lastline = lines.length; } if (lastline <= firstline) { return; } final inviteDataBase64 = lines .sublist(firstline, lastline) .join() .replaceAll(RegExp(r'[^A-Za-z0-9\-_]'), ''); var inviteData = Uint8List(0); try { inviteData = base64UrlNoPadDecode(inviteDataBase64); } on Exception { // } await validateInviteData(inviteData: inviteData); } void onValidationCancelled() { _pasteTextController.clear(); } void onValidationSuccess() { //_pasteTextController.clear(); } void onValidationFailed() { _pasteTextController.clear(); } bool inviteControlIsValid() => _pasteTextController.text.isNotEmpty; Widget buildInviteControl( BuildContext context, InvitationDialogState dialogState, Future Function({required Uint8List inviteData}) validateInviteData) { final theme = Theme.of(context); final scale = theme.extension()!; //final textTheme = theme.textTheme; //final height = MediaQuery.of(context).size.height; final monoStyle = TextStyle( fontFamily: 'Source Code Pro', fontSize: 11, color: scale.primaryScale.appText, ); return Column(mainAxisSize: MainAxisSize.min, children: [ Text( translate('paste_invitation_dialog.paste_invite_here'), ).paddingLTRB(0, 0, 0, 16), Container( constraints: const BoxConstraints(maxHeight: 200), child: TextField( enabled: !dialogState.isValidating, onChanged: (text) async => _onPasteChanged(text, validateInviteData), style: monoStyle, keyboardType: TextInputType.multiline, maxLines: null, controller: _pasteTextController, decoration: const InputDecoration( hintText: '--- BEGIN VEILIDCHAT CONTACT INVITE ----\n' 'XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX\n' '---- END VEILIDCHAT CONTACT INVITE -----\n', //labelText: translate('paste_invitation_dialog.paste') ), )).paddingLTRB(0, 0, 0, 8) ]); } @override // ignore: prefer_expression_function_bodies Widget build(BuildContext context) { return InvitationDialog( locator: widget._locator, onValidationCancelled: onValidationCancelled, onValidationSuccess: onValidationSuccess, onValidationFailed: onValidationFailed, inviteControlIsValid: inviteControlIsValid, buildInviteControl: buildInviteControl); } }