2023-09-27 13:34:19 -04:00
|
|
|
import 'dart:async';
|
|
|
|
|
|
|
|
import 'package:awesome_extensions/awesome_extensions.dart';
|
|
|
|
import 'package:flutter/foundation.dart';
|
|
|
|
import 'package:flutter/material.dart';
|
2024-01-30 14:14:11 -05:00
|
|
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
2023-09-27 13:34:19 -04:00
|
|
|
import 'package:flutter_translate/flutter_translate.dart';
|
|
|
|
|
2024-01-09 20:58:27 -05:00
|
|
|
import '../../account_manager/account_manager.dart';
|
2024-01-30 19:47:22 -05:00
|
|
|
import '../../contacts/contacts.dart';
|
2024-01-09 20:58:27 -05:00
|
|
|
import '../../tools/tools.dart';
|
2024-01-30 14:14:11 -05:00
|
|
|
import '../contact_invitation.dart';
|
2023-09-27 13:34:19 -04:00
|
|
|
|
2024-01-09 20:58:27 -05:00
|
|
|
class InviteDialog extends StatefulWidget {
|
2023-09-27 13:34:19 -04:00
|
|
|
const InviteDialog(
|
2024-02-14 21:33:15 -05:00
|
|
|
{required this.modalContext,
|
|
|
|
required this.onValidationCancelled,
|
2023-09-27 13:34:19 -04:00
|
|
|
required this.onValidationSuccess,
|
|
|
|
required this.onValidationFailed,
|
|
|
|
required this.inviteControlIsValid,
|
|
|
|
required this.buildInviteControl,
|
|
|
|
super.key});
|
|
|
|
|
|
|
|
final void Function() onValidationCancelled;
|
|
|
|
final void Function() onValidationSuccess;
|
|
|
|
final void Function() onValidationFailed;
|
|
|
|
final bool Function() inviteControlIsValid;
|
|
|
|
final Widget Function(
|
|
|
|
BuildContext context,
|
|
|
|
InviteDialogState dialogState,
|
|
|
|
Future<void> Function({required Uint8List inviteData})
|
|
|
|
validateInviteData) buildInviteControl;
|
2024-02-14 21:33:15 -05:00
|
|
|
final BuildContext modalContext;
|
2023-09-27 13:34:19 -04:00
|
|
|
|
|
|
|
@override
|
|
|
|
InviteDialogState createState() => InviteDialogState();
|
|
|
|
@override
|
|
|
|
void debugFillProperties(DiagnosticPropertiesBuilder properties) {
|
|
|
|
super.debugFillProperties(properties);
|
|
|
|
properties
|
|
|
|
..add(ObjectFlagProperty<void Function()>.has(
|
|
|
|
'onValidationCancelled', onValidationCancelled))
|
|
|
|
..add(ObjectFlagProperty<void Function()>.has(
|
|
|
|
'onValidationSuccess', onValidationSuccess))
|
|
|
|
..add(ObjectFlagProperty<void Function()>.has(
|
|
|
|
'onValidationFailed', onValidationFailed))
|
|
|
|
..add(ObjectFlagProperty<void Function()>.has(
|
|
|
|
'inviteControlIsValid', inviteControlIsValid))
|
|
|
|
..add(ObjectFlagProperty<
|
|
|
|
Widget Function(
|
|
|
|
BuildContext context,
|
|
|
|
InviteDialogState dialogState,
|
|
|
|
Future<void> Function({required Uint8List inviteData})
|
|
|
|
validateInviteData)>.has(
|
2024-02-14 21:33:15 -05:00
|
|
|
'buildInviteControl', buildInviteControl))
|
|
|
|
..add(DiagnosticsProperty<BuildContext>('modalContext', modalContext));
|
2023-09-27 13:34:19 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-09 20:58:27 -05:00
|
|
|
class InviteDialogState extends State<InviteDialog> {
|
2023-09-27 13:34:19 -04:00
|
|
|
ValidContactInvitation? _validInvitation;
|
|
|
|
bool _isValidating = false;
|
|
|
|
bool _isAccepting = false;
|
|
|
|
|
|
|
|
@override
|
|
|
|
void initState() {
|
|
|
|
super.initState();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool get isValidating => _isValidating;
|
|
|
|
bool get isAccepting => _isAccepting;
|
|
|
|
|
|
|
|
Future<void> _onAccept() async {
|
|
|
|
final navigator = Navigator.of(context);
|
2024-02-14 21:33:15 -05:00
|
|
|
final activeAccountInfo = widget.modalContext.read<ActiveAccountInfo>();
|
|
|
|
final contactList = widget.modalContext.read<ContactListCubit>();
|
2023-09-27 13:34:19 -04:00
|
|
|
|
|
|
|
setState(() {
|
|
|
|
_isAccepting = true;
|
|
|
|
});
|
|
|
|
final validInvitation = _validInvitation;
|
|
|
|
if (validInvitation != null) {
|
2024-01-30 14:14:11 -05:00
|
|
|
final acceptedContact = await validInvitation.accept();
|
2023-09-27 13:34:19 -04:00
|
|
|
if (acceptedContact != null) {
|
|
|
|
// initiator when accept is received will create
|
|
|
|
// contact in the case of a 'note to self'
|
|
|
|
final isSelf =
|
|
|
|
activeAccountInfo.localAccount.identityMaster.identityPublicKey ==
|
|
|
|
acceptedContact.remoteIdentity.identityPublicKey;
|
|
|
|
if (!isSelf) {
|
2024-01-30 19:47:22 -05:00
|
|
|
await contactList.createContact(
|
|
|
|
remoteProfile: acceptedContact.remoteProfile,
|
2023-09-27 13:34:19 -04:00
|
|
|
remoteIdentity: acceptedContact.remoteIdentity,
|
|
|
|
remoteConversationRecordKey:
|
|
|
|
acceptedContact.remoteConversationRecordKey,
|
|
|
|
localConversationRecordKey:
|
|
|
|
acceptedContact.localConversationRecordKey,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (context.mounted) {
|
|
|
|
showErrorToast(context, 'invite_dialog.failed_to_accept');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
setState(() {
|
|
|
|
_isAccepting = false;
|
|
|
|
});
|
|
|
|
navigator.pop();
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> _onReject() async {
|
|
|
|
final navigator = Navigator.of(context);
|
|
|
|
|
|
|
|
setState(() {
|
|
|
|
_isAccepting = true;
|
|
|
|
});
|
|
|
|
final validInvitation = _validInvitation;
|
|
|
|
if (validInvitation != null) {
|
2024-01-30 14:14:11 -05:00
|
|
|
if (await validInvitation.reject()) {
|
2023-09-27 13:34:19 -04:00
|
|
|
// do nothing right now
|
|
|
|
} else {
|
|
|
|
if (context.mounted) {
|
|
|
|
showErrorToast(context, 'invite_dialog.failed_to_reject');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
setState(() {
|
|
|
|
_isAccepting = false;
|
|
|
|
});
|
|
|
|
navigator.pop();
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> _validateInviteData({
|
|
|
|
required Uint8List inviteData,
|
|
|
|
}) async {
|
|
|
|
try {
|
2024-01-30 14:14:11 -05:00
|
|
|
final contactInvitationListCubit =
|
2024-02-14 21:33:15 -05:00
|
|
|
widget.modalContext.read<ContactInvitationListCubit>();
|
2023-09-27 13:34:19 -04:00
|
|
|
|
|
|
|
setState(() {
|
|
|
|
_isValidating = true;
|
|
|
|
_validInvitation = null;
|
|
|
|
});
|
2024-01-30 14:14:11 -05:00
|
|
|
final validatedContactInvitation =
|
|
|
|
await contactInvitationListCubit.validateInvitation(
|
|
|
|
inviteData: inviteData,
|
|
|
|
getEncryptionKeyCallback:
|
|
|
|
(cs, encryptionKeyType, encryptedSecret) async {
|
|
|
|
String encryptionKey;
|
|
|
|
switch (encryptionKeyType) {
|
|
|
|
case EncryptionKeyType.none:
|
|
|
|
encryptionKey = '';
|
|
|
|
case EncryptionKeyType.pin:
|
|
|
|
final description =
|
|
|
|
translate('invite_dialog.protected_with_pin');
|
|
|
|
if (!context.mounted) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
final pin = await showDialog<String>(
|
|
|
|
context: context,
|
|
|
|
builder: (context) => EnterPinDialog(
|
|
|
|
reenter: false, description: description));
|
|
|
|
if (pin == null) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
encryptionKey = pin;
|
|
|
|
case EncryptionKeyType.password:
|
|
|
|
final description =
|
|
|
|
translate('invite_dialog.protected_with_password');
|
|
|
|
if (!context.mounted) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
final password = await showDialog<String>(
|
|
|
|
context: context,
|
|
|
|
builder: (context) =>
|
|
|
|
EnterPasswordDialog(description: description));
|
|
|
|
if (password == null) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
encryptionKey = password;
|
2023-09-27 13:34:19 -04:00
|
|
|
}
|
2024-01-30 14:14:11 -05:00
|
|
|
return encryptionKeyType.decryptSecretFromBytes(
|
|
|
|
secretBytes: encryptedSecret,
|
|
|
|
cryptoKind: cs.kind(),
|
|
|
|
encryptionKey: encryptionKey);
|
|
|
|
});
|
2023-09-27 13:34:19 -04:00
|
|
|
|
|
|
|
// Check if validation was cancelled
|
|
|
|
if (validatedContactInvitation == null) {
|
|
|
|
setState(() {
|
|
|
|
_isValidating = false;
|
|
|
|
_validInvitation = null;
|
|
|
|
widget.onValidationCancelled();
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Verify expiration
|
|
|
|
// xxx
|
|
|
|
|
|
|
|
setState(() {
|
|
|
|
widget.onValidationSuccess();
|
|
|
|
_isValidating = false;
|
|
|
|
_validInvitation = validatedContactInvitation;
|
|
|
|
});
|
|
|
|
} on ContactInviteInvalidKeyException catch (e) {
|
|
|
|
String errorText;
|
|
|
|
switch (e.type) {
|
|
|
|
case EncryptionKeyType.none:
|
|
|
|
errorText = translate('invite_dialog.invalid_invitation');
|
|
|
|
case EncryptionKeyType.pin:
|
|
|
|
errorText = translate('invite_dialog.invalid_pin');
|
|
|
|
case EncryptionKeyType.password:
|
|
|
|
errorText = translate('invite_dialog.invalid_password');
|
|
|
|
}
|
|
|
|
if (context.mounted) {
|
|
|
|
showErrorToast(context, errorText);
|
|
|
|
}
|
|
|
|
setState(() {
|
|
|
|
_isValidating = false;
|
|
|
|
_validInvitation = null;
|
|
|
|
widget.onValidationFailed();
|
|
|
|
});
|
2023-09-28 12:51:44 -04:00
|
|
|
} on Exception catch (e) {
|
|
|
|
log.debug('exception: $e', e);
|
2023-09-27 13:34:19 -04:00
|
|
|
setState(() {
|
|
|
|
_isValidating = false;
|
|
|
|
_validInvitation = null;
|
|
|
|
widget.onValidationFailed();
|
|
|
|
});
|
|
|
|
rethrow;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
// ignore: prefer_expression_function_bodies
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
// final theme = Theme.of(context);
|
|
|
|
// final scale = theme.extension<ScaleScheme>()!;
|
|
|
|
// final textTheme = theme.textTheme;
|
|
|
|
// final height = MediaQuery.of(context).size.height;
|
|
|
|
|
|
|
|
if (_isAccepting) {
|
|
|
|
return SizedBox(
|
|
|
|
height: 300,
|
|
|
|
width: 300,
|
2024-02-11 14:17:10 -05:00
|
|
|
child: buildProgressIndicator().toCenter())
|
2023-09-27 13:34:19 -04:00
|
|
|
.paddingAll(16);
|
|
|
|
}
|
|
|
|
return ConstrainedBox(
|
|
|
|
constraints: const BoxConstraints(maxHeight: 400, maxWidth: 400),
|
|
|
|
child: SingleChildScrollView(
|
|
|
|
padding: const EdgeInsets.all(16),
|
|
|
|
child: Column(
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
|
|
mainAxisSize: MainAxisSize.min,
|
|
|
|
children: <Widget>[
|
|
|
|
widget.buildInviteControl(context, this, _validateInviteData),
|
|
|
|
if (_isValidating)
|
|
|
|
Column(children: [
|
|
|
|
Text(translate('invite_dialog.validating'))
|
|
|
|
.paddingLTRB(0, 0, 0, 16),
|
2024-02-11 14:17:10 -05:00
|
|
|
buildProgressIndicator().paddingAll(16),
|
2023-09-27 13:34:19 -04:00
|
|
|
]).toCenter(),
|
|
|
|
if (_validInvitation == null &&
|
|
|
|
!_isValidating &&
|
|
|
|
widget.inviteControlIsValid())
|
|
|
|
Column(children: [
|
|
|
|
Text(translate('invite_dialog.invalid_invitation')),
|
|
|
|
const Icon(Icons.error)
|
|
|
|
]).paddingAll(16).toCenter(),
|
|
|
|
if (_validInvitation != null && !_isValidating)
|
|
|
|
Column(children: [
|
|
|
|
Container(
|
2024-01-30 14:14:11 -05:00
|
|
|
constraints: const BoxConstraints(maxHeight: 64),
|
|
|
|
width: double.infinity,
|
|
|
|
child: ProfileWidget(
|
|
|
|
profile: _validInvitation!.remoteProfile))
|
|
|
|
.paddingLTRB(0, 0, 0, 8),
|
2023-09-27 13:34:19 -04:00
|
|
|
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,
|
|
|
|
)
|
|
|
|
],
|
|
|
|
),
|
|
|
|
])
|
|
|
|
]),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void debugFillProperties(DiagnosticPropertiesBuilder properties) {
|
|
|
|
super.debugFillProperties(properties);
|
|
|
|
properties
|
|
|
|
..add(DiagnosticsProperty<bool>('isValidating', isValidating))
|
|
|
|
..add(DiagnosticsProperty<bool>('isAccepting', isAccepting));
|
|
|
|
}
|
|
|
|
}
|