veilidchat/lib/contact_invitation/views/contact_invitation_display.dart

167 lines
6.9 KiB
Dart
Raw Normal View History

2023-09-22 11:20:28 -04:00
import 'dart:math';
2023-08-03 00:49:48 -04:00
2023-08-09 02:33:31 -04:00
import 'package:awesome_extensions/awesome_extensions.dart';
2023-08-04 01:00:38 -04:00
import 'package:basic_utils/basic_utils.dart';
2023-08-01 00:39:50 -04:00
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
2023-08-04 01:00:38 -04:00
import 'package:flutter/services.dart';
2024-04-05 22:03:04 -04:00
import 'package:flutter_bloc/flutter_bloc.dart';
2023-08-02 21:09:28 -04:00
import 'package:flutter_translate/flutter_translate.dart';
import 'package:provider/provider.dart';
2023-08-04 01:00:38 -04:00
import 'package:qr_flutter/qr_flutter.dart';
2024-01-09 20:58:27 -05:00
import 'package:veilid_support/veilid_support.dart';
2023-08-01 00:39:50 -04:00
2024-07-27 18:36:06 -04:00
import '../../account_manager/account_manager.dart';
import '../../notifications/notifications.dart';
import '../../proto/proto.dart' as proto;
import '../../theme/theme.dart';
2024-03-24 12:13:27 -04:00
import '../contact_invitation.dart';
2024-01-29 22:38:19 -05:00
2024-04-05 22:03:04 -04:00
class ContactInvitationDisplayDialog extends StatelessWidget {
const ContactInvitationDisplayDialog._({
required this.locator,
2023-08-03 00:49:48 -04:00
required this.message,
2024-07-27 18:36:06 -04:00
required this.fingerprint,
2023-08-02 21:09:28 -04:00
});
final Locator locator;
2023-08-03 00:49:48 -04:00
final String message;
2024-07-27 18:36:06 -04:00
final String fingerprint;
2023-08-02 21:09:28 -04:00
2023-08-03 00:49:48 -04:00
@override
void debugFillProperties(DiagnosticPropertiesBuilder properties) {
super.debugFillProperties(properties);
2024-04-05 22:03:04 -04:00
properties
..add(StringProperty('message', message))
2024-07-27 18:36:06 -04:00
..add(DiagnosticsProperty<Locator>('locator', locator))
..add(StringProperty('fingerprint', fingerprint));
2023-08-01 00:39:50 -04:00
}
2023-08-04 01:00:38 -04:00
String makeTextInvite(String message, Uint8List data) {
final invite = StringUtils.addCharAtPosition(
base64UrlNoPadEncode(data), '\n', 40,
repeat: true);
final msg = message.isNotEmpty ? '$message\n' : '';
2024-07-27 18:36:06 -04:00
2023-08-04 01:00:38 -04:00
return '$msg'
2023-08-05 01:00:46 -04:00
'--- BEGIN VEILIDCHAT CONTACT INVITE ----\n'
2023-08-04 01:00:38 -04:00
'$invite\n'
2024-07-27 18:36:06 -04:00
'---- END VEILIDCHAT CONTACT INVITE -----\n'
'Fingerprint:\n$fingerprint\n';
2023-08-04 01:00:38 -04:00
}
2023-08-01 00:39:50 -04:00
@override
2023-08-02 21:09:28 -04:00
// ignore: prefer_expression_function_bodies
Widget build(BuildContext context) {
final theme = Theme.of(context);
2023-08-04 01:00:38 -04:00
final textTheme = theme.textTheme;
2024-07-06 20:09:18 -04:00
final scaleConfig = theme.extension<ScaleConfig>()!;
2023-08-04 01:00:38 -04:00
final generatorOutputV = context.watch<InvitationGeneratorCubit>().state;
2024-01-29 22:38:19 -05:00
2023-09-22 11:20:28 -04:00
final cardsize =
min<double>(MediaQuery.of(context).size.shortestSide - 48.0, 400);
2023-08-02 21:09:28 -04:00
return BlocListener<ContactInvitationListCubit,
ContactInvitiationListState>(
bloc: locator<ContactInvitationListCubit>(),
listener: (context, state) {
final listState = state.state.asData?.value;
final data = generatorOutputV.asData?.value;
if (listState != null && data != null) {
final idx = listState.indexWhere((x) =>
x.value.contactRequestInbox.recordKey.toVeilid() == data.$2);
if (idx == -1) {
// This invitation is gone, close it
Navigator.pop(context);
}
}
},
child: PopControl(
dismissible: !generatorOutputV.isLoading,
child: Dialog(
shape: RoundedRectangleBorder(
side: const BorderSide(width: 2),
borderRadius: BorderRadius.circular(
16 * scaleConfig.borderRadiusScale)),
backgroundColor: Colors.white,
child: ConstrainedBox(
constraints: BoxConstraints(
minWidth: cardsize,
maxWidth: cardsize,
minHeight: cardsize,
maxHeight: cardsize),
child: generatorOutputV.when(
loading: buildProgressIndicator,
data: (data) => Column(children: [
FittedBox(
child: Text(
translate('create_invitation_dialog'
'.contact_invitation'),
style: textTheme.headlineSmall!
.copyWith(color: Colors.black)))
.paddingAll(8),
FittedBox(
2024-07-27 18:36:06 -04:00
child: QrImageView.withQr(
size: 300,
qr: QrCode.fromUint8List(
data: data.$1,
errorCorrectLevel:
QrErrorCorrectLevel.L)),
).expanded(),
Text(message,
softWrap: true,
style: textTheme.labelLarge!
.copyWith(color: Colors.black))
.paddingAll(8),
2024-07-27 18:36:06 -04:00
Text(
'${translate('create_invitation_dialog.fingerprint')}\n'
'$fingerprint',
softWrap: true,
textAlign: TextAlign.center,
style: textTheme.labelSmall!.copyWith(
color: Colors.black,
fontFamily: 'Source Code Pro'))
.paddingAll(2),
ElevatedButton.icon(
icon: const Icon(Icons.copy),
style: ElevatedButton.styleFrom(
foregroundColor: Colors.black,
backgroundColor: Colors.white,
side: const BorderSide()),
label: Text(translate(
'create_invitation_dialog.copy_invitation')),
onPressed: () async {
context.read<NotificationsCubit>().info(
text: translate('create_invitation_dialog'
'.invitation_copied'));
await Clipboard.setData(ClipboardData(
text: makeTextInvite(message, data.$1)));
},
).paddingAll(16),
]),
error: errorPage)))));
2023-08-03 00:49:48 -04:00
}
2024-07-27 18:36:06 -04:00
static Future<void> show({
required BuildContext context,
required Locator locator,
required InvitationGeneratorCubit Function(BuildContext) create,
required String message,
}) async {
final fingerprint =
locator<AccountInfoCubit>().state.identityPublicKey.toString();
2024-04-05 22:03:04 -04:00
await showPopControlDialog<void>(
context: context,
builder: (context) => BlocProvider(
create: create,
child: ContactInvitationDisplayDialog._(
locator: locator,
2024-04-05 22:03:04 -04:00
message: message,
2024-07-27 18:36:06 -04:00
fingerprint: fingerprint,
2024-04-05 22:03:04 -04:00
)));
2023-08-01 00:39:50 -04:00
}
}