veilidchat/lib/contact_invitation/views/scan_invitation_dialog.dart

414 lines
14 KiB
Dart
Raw Normal View History

2023-09-23 12:56:54 -04:00
import 'dart:async';
2023-09-27 13:34:19 -04:00
import 'dart:typed_data';
2023-09-23 12:56:54 -04:00
import 'package:awesome_extensions/awesome_extensions.dart';
2023-09-30 21:22:12 -04:00
import 'package:flutter/foundation.dart';
2023-09-23 12:56:54 -04:00
import 'package:flutter/material.dart';
2023-09-28 10:06:22 -04:00
import 'package:flutter/scheduler.dart';
2023-09-23 12:56:54 -04:00
import 'package:flutter_translate/flutter_translate.dart';
2023-09-29 22:45:50 -04:00
import 'package:image/image.dart' as img;
import 'package:mobile_scanner/mobile_scanner.dart';
2023-09-28 10:06:22 -04:00
import 'package:pasteboard/pasteboard.dart';
2024-06-16 22:12:24 -04:00
import 'package:provider/provider.dart';
2023-09-28 10:06:22 -04:00
import 'package:zxing2/qrcode.dart';
2023-09-23 12:56:54 -04:00
import '../../notifications/notifications.dart';
2024-01-09 20:58:27 -05:00
import '../../theme/theme.dart';
2024-04-05 22:03:04 -04:00
import 'invitation_dialog.dart';
2023-09-23 12:56:54 -04:00
2024-05-20 20:48:17 -04:00
// class BarcodeOverlay extends CustomPainter {
// BarcodeOverlay({
// required this.barcode,
// required this.boxFit,
// required this.capture,
// required this.size,
// });
// final BarcodeCapture capture;
// final Barcode barcode;
// final BoxFit boxFit;
// final Size size;
// @override
// void paint(Canvas canvas, Size size) {
// final adjustedSize = applyBoxFit(boxFit, size, size);
// var verticalPadding = size.height - adjustedSize.destination.height;
// var horizontalPadding = size.width - adjustedSize.destination.width;
// if (verticalPadding > 0) {
// verticalPadding = verticalPadding / 2;
// } else {
// verticalPadding = 0;
// }
// if (horizontalPadding > 0) {
// horizontalPadding = horizontalPadding / 2;
// } else {
// horizontalPadding = 0;
// }
// final ratioWidth = (Platform.isIOS ? capture.size.width : size.width) /
// adjustedSize.destination.width;
// final ratioHeight = (Platform.isIOS ? capture.size.height : size.height) /
// adjustedSize.destination.height;
// final adjustedOffset = <Offset>[];
// for (final offset in barcode.corners) {
// adjustedOffset.add(
// Offset(
// offset.dx / ratioWidth + horizontalPadding,
// offset.dy / ratioHeight + verticalPadding,
// ),
// );
// }
// final cutoutPath = Path()..addPolygon(adjustedOffset, true);
// final backgroundPaint = Paint()
// ..color = Colors.red.withOpacity(0.3)
// ..style = PaintingStyle.fill
// ..blendMode = BlendMode.dstOut;
// canvas.drawPath(cutoutPath, backgroundPaint);
// }
// @override
// bool shouldRepaint(covariant CustomPainter oldDelegate) => false;
// }
2023-09-28 10:06:22 -04:00
class ScannerOverlay extends CustomPainter {
ScannerOverlay(this.scanWindow);
final Rect scanWindow;
@override
void paint(Canvas canvas, Size size) {
final backgroundPath = Path()..addRect(Rect.largest);
final cutoutPath = Path()..addRect(scanWindow);
final backgroundPaint = Paint()
..color = Colors.black.withOpacity(0.5)
..style = PaintingStyle.fill
..blendMode = BlendMode.dstOut;
final backgroundWithCutout = Path.combine(
PathOperation.difference,
backgroundPath,
cutoutPath,
);
canvas.drawPath(backgroundWithCutout, backgroundPaint);
}
@override
bool shouldRepaint(covariant CustomPainter oldDelegate) => false;
}
2024-04-05 22:03:04 -04:00
class ScanInvitationDialog extends StatefulWidget {
2024-06-16 22:12:24 -04:00
const ScanInvitationDialog({required Locator locator, super.key})
: _locator = locator;
2023-09-23 12:56:54 -04:00
@override
2024-04-05 22:03:04 -04:00
ScanInvitationDialogState createState() => ScanInvitationDialogState();
2023-09-27 13:34:19 -04:00
static Future<void> show(BuildContext context) async {
2024-06-16 22:12:24 -04:00
final locator = context.read;
await showPopControlDialog<void>(
2023-09-27 13:34:19 -04:00
context: context,
builder: (context) => StyledDialog(
title: translate('scan_invitation_dialog.title'),
2024-06-16 22:12:24 -04:00
child: ScanInvitationDialog(locator: locator)));
2024-02-14 21:33:15 -05:00
}
2024-06-16 22:12:24 -04:00
final Locator _locator;
2023-09-23 12:56:54 -04:00
}
2024-04-05 22:03:04 -04:00
class ScanInvitationDialogState extends State<ScanInvitationDialog> {
2023-09-28 10:06:22 -04:00
bool scanned = false;
2023-09-23 12:56:54 -04:00
@override
void initState() {
super.initState();
}
2023-09-27 13:34:19 -04:00
void onValidationCancelled() {
2023-09-28 10:06:22 -04:00
setState(() {
scanned = false;
});
2023-09-23 12:56:54 -04:00
}
2023-09-28 10:06:22 -04:00
void onValidationSuccess() {}
2023-09-27 13:34:19 -04:00
void onValidationFailed() {
2023-09-28 10:06:22 -04:00
setState(() {
scanned = false;
});
2023-09-23 12:56:54 -04:00
}
2023-09-28 10:06:22 -04:00
2023-09-27 13:34:19 -04:00
bool inviteControlIsValid() => false; // _pasteTextController.text.isNotEmpty;
2023-09-23 12:56:54 -04:00
2023-09-28 10:06:22 -04:00
Future<Uint8List?> scanQRImage(BuildContext context) async {
final theme = Theme.of(context);
//final textTheme = theme.textTheme;
final scale = theme.extension<ScaleScheme>()!;
final windowSize = MediaQuery.of(context).size;
//final maxDialogWidth = min(windowSize.width - 64.0, 800.0 - 64.0);
//final maxDialogHeight = windowSize.height - 64.0;
final scanWindow = Rect.fromCenter(
center: MediaQuery.of(context).size.center(Offset.zero),
width: 200,
height: 200,
);
final cameraController = MobileScannerController();
try {
return showDialog(
context: context,
builder: (context) => Stack(
fit: StackFit.expand,
children: [
MobileScanner(
fit: BoxFit.contain,
scanWindow: scanWindow,
controller: cameraController,
errorBuilder: (context, error, child) =>
ScannerErrorWidget(error: error),
onDetect: (c) {
final barcode = c.barcodes.firstOrNull;
final barcodeBytes = barcode?.rawBytes;
if (barcodeBytes != null) {
cameraController.dispose();
Navigator.pop(context, barcodeBytes);
}
}),
CustomPaint(
painter: ScannerOverlay(scanWindow),
),
Align(
alignment: Alignment.bottomCenter,
child: Container(
alignment: Alignment.bottomCenter,
height: 100,
color: Colors.black.withOpacity(0.4),
child: Row(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: [
IconButton(
color: Colors.white,
icon: ValueListenableBuilder(
2024-05-20 20:48:17 -04:00
valueListenable: cameraController,
2023-09-28 10:06:22 -04:00
builder: (context, state, child) {
2024-05-20 20:48:17 -04:00
switch (state.torchState) {
2023-09-28 10:06:22 -04:00
case TorchState.off:
return Icon(Icons.flash_off,
color:
scale.grayScale.subtleBackground);
case TorchState.on:
return Icon(Icons.flash_on,
color: scale.primaryScale.primary);
2024-05-20 20:48:17 -04:00
case TorchState.auto:
return Icon(Icons.flash_auto,
color: scale.primaryScale.primary);
case TorchState.unavailable:
return Icon(Icons.no_flash,
color: scale.primaryScale.primary);
2023-09-28 10:06:22 -04:00
}
},
),
iconSize: 32,
onPressed: cameraController.toggleTorch,
),
SizedBox(
width: windowSize.width - 120,
height: 50,
child: FittedBox(
child: Text(
2024-04-05 22:03:04 -04:00
translate(
'scan_invitation_dialog.instructions'),
2023-09-28 10:06:22 -04:00
overflow: TextOverflow.fade,
style: Theme.of(context)
.textTheme
.labelLarge!
.copyWith(color: Colors.white),
),
),
),
IconButton(
color: Colors.white,
icon: ValueListenableBuilder(
2024-05-20 20:48:17 -04:00
valueListenable: cameraController,
2023-09-28 10:06:22 -04:00
builder: (context, state, child) {
2024-05-20 20:48:17 -04:00
switch (state.cameraDirection) {
2023-09-28 10:06:22 -04:00
case CameraFacing.front:
return const Icon(Icons.camera_front);
case CameraFacing.back:
return const Icon(Icons.camera_rear);
}
},
),
iconSize: 32,
onPressed: cameraController.switchCamera,
),
],
),
),
),
Align(
alignment: Alignment.topRight,
child: IconButton(
color: Colors.white,
icon:
Icon(Icons.close, color: scale.grayScale.primary),
2023-09-28 10:06:22 -04:00
iconSize: 32,
onPressed: () => {
SchedulerBinding.instance
.addPostFrameCallback((_) {
cameraController.dispose();
2024-05-20 20:48:17 -04:00
Navigator.pop(context);
2023-09-28 10:06:22 -04:00
})
})),
],
));
} on MobileScannerException catch (e) {
if (e.errorCode == MobileScannerErrorCode.permissionDenied) {
context
.read<NotificationsCubit>()
.error(text: translate('scan_invitation_dialog.permission_error'));
2023-09-28 10:06:22 -04:00
} else {
context
.read<NotificationsCubit>()
.error(text: translate('scan_invitation_dialog.error'));
2023-09-28 10:06:22 -04:00
}
} on Exception catch (_) {
context
.read<NotificationsCubit>()
.error(text: translate('scan_invitation_dialog.error'));
2023-09-28 10:06:22 -04:00
}
return null;
}
Future<Uint8List?> pasteQRImage(BuildContext context) async {
final imageBytes = await Pasteboard.image;
if (imageBytes == null) {
if (context.mounted) {
context
.read<NotificationsCubit>()
.error(text: translate('scan_invitation_dialog.not_an_image'));
2023-09-28 10:06:22 -04:00
}
return null;
}
final image = img.decodeImage(imageBytes);
if (image == null) {
if (context.mounted) {
context.read<NotificationsCubit>().error(
text: translate('scan_invitation_dialog.could_not_decode_image'));
2023-09-28 10:06:22 -04:00
}
return null;
}
try {
final source = RGBLuminanceSource(
image.width,
image.height,
image
.convert(numChannels: 4)
.getBytes(order: img.ChannelOrder.abgr)
.buffer
.asInt32List());
final bitmap = BinaryBitmap(HybridBinarizer(source));
final reader = QRCodeReader();
final result = reader.decode(bitmap);
final segs = result.resultMetadata[ResultMetadataType.byteSegments]!
as List<Int8List>;
return Uint8List.fromList(segs[0].toList());
} on Exception catch (_) {
if (context.mounted) {
context.read<NotificationsCubit>().error(
text: translate('scan_invitation_dialog.not_a_valid_qr_code'));
2023-09-28 10:06:22 -04:00
}
return null;
}
}
2023-09-27 13:34:19 -04:00
Widget buildInviteControl(
BuildContext context,
2024-04-05 22:03:04 -04:00
InvitationDialogState dialogState,
2023-09-27 13:34:19 -04:00
Future<void> Function({required Uint8List inviteData})
validateInviteData) {
2023-09-30 21:22:12 -04:00
//final theme = Theme.of(context);
2023-09-23 12:56:54 -04:00
//final scale = theme.extension<ScaleScheme>()!;
2023-09-28 10:06:22 -04:00
//final textTheme = theme.textTheme;
2023-09-23 12:56:54 -04:00
//final height = MediaQuery.of(context).size.height;
2023-09-28 10:06:22 -04:00
if (isiOS || isAndroid) {
return Column(mainAxisSize: MainAxisSize.min, children: [
if (!scanned)
Text(
2024-04-05 22:03:04 -04:00
translate('scan_invitation_dialog.scan_qr_here'),
2023-09-28 10:06:22 -04:00
).paddingLTRB(0, 0, 0, 8),
if (!scanned)
Container(
constraints: const BoxConstraints(maxHeight: 200),
child: ElevatedButton(
onPressed: dialogState.isValidating
? null
: () async {
final inviteData = await scanQRImage(context);
if (inviteData != null) {
setState(() {
scanned = true;
});
await validateInviteData(inviteData: inviteData);
}
},
2024-04-05 22:03:04 -04:00
child: Text(translate('scan_invitation_dialog.scan'))),
2023-09-28 10:06:22 -04:00
).paddingLTRB(0, 0, 0, 8)
]);
}
2023-09-27 13:34:19 -04:00
return Column(mainAxisSize: MainAxisSize.min, children: [
2023-09-28 10:06:22 -04:00
if (!scanned)
Text(
2024-04-05 22:03:04 -04:00
translate('scan_invitation_dialog.paste_qr_here'),
2023-09-28 10:06:22 -04:00
).paddingLTRB(0, 0, 0, 8),
if (!scanned)
Container(
constraints: const BoxConstraints(maxHeight: 200),
child: ElevatedButton(
onPressed: dialogState.isValidating
? null
: () async {
final inviteData = await pasteQRImage(context);
if (inviteData != null) {
2023-10-11 23:12:10 -04:00
await validateInviteData(inviteData: inviteData);
2023-09-28 10:06:22 -04:00
setState(() {
scanned = true;
});
}
},
2024-04-05 22:03:04 -04:00
child: Text(translate('scan_invitation_dialog.paste'))),
2023-09-28 10:06:22 -04:00
).paddingLTRB(0, 0, 0, 8)
2023-09-27 13:34:19 -04:00
]);
2023-09-23 12:56:54 -04:00
}
@override
2023-09-27 13:34:19 -04:00
// ignore: prefer_expression_function_bodies
Widget build(BuildContext context) {
2024-04-05 22:03:04 -04:00
return InvitationDialog(
2024-06-16 22:12:24 -04:00
locator: widget._locator,
2023-09-27 13:34:19 -04:00
onValidationCancelled: onValidationCancelled,
onValidationSuccess: onValidationSuccess,
onValidationFailed: onValidationFailed,
inviteControlIsValid: inviteControlIsValid,
buildInviteControl: buildInviteControl);
2023-09-23 12:56:54 -04:00
}
2023-09-30 21:22:12 -04:00
2023-09-29 22:45:50 -04:00
@override
void debugFillProperties(DiagnosticPropertiesBuilder properties) {
super.debugFillProperties(properties);
properties.add(DiagnosticsProperty<bool>('scanned', scanned));
}
2023-09-23 12:56:54 -04:00
}