|
|
|
import 'dart:async';
|
|
|
|
import 'dart:io';
|
|
|
|
import 'dart:isolate';
|
|
|
|
import 'dart:math';
|
|
|
|
|
|
|
|
import 'package:camera/camera.dart';
|
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
import 'package:flutter/services.dart';
|
|
|
|
import 'package:flutter_beep/flutter_beep.dart';
|
|
|
|
import 'flutter_zxing.dart';
|
|
|
|
|
|
|
|
import 'isolate_utils.dart';
|
|
|
|
|
|
|
|
class ZxingReaderWidget extends StatefulWidget {
|
|
|
|
const ZxingReaderWidget({
|
|
|
|
Key? key,
|
|
|
|
required this.onScan,
|
|
|
|
this.onControllerCreated,
|
|
|
|
this.codeFormat = Format.Any,
|
|
|
|
this.beep = true,
|
|
|
|
this.showCroppingRect = true,
|
|
|
|
this.scanDelay = const Duration(milliseconds: 500), // 500ms delay
|
|
|
|
this.cropPercent = 0.5, // 50% of the screen
|
|
|
|
this.resolution = ResolutionPreset.high,
|
|
|
|
}) : super(key: key);
|
|
|
|
|
|
|
|
final Function(CodeResult) onScan;
|
|
|
|
final Function(CameraController?)? onControllerCreated;
|
|
|
|
final int codeFormat;
|
|
|
|
final bool beep;
|
|
|
|
final bool showCroppingRect;
|
|
|
|
final Duration scanDelay;
|
|
|
|
final double cropPercent;
|
|
|
|
final ResolutionPreset resolution;
|
|
|
|
|
|
|
|
@override
|
|
|
|
State<ZxingReaderWidget> createState() => _ZxingReaderWidgetState();
|
|
|
|
}
|
|
|
|
|
|
|
|
class _ZxingReaderWidgetState extends State<ZxingReaderWidget>
|
|
|
|
with TickerProviderStateMixin {
|
|
|
|
late List<CameraDescription> cameras;
|
|
|
|
CameraController? controller;
|
|
|
|
|
|
|
|
bool isAndroid() => Theme.of(context).platform == TargetPlatform.android;
|
|
|
|
|
|
|
|
// true when code detecting is ongoing
|
|
|
|
bool _isProcessing = false;
|
|
|
|
|
|
|
|
/// Instance of [IsolateUtils]
|
|
|
|
IsolateUtils? isolateUtils;
|
|
|
|
|
|
|
|
@override
|
|
|
|
void initState() {
|
|
|
|
super.initState();
|
|
|
|
|
|
|
|
initStateAsync();
|
|
|
|
}
|
|
|
|
|
|
|
|
void initStateAsync() async {
|
|
|
|
// Spawn a new isolate
|
|
|
|
isolateUtils = IsolateUtils();
|
|
|
|
await isolateUtils?.start();
|
|
|
|
|
|
|
|
availableCameras().then((cameras) {
|
|
|
|
setState(() {
|
|
|
|
this.cameras = cameras;
|
|
|
|
if (cameras.isNotEmpty) {
|
|
|
|
onNewCameraSelected(cameras.first);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
SystemChannels.lifecycle.setMessageHandler((message) async {
|
|
|
|
debugPrint(message);
|
|
|
|
final CameraController? cameraController = controller;
|
|
|
|
if (cameraController == null || !cameraController.value.isInitialized) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (mounted) {
|
|
|
|
if (message == AppLifecycleState.paused.toString()) {
|
|
|
|
cameraController.dispose();
|
|
|
|
}
|
|
|
|
if (message == AppLifecycleState.resumed.toString()) {
|
|
|
|
onNewCameraSelected(cameraController.description);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void dispose() {
|
|
|
|
controller?.dispose();
|
|
|
|
isolateUtils?.stop();
|
|
|
|
super.dispose();
|
|
|
|
}
|
|
|
|
|
|
|
|
void onNewCameraSelected(CameraDescription cameraDescription) async {
|
|
|
|
if (controller != null) {
|
|
|
|
await controller!.dispose();
|
|
|
|
}
|
|
|
|
|
|
|
|
controller = CameraController(
|
|
|
|
cameraDescription,
|
|
|
|
widget.resolution,
|
|
|
|
enableAudio: false,
|
|
|
|
imageFormatGroup:
|
|
|
|
isAndroid() ? ImageFormatGroup.yuv420 : ImageFormatGroup.bgra8888,
|
|
|
|
);
|
|
|
|
|
|
|
|
try {
|
|
|
|
await controller?.initialize();
|
|
|
|
controller?.startImageStream(processCameraImage);
|
|
|
|
} on CameraException catch (e) {
|
|
|
|
_showCameraException(e);
|
|
|
|
}
|
|
|
|
|
|
|
|
controller?.addListener(() {
|
|
|
|
if (mounted) setState(() {});
|
|
|
|
});
|
|
|
|
|
|
|
|
if (mounted) {
|
|
|
|
setState(() {});
|
|
|
|
}
|
|
|
|
|
|
|
|
widget.onControllerCreated?.call(controller);
|
|
|
|
}
|
|
|
|
|
|
|
|
void _showCameraException(CameraException e) {
|
|
|
|
logError(e.code, e.description);
|
|
|
|
showInSnackBar('Error: ${e.code}\n${e.description}');
|
|
|
|
}
|
|
|
|
|
|
|
|
void showInSnackBar(String message) {}
|
|
|
|
|
|
|
|
void logError(String code, String? message) {
|
|
|
|
if (message != null) {
|
|
|
|
debugPrint('Error: $code\nError Message: $message');
|
|
|
|
} else {
|
|
|
|
debugPrint('Error: $code');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
processCameraImage(CameraImage image) async {
|
|
|
|
if (!_isProcessing) {
|
|
|
|
_isProcessing = true;
|
|
|
|
try {
|
|
|
|
var isolateData =
|
|
|
|
IsolateData(image, widget.codeFormat, widget.cropPercent);
|
|
|
|
|
|
|
|
/// perform inference in separate isolate
|
|
|
|
CodeResult result = await inference(isolateData);
|
|
|
|
if (result.isValidBool) {
|
|
|
|
if (widget.beep) {
|
|
|
|
FlutterBeep.beep();
|
|
|
|
}
|
|
|
|
widget.onScan(result);
|
|
|
|
setState(() {});
|
|
|
|
await Future.delayed(const Duration(seconds: 1));
|
|
|
|
}
|
|
|
|
} on FileSystemException catch (e) {
|
|
|
|
debugPrint(e.message);
|
|
|
|
} catch (e) {
|
|
|
|
debugPrint(e.toString());
|
|
|
|
}
|
|
|
|
await Future.delayed(widget.scanDelay);
|
|
|
|
_isProcessing = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Runs inference in another isolate
|
|
|
|
Future<CodeResult> inference(IsolateData isolateData) async {
|
|
|
|
ReceivePort responsePort = ReceivePort();
|
|
|
|
isolateUtils?.sendPort
|
|
|
|
?.send(isolateData..responsePort = responsePort.sendPort);
|
|
|
|
var results = await responsePort.first;
|
|
|
|
return results;
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
final size = MediaQuery.of(context).size;
|
|
|
|
final cropSize = min(size.width, size.height) * widget.cropPercent;
|
|
|
|
return Stack(
|
|
|
|
children: [
|
|
|
|
// Camera preview
|
|
|
|
Center(child: _cameraPreviewWidget(cropSize)),
|
|
|
|
],
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Display the preview from the camera.
|
|
|
|
Widget _cameraPreviewWidget(double cropSize) {
|
|
|
|
final CameraController? cameraController = controller;
|
|
|
|
if (cameraController == null || !cameraController.value.isInitialized) {
|
|
|
|
return const CircularProgressIndicator();
|
|
|
|
} else {
|
|
|
|
final size = MediaQuery.of(context).size;
|
|
|
|
var cameraMaxSize = max(size.width, size.height);
|
|
|
|
return Stack(
|
|
|
|
children: [
|
|
|
|
SizedBox(
|
|
|
|
width: cameraMaxSize,
|
|
|
|
height: cameraMaxSize,
|
|
|
|
child: ClipRRect(
|
|
|
|
child: OverflowBox(
|
|
|
|
alignment: Alignment.center,
|
|
|
|
child: FittedBox(
|
|
|
|
fit: BoxFit.cover,
|
|
|
|
child: SizedBox(
|
|
|
|
width: cameraMaxSize,
|
|
|
|
child: CameraPreview(
|
|
|
|
cameraController,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
widget.showCroppingRect
|
|
|
|
? Container(
|
|
|
|
decoration: ShapeDecoration(
|
|
|
|
shape: ScannerOverlay(
|
|
|
|
borderColor: Theme.of(context).primaryColor,
|
|
|
|
overlayColor: const Color.fromRGBO(0, 0, 0, 0.5),
|
|
|
|
borderRadius: 1,
|
|
|
|
borderLength: 16,
|
|
|
|
borderWidth: 8,
|
|
|
|
cutOutSize: cropSize,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
)
|
|
|
|
: Container()
|
|
|
|
],
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|