2023-08-05 01:00:46 -04:00
|
|
|
import 'dart:async';
|
|
|
|
|
|
|
|
import 'package:awesome_extensions/awesome_extensions.dart';
|
|
|
|
import 'package:flutter/foundation.dart';
|
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
|
|
|
import 'package:flutter_translate/flutter_translate.dart';
|
|
|
|
import 'package:quickalert/quickalert.dart';
|
|
|
|
|
|
|
|
import '../entities/local_account.dart';
|
|
|
|
import '../providers/account.dart';
|
|
|
|
import '../providers/contact.dart';
|
2023-08-05 23:58:13 -04:00
|
|
|
import '../providers/contact_invite.dart';
|
2023-08-05 01:00:46 -04:00
|
|
|
import '../tools/tools.dart';
|
|
|
|
import '../veilid_support/veilid_support.dart';
|
|
|
|
import 'enter_pin.dart';
|
2023-08-05 12:38:03 -04:00
|
|
|
import 'profile_widget.dart';
|
2023-08-05 01:00:46 -04:00
|
|
|
|
|
|
|
class PasteInviteDialog extends ConsumerStatefulWidget {
|
|
|
|
const PasteInviteDialog({super.key});
|
|
|
|
|
|
|
|
@override
|
|
|
|
PasteInviteDialogState createState() => PasteInviteDialogState();
|
|
|
|
}
|
|
|
|
|
|
|
|
class PasteInviteDialogState extends ConsumerState<PasteInviteDialog> {
|
|
|
|
final _pasteTextController = TextEditingController();
|
|
|
|
|
|
|
|
EncryptionKeyType _encryptionKeyType = EncryptionKeyType.none;
|
|
|
|
String _encryptionKey = '';
|
|
|
|
Timestamp? _expiration;
|
2023-08-05 12:38:03 -04:00
|
|
|
ValidContactInvitation? _validInvitation;
|
|
|
|
bool _validatingPaste = false;
|
2023-08-05 13:50:31 -04:00
|
|
|
bool _isAccepting = false;
|
2023-08-05 01:00:46 -04:00
|
|
|
|
|
|
|
@override
|
|
|
|
void initState() {
|
|
|
|
super.initState();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Future<void> _onNoneEncryptionSelected(bool selected) async {
|
|
|
|
// setState(() {
|
|
|
|
// if (selected) {
|
|
|
|
// _encryptionKeyType = EncryptionKeyType.none;
|
|
|
|
// }
|
|
|
|
// });
|
|
|
|
// }
|
|
|
|
|
|
|
|
// Future<void> _onPinEncryptionSelected(bool selected) async {
|
|
|
|
// final description = translate('receive_invite_dialog.pin_description');
|
|
|
|
// final pin = await showDialog<String>(
|
|
|
|
// context: context,
|
|
|
|
// builder: (context) => EnterPinDialog(description: description));
|
|
|
|
// if (pin == null) {
|
|
|
|
// return;
|
|
|
|
// }
|
|
|
|
// // ignore: use_build_context_synchronously
|
|
|
|
// if (!context.mounted) {
|
|
|
|
// return;
|
|
|
|
// }
|
|
|
|
// final matchpin = await showDialog<String>(
|
|
|
|
// context: context,
|
|
|
|
// builder: (context) => EnterPinDialog(
|
|
|
|
// matchPin: pin,
|
|
|
|
// 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('receive_invite_dialog.pin_does_not_match'));
|
|
|
|
// setState(() {
|
|
|
|
// _encryptionKeyType = EncryptionKeyType.none;
|
|
|
|
// _encryptionKey = '';
|
|
|
|
// });
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
|
|
|
|
// Future<void> _onPasswordEncryptionSelected(bool selected) async {
|
|
|
|
// setState(() {
|
|
|
|
// if (selected) {
|
|
|
|
// _encryptionKeyType = EncryptionKeyType.password;
|
|
|
|
// }
|
|
|
|
// });
|
|
|
|
// }
|
|
|
|
|
2023-08-05 12:38:03 -04:00
|
|
|
Future<void> _onAccept() async {
|
2023-08-05 13:50:31 -04:00
|
|
|
final navigator = Navigator.of(context);
|
|
|
|
|
|
|
|
setState(() {
|
|
|
|
_isAccepting = true;
|
|
|
|
});
|
|
|
|
final activeAccountInfo = await ref.read(fetchActiveAccountProvider.future);
|
|
|
|
if (activeAccountInfo == null) {
|
|
|
|
setState(() {
|
|
|
|
_isAccepting = false;
|
|
|
|
});
|
|
|
|
navigator.pop();
|
|
|
|
return;
|
2023-08-05 12:38:03 -04:00
|
|
|
}
|
2023-08-05 13:50:31 -04:00
|
|
|
final validInvitation = _validInvitation;
|
|
|
|
if (validInvitation != null) {
|
2023-08-05 21:01:27 -04:00
|
|
|
final acceptedContact =
|
|
|
|
await acceptContactInvitation(activeAccountInfo, validInvitation);
|
|
|
|
if (acceptedContact != null) {
|
|
|
|
await createContact(
|
|
|
|
activeAccountInfo: activeAccountInfo,
|
|
|
|
profile: acceptedContact.profile,
|
|
|
|
remoteIdentity: acceptedContact.remoteIdentity,
|
2023-08-05 23:58:13 -04:00
|
|
|
remoteConversationKey: acceptedContact.remoteConversationKey,
|
2023-08-05 21:01:27 -04:00
|
|
|
localConversation: acceptedContact.localConversation,
|
|
|
|
);
|
|
|
|
ref
|
|
|
|
..invalidate(fetchContactInvitationRecordsProvider)
|
|
|
|
..invalidate(fetchContactListProvider);
|
|
|
|
} else {
|
|
|
|
if (context.mounted) {
|
|
|
|
showErrorToast(context, 'paste_invite_dialog.failed_to_accept');
|
|
|
|
}
|
|
|
|
}
|
2023-08-05 13:50:31 -04:00
|
|
|
}
|
|
|
|
setState(() {
|
|
|
|
_isAccepting = false;
|
|
|
|
});
|
|
|
|
navigator.pop();
|
2023-08-05 12:38:03 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> _onReject() async {
|
2023-08-05 13:50:31 -04:00
|
|
|
final navigator = Navigator.of(context);
|
|
|
|
|
|
|
|
setState(() {
|
|
|
|
_isAccepting = true;
|
|
|
|
});
|
|
|
|
final activeAccountInfo = await ref.read(fetchActiveAccountProvider.future);
|
|
|
|
if (activeAccountInfo == null) {
|
|
|
|
setState(() {
|
|
|
|
_isAccepting = false;
|
|
|
|
});
|
|
|
|
navigator.pop();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
final validInvitation = _validInvitation;
|
|
|
|
if (validInvitation != null) {
|
2023-08-05 21:01:27 -04:00
|
|
|
if (await rejectContactInvitation(activeAccountInfo, validInvitation)) {
|
|
|
|
// do nothing right now
|
|
|
|
} else {
|
|
|
|
if (context.mounted) {
|
|
|
|
showErrorToast(context, 'paste_invite_dialog.failed_to_reject');
|
|
|
|
}
|
|
|
|
}
|
2023-08-05 12:38:03 -04:00
|
|
|
}
|
2023-08-05 13:50:31 -04:00
|
|
|
setState(() {
|
|
|
|
_isAccepting = false;
|
|
|
|
});
|
|
|
|
navigator.pop();
|
2023-08-05 12:38:03 -04:00
|
|
|
}
|
|
|
|
|
2023-08-05 01:00:46 -04:00
|
|
|
Future<void> _onPasteChanged(String text) async {
|
|
|
|
try {
|
|
|
|
final lines = text.split('\n');
|
|
|
|
if (lines.isEmpty) {
|
2023-08-05 12:38:03 -04:00
|
|
|
setState(() {
|
|
|
|
_validatingPaste = false;
|
|
|
|
_validInvitation = null;
|
|
|
|
});
|
2023-08-05 01:00:46 -04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
var firstline =
|
|
|
|
lines.indexWhere((element) => element.contains('BEGIN VEILIDCHAT'));
|
|
|
|
firstline += 1;
|
|
|
|
|
|
|
|
var lastline =
|
|
|
|
lines.indexWhere((element) => element.contains('END VEILIDCHAT'));
|
2023-08-05 01:15:08 -04:00
|
|
|
if (lastline == -1) {
|
2023-08-05 01:00:46 -04:00
|
|
|
lastline = lines.length;
|
|
|
|
}
|
|
|
|
if (lastline <= firstline) {
|
2023-08-05 12:38:03 -04:00
|
|
|
setState(() {
|
|
|
|
_validatingPaste = false;
|
|
|
|
_validInvitation = null;
|
|
|
|
});
|
2023-08-05 01:00:46 -04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
final inviteDataBase64 = lines.sublist(firstline, lastline).join();
|
|
|
|
final inviteData = base64UrlNoPadDecode(inviteDataBase64);
|
2023-08-05 01:15:08 -04:00
|
|
|
|
2023-08-05 12:38:03 -04:00
|
|
|
setState(() {
|
|
|
|
_validatingPaste = true;
|
|
|
|
_validInvitation = null;
|
|
|
|
});
|
|
|
|
final validatedContactInvitation = await validateContactInvitation(
|
|
|
|
inviteData, (encryptionKeyType, encryptedSecret) async {
|
2023-08-05 01:00:46 -04:00
|
|
|
switch (encryptionKeyType) {
|
|
|
|
case EncryptionKeyType.none:
|
2023-08-05 12:38:03 -04:00
|
|
|
return SecretKey.fromBytes(encryptedSecret);
|
2023-08-05 01:00:46 -04:00
|
|
|
case EncryptionKeyType.pin:
|
2023-08-05 12:38:03 -04:00
|
|
|
//xxx
|
|
|
|
return SecretKey.fromBytes(encryptedSecret);
|
2023-08-05 01:00:46 -04:00
|
|
|
case EncryptionKeyType.password:
|
2023-08-05 12:38:03 -04:00
|
|
|
//xxx
|
|
|
|
return SecretKey.fromBytes(encryptedSecret);
|
2023-08-05 01:00:46 -04:00
|
|
|
}
|
2023-08-05 12:38:03 -04:00
|
|
|
});
|
|
|
|
// Verify expiration
|
|
|
|
// xxx
|
2023-08-05 01:00:46 -04:00
|
|
|
|
2023-08-05 12:38:03 -04:00
|
|
|
setState(() {
|
|
|
|
_validatingPaste = false;
|
|
|
|
_validInvitation = validatedContactInvitation;
|
2023-08-05 01:00:46 -04:00
|
|
|
});
|
|
|
|
} on Exception catch (_) {
|
2023-08-05 12:38:03 -04:00
|
|
|
setState(() {
|
|
|
|
_validatingPaste = false;
|
|
|
|
_validInvitation = null;
|
|
|
|
});
|
2023-08-05 01:00:46 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
// ignore: prefer_expression_function_bodies
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
final theme = Theme.of(context);
|
|
|
|
//final scale = theme.extension<ScaleScheme>()!;
|
|
|
|
final textTheme = theme.textTheme;
|
2023-08-05 12:38:03 -04:00
|
|
|
//final height = MediaQuery.of(context).size.height;
|
2023-08-05 01:00:46 -04:00
|
|
|
|
2023-08-05 13:50:31 -04:00
|
|
|
if (_isAccepting) {
|
|
|
|
return SizedBox(height: 400, child: waitingPage(context));
|
|
|
|
}
|
2023-08-05 14:31:32 -04:00
|
|
|
return ConstrainedBox(
|
|
|
|
constraints: const BoxConstraints(maxHeight: 400),
|
2023-08-05 01:00:46 -04:00
|
|
|
child: SingleChildScrollView(
|
|
|
|
padding: const EdgeInsets.all(8),
|
|
|
|
child: Column(
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
|
|
mainAxisSize: MainAxisSize.min,
|
|
|
|
children: <Widget>[
|
|
|
|
Text(
|
|
|
|
translate('paste_invite_dialog.paste_invite_here'),
|
|
|
|
).paddingAll(8),
|
|
|
|
Container(
|
|
|
|
constraints: const BoxConstraints(maxHeight: 200),
|
|
|
|
child: TextField(
|
2023-08-05 12:38:03 -04:00
|
|
|
enabled: !_validatingPaste,
|
2023-08-05 01:00:46 -04:00
|
|
|
onChanged: _onPasteChanged,
|
|
|
|
style: textTheme.labelSmall!
|
|
|
|
.copyWith(fontFamily: 'Victor Mono', fontSize: 11),
|
|
|
|
keyboardType: TextInputType.multiline,
|
|
|
|
maxLines: null,
|
|
|
|
controller: _pasteTextController,
|
|
|
|
decoration: const InputDecoration(
|
|
|
|
border: OutlineInputBorder(),
|
|
|
|
hintText: '--- BEGIN VEILIDCHAT CONTACT INVITE ----\n'
|
|
|
|
'XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX\n'
|
|
|
|
'---- END VEILIDCHAT CONTACT INVITE -----\n',
|
|
|
|
//labelText: translate('paste_invite_dialog.paste')
|
|
|
|
),
|
|
|
|
).paddingAll(8)),
|
2023-08-05 13:50:31 -04:00
|
|
|
if (_validatingPaste)
|
|
|
|
Column(children: [
|
|
|
|
Text(translate('paste_invite_dialog.validating')),
|
|
|
|
buildProgressIndicator(context),
|
|
|
|
]),
|
2023-08-05 14:31:32 -04:00
|
|
|
if (_validInvitation == null &&
|
|
|
|
!_validatingPaste &&
|
|
|
|
_pasteTextController.text.isNotEmpty)
|
|
|
|
Column(children: [
|
|
|
|
Text(translate('paste_invite_dialog.invalid_invitation')),
|
|
|
|
const Icon(Icons.error)
|
|
|
|
]).paddingAll(16).toCenter(),
|
2023-08-05 12:38:03 -04:00
|
|
|
if (_validInvitation != null && !_validatingPaste)
|
|
|
|
Column(children: [
|
|
|
|
ProfileWidget(
|
|
|
|
name: _validInvitation!.contactRequestPrivate.profile.name,
|
|
|
|
title:
|
|
|
|
_validInvitation!.contactRequestPrivate.profile.title),
|
|
|
|
Row(
|
|
|
|
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
|
|
|
|
children: [
|
|
|
|
ElevatedButton.icon(
|
|
|
|
icon: const Icon(Icons.check_circle),
|
|
|
|
label: Text(translate('button.accept')),
|
|
|
|
onPressed: _onAccept,
|
|
|
|
),
|
|
|
|
ElevatedButton.icon(
|
|
|
|
icon: const Icon(Icons.cancel),
|
|
|
|
label: Text(translate('button.reject')),
|
|
|
|
onPressed: _onReject,
|
|
|
|
)
|
|
|
|
],
|
|
|
|
),
|
|
|
|
])
|
2023-08-05 01:00:46 -04:00
|
|
|
],
|
|
|
|
),
|
|
|
|
),
|
2023-08-05 14:31:32 -04:00
|
|
|
);
|
2023-08-05 01:00:46 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void debugFillProperties(DiagnosticPropertiesBuilder properties) {
|
|
|
|
super.debugFillProperties(properties);
|
|
|
|
}
|
|
|
|
}
|