mirror of
https://gitlab.com/veilid/veilidchat.git
synced 2025-01-13 00:29:27 -05:00
249 lines
8.0 KiB
Dart
249 lines
8.0 KiB
Dart
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_riverpod/flutter_riverpod.dart';
|
|
import 'package:flutter_translate/flutter_translate.dart';
|
|
|
|
import '../entities/local_account.dart';
|
|
import '../providers/account.dart';
|
|
import '../providers/contact_invite.dart';
|
|
import '../tools/tools.dart';
|
|
import '../veilid_support/veilid_support.dart';
|
|
import 'contact_invitation_display.dart';
|
|
import 'enter_password.dart';
|
|
import 'enter_pin.dart';
|
|
|
|
class SendInviteDialog extends ConsumerStatefulWidget {
|
|
const SendInviteDialog({super.key});
|
|
|
|
@override
|
|
SendInviteDialogState createState() => SendInviteDialogState();
|
|
|
|
static Future<void> show(BuildContext context) async {
|
|
await showStyledDialog<void>(
|
|
context: context,
|
|
title: translate('send_invite_dialog.title'),
|
|
child: const SendInviteDialog());
|
|
}
|
|
}
|
|
|
|
class SendInviteDialogState extends ConsumerState<SendInviteDialog> {
|
|
final _messageTextController = TextEditingController(
|
|
text: translate('send_invite_dialog.connect_with_me'));
|
|
|
|
EncryptionKeyType _encryptionKeyType = EncryptionKeyType.none;
|
|
String _encryptionKey = '';
|
|
Timestamp? _expiration;
|
|
|
|
@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('send_invite_dialog.pin_description');
|
|
final pin = await showDialog<String>(
|
|
context: context,
|
|
builder: (context) =>
|
|
EnterPinDialog(reenter: false, 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(
|
|
reenter: true,
|
|
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('send_invite_dialog.pin_does_not_match'));
|
|
setState(() {
|
|
_encryptionKeyType = EncryptionKeyType.none;
|
|
_encryptionKey = '';
|
|
});
|
|
}
|
|
}
|
|
|
|
Future<void> _onPasswordEncryptionSelected(bool selected) async {
|
|
final description = translate('send_invite_dialog.password_description');
|
|
final password = await showDialog<String>(
|
|
context: context,
|
|
builder: (context) => EnterPasswordDialog(description: description));
|
|
if (password == null) {
|
|
return;
|
|
}
|
|
// ignore: use_build_context_synchronously
|
|
if (!context.mounted) {
|
|
return;
|
|
}
|
|
final matchpass = await showDialog<String>(
|
|
context: context,
|
|
builder: (context) => EnterPasswordDialog(
|
|
matchPass: password,
|
|
description: description,
|
|
));
|
|
if (matchpass == null) {
|
|
return;
|
|
} else if (password == matchpass) {
|
|
setState(() {
|
|
_encryptionKeyType = EncryptionKeyType.password;
|
|
_encryptionKey = password;
|
|
});
|
|
} else {
|
|
// ignore: use_build_context_synchronously
|
|
if (!context.mounted) {
|
|
return;
|
|
}
|
|
showErrorToast(
|
|
context, translate('send_invite_dialog.password_does_not_match'));
|
|
setState(() {
|
|
_encryptionKeyType = EncryptionKeyType.none;
|
|
_encryptionKey = '';
|
|
});
|
|
}
|
|
}
|
|
|
|
Future<void> _onGenerateButtonPressed() async {
|
|
final navigator = Navigator.of(context);
|
|
|
|
// Start generation
|
|
final activeAccountInfo = await ref.read(fetchActiveAccountProvider.future);
|
|
if (activeAccountInfo == null) {
|
|
navigator.pop();
|
|
return;
|
|
}
|
|
final generator = createContactInvitation(
|
|
activeAccountInfo: activeAccountInfo,
|
|
encryptionKeyType: _encryptionKeyType,
|
|
encryptionKey: _encryptionKey,
|
|
message: _messageTextController.text,
|
|
expiration: _expiration);
|
|
// ignore: use_build_context_synchronously
|
|
if (!context.mounted) {
|
|
return;
|
|
}
|
|
await showDialog<void>(
|
|
context: context,
|
|
builder: (context) => ContactInvitationDisplayDialog(
|
|
name: activeAccountInfo.localAccount.name,
|
|
message: _messageTextController.text,
|
|
generator: generator,
|
|
));
|
|
// if (ret == null) {
|
|
// return;
|
|
// }
|
|
ref.invalidate(fetchContactInvitationRecordsProvider);
|
|
navigator.pop();
|
|
}
|
|
|
|
@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<ScaleScheme>()!;
|
|
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: <Widget>[
|
|
Text(
|
|
translate('send_invite_dialog.message_to_contact'),
|
|
).paddingAll(8),
|
|
TextField(
|
|
controller: _messageTextController,
|
|
inputFormatters: [
|
|
LengthLimitingTextInputFormatter(128),
|
|
],
|
|
decoration: InputDecoration(
|
|
border: const OutlineInputBorder(),
|
|
hintText: translate('send_invite_dialog.enter_message_hint'),
|
|
labelText: translate('send_invite_dialog.message')),
|
|
).paddingAll(8),
|
|
const SizedBox(height: 10),
|
|
Text(translate('send_invite_dialog.protect_this_invitation'),
|
|
style: textTheme.labelLarge)
|
|
.paddingAll(8),
|
|
Wrap(spacing: 5, children: [
|
|
ChoiceChip(
|
|
label: Text(translate('send_invite_dialog.unlocked')),
|
|
selected: _encryptionKeyType == EncryptionKeyType.none,
|
|
onSelected: _onNoneEncryptionSelected,
|
|
),
|
|
ChoiceChip(
|
|
label: Text(translate('send_invite_dialog.pin')),
|
|
selected: _encryptionKeyType == EncryptionKeyType.pin,
|
|
onSelected: _onPinEncryptionSelected,
|
|
),
|
|
ChoiceChip(
|
|
label: Text(translate('send_invite_dialog.password')),
|
|
selected: _encryptionKeyType == EncryptionKeyType.password,
|
|
onSelected: _onPasswordEncryptionSelected,
|
|
)
|
|
]).paddingAll(8),
|
|
Container(
|
|
width: double.infinity,
|
|
height: 60,
|
|
padding: const EdgeInsets.all(8),
|
|
child: ElevatedButton(
|
|
onPressed: _onGenerateButtonPressed,
|
|
child: Text(
|
|
translate('send_invite_dialog.generate'),
|
|
),
|
|
),
|
|
),
|
|
Text(translate('send_invite_dialog.note')).paddingAll(8),
|
|
Text(
|
|
translate('send_invite_dialog.note_text'),
|
|
style: Theme.of(context).textTheme.bodySmall,
|
|
).paddingAll(8),
|
|
],
|
|
),
|
|
),
|
|
);
|
|
}
|
|
|
|
@override
|
|
void debugFillProperties(DiagnosticPropertiesBuilder properties) {
|
|
super.debugFillProperties(properties);
|
|
properties.add(DiagnosticsProperty<TextEditingController>(
|
|
'messageTextController', _messageTextController));
|
|
}
|
|
}
|