mirror of
https://gitlab.com/veilid/veilidchat.git
synced 2025-01-05 13:00:52 -05:00
400 lines
13 KiB
Dart
400 lines
13 KiB
Dart
import 'dart:async';
|
|
import 'dart:io';
|
|
import 'dart:typed_data';
|
|
|
|
import 'package:awesome_extensions/awesome_extensions.dart';
|
|
import 'package:flutter/foundation.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:flutter/scheduler.dart';
|
|
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
|
import 'package:flutter_translate/flutter_translate.dart';
|
|
import 'package:image/image.dart' as img;
|
|
import 'package:mobile_scanner/mobile_scanner.dart';
|
|
import 'package:pasteboard/pasteboard.dart';
|
|
import 'package:zxing2/qrcode.dart';
|
|
|
|
import '../tools/tools.dart';
|
|
import 'invite_dialog.dart';
|
|
|
|
class BarcodeOverlay extends CustomPainter {
|
|
BarcodeOverlay({
|
|
required this.barcode,
|
|
required this.arguments,
|
|
required this.boxFit,
|
|
required this.capture,
|
|
});
|
|
|
|
final BarcodeCapture capture;
|
|
final Barcode barcode;
|
|
final MobileScannerArguments arguments;
|
|
final BoxFit boxFit;
|
|
|
|
@override
|
|
void paint(Canvas canvas, Size size) {
|
|
if (barcode.corners == null) {
|
|
return;
|
|
}
|
|
final adjustedSize = applyBoxFit(boxFit, arguments.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.width! : arguments.size.width) /
|
|
adjustedSize.destination.width;
|
|
final ratioHeight =
|
|
(Platform.isIOS ? capture.height! : arguments.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;
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
class ScanInviteDialog extends ConsumerStatefulWidget {
|
|
const ScanInviteDialog({super.key});
|
|
|
|
@override
|
|
ScanInviteDialogState createState() => ScanInviteDialogState();
|
|
|
|
static Future<void> show(BuildContext context) async {
|
|
await showStyledDialog<void>(
|
|
context: context,
|
|
title: translate('scan_invite_dialog.title'),
|
|
child: const ScanInviteDialog());
|
|
}
|
|
}
|
|
|
|
class ScanInviteDialogState extends ConsumerState<ScanInviteDialog> {
|
|
bool scanned = false;
|
|
|
|
@override
|
|
void initState() {
|
|
super.initState();
|
|
}
|
|
|
|
void onValidationCancelled() {
|
|
setState(() {
|
|
scanned = false;
|
|
});
|
|
}
|
|
|
|
void onValidationSuccess() {}
|
|
void onValidationFailed() {
|
|
setState(() {
|
|
scanned = false;
|
|
});
|
|
}
|
|
|
|
bool inviteControlIsValid() => false; // _pasteTextController.text.isNotEmpty;
|
|
|
|
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(
|
|
valueListenable: cameraController.torchState,
|
|
builder: (context, state, child) {
|
|
switch (state) {
|
|
case TorchState.off:
|
|
return Icon(Icons.flash_off,
|
|
color:
|
|
scale.grayScale.subtleBackground);
|
|
case TorchState.on:
|
|
return Icon(Icons.flash_on,
|
|
color: scale.primaryScale.background);
|
|
}
|
|
},
|
|
),
|
|
iconSize: 32,
|
|
onPressed: cameraController.toggleTorch,
|
|
),
|
|
SizedBox(
|
|
width: windowSize.width - 120,
|
|
height: 50,
|
|
child: FittedBox(
|
|
child: Text(
|
|
translate('scan_invite_dialog.instructions'),
|
|
overflow: TextOverflow.fade,
|
|
style: Theme.of(context)
|
|
.textTheme
|
|
.labelLarge!
|
|
.copyWith(color: Colors.white),
|
|
),
|
|
),
|
|
),
|
|
IconButton(
|
|
color: Colors.white,
|
|
icon: ValueListenableBuilder(
|
|
valueListenable:
|
|
cameraController.cameraFacingState,
|
|
builder: (context, state, child) {
|
|
switch (state) {
|
|
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.background),
|
|
iconSize: 32,
|
|
onPressed: () => {
|
|
SchedulerBinding.instance
|
|
.addPostFrameCallback((_) {
|
|
cameraController.dispose();
|
|
Navigator.pop(context, null);
|
|
})
|
|
})),
|
|
],
|
|
));
|
|
} on MobileScannerException catch (e) {
|
|
if (e.errorCode == MobileScannerErrorCode.permissionDenied) {
|
|
showErrorToast(
|
|
context, translate('scan_invite_dialog.permission_error'));
|
|
} else {
|
|
showErrorToast(context, translate('scan_invite_dialog.error'));
|
|
}
|
|
} on Exception catch (_) {
|
|
showErrorToast(context, translate('scan_invite_dialog.error'));
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
Future<Uint8List?> pasteQRImage(BuildContext context) async {
|
|
final imageBytes = await Pasteboard.image;
|
|
if (imageBytes == null) {
|
|
if (context.mounted) {
|
|
showErrorToast(context, translate('scan_invite_dialog.not_an_image'));
|
|
}
|
|
return null;
|
|
}
|
|
|
|
final image = img.decodeImage(imageBytes);
|
|
if (image == null) {
|
|
if (context.mounted) {
|
|
showErrorToast(
|
|
context, translate('scan_invite_dialog.could_not_decode_image'));
|
|
}
|
|
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) {
|
|
showErrorToast(
|
|
context, translate('scan_invite_dialog.not_a_valid_qr_code'));
|
|
}
|
|
return null;
|
|
}
|
|
}
|
|
|
|
Widget buildInviteControl(
|
|
BuildContext context,
|
|
InviteDialogState dialogState,
|
|
Future<void> Function({required Uint8List inviteData})
|
|
validateInviteData) {
|
|
//final theme = Theme.of(context);
|
|
//final scale = theme.extension<ScaleScheme>()!;
|
|
//final textTheme = theme.textTheme;
|
|
//final height = MediaQuery.of(context).size.height;
|
|
|
|
if (isiOS || isAndroid) {
|
|
return Column(mainAxisSize: MainAxisSize.min, children: [
|
|
if (!scanned)
|
|
Text(
|
|
translate('scan_invite_dialog.scan_qr_here'),
|
|
).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);
|
|
}
|
|
},
|
|
child: Text(translate('scan_invite_dialog.scan'))),
|
|
).paddingLTRB(0, 0, 0, 8)
|
|
]);
|
|
}
|
|
return Column(mainAxisSize: MainAxisSize.min, children: [
|
|
if (!scanned)
|
|
Text(
|
|
translate('scan_invite_dialog.paste_qr_here'),
|
|
).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) {
|
|
setState(() {
|
|
scanned = true;
|
|
});
|
|
await validateInviteData(inviteData: inviteData);
|
|
}
|
|
},
|
|
child: Text(translate('scan_invite_dialog.paste'))),
|
|
).paddingLTRB(0, 0, 0, 8)
|
|
]);
|
|
}
|
|
|
|
@override
|
|
// ignore: prefer_expression_function_bodies
|
|
Widget build(BuildContext context) {
|
|
return InviteDialog(
|
|
onValidationCancelled: onValidationCancelled,
|
|
onValidationSuccess: onValidationSuccess,
|
|
onValidationFailed: onValidationFailed,
|
|
inviteControlIsValid: inviteControlIsValid,
|
|
buildInviteControl: buildInviteControl);
|
|
}
|
|
|
|
@override
|
|
void debugFillProperties(DiagnosticPropertiesBuilder properties) {
|
|
super.debugFillProperties(properties);
|
|
properties.add(DiagnosticsProperty<bool>('scanned', scanned));
|
|
}
|
|
}
|