Add QR scanner

This commit is contained in:
Anton Tananaev 2025-06-23 21:32:17 -07:00
parent ee500f541f
commit 3437b670ea
7 changed files with 128 additions and 25 deletions

89
lib/qr_code_screen.dart Normal file
View file

@ -0,0 +1,89 @@
import 'package:flutter/material.dart';
import 'package:mobile_scanner/mobile_scanner.dart';
import 'package:flutter_background_geolocation/flutter_background_geolocation.dart' as bg;
import 'preferences.dart';
import 'l10n/app_localizations.dart';
class QrCodeScreen extends StatefulWidget {
const QrCodeScreen({super.key});
@override
State<QrCodeScreen> createState() => _QrCodeScreenState();
}
class _QrCodeScreenState extends State<QrCodeScreen> {
bool _scanned = false;
Future<void> _applySettings(Uri uri) async {
await Preferences.instance.setString(Preferences.url, '${uri.origin}${uri.path}');
final parameters = uri.queryParameters;
await _applyStringParameter(parameters, Preferences.id);
await _applyStringParameter(parameters, Preferences.accuracy);
await _applyIntParameter(parameters, Preferences.distance);
await _applyIntParameter(parameters, Preferences.interval);
await _applyIntParameter(parameters, Preferences.angle);
await _applyIntParameter(parameters, Preferences.heartbeat);
await _applyIntParameter(parameters, Preferences.fastestInterval);
await _applyBoolParameter(parameters, Preferences.buffer);
await _applyBoolParameter(parameters, Preferences.wakelock);
await _applyBoolParameter(parameters, Preferences.stopDetection);
await bg.BackgroundGeolocation.setConfig(Preferences.geolocationConfig());
}
Future<void> _applyStringParameter(Map<String, String> parameters, String key) async {
final value = parameters[key];
if (value != null) {
await Preferences.instance.setString(key, value);
}
}
Future<void> _applyIntParameter(Map<String, String> parameters, String key) async {
final stringValue = parameters[key];
if (stringValue != null) {
final value = int.tryParse(stringValue);
if (value != null) {
await Preferences.instance.setInt(key, value);
}
}
}
Future<void> _applyBoolParameter(Map<String, String> parameters, String key) async {
final value = parameters[key];
if (value != null) {
switch (value) {
case 'false':
await Preferences.instance.setBool(key, false);
break;
case 'true':
await Preferences.instance.setBool(key, true);
break;
}
}
}
void _onDetect(BarcodeCapture capture) async {
if (_scanned) return;
final barcode = capture.barcodes.first;
final rawValue = barcode.rawValue;
if (rawValue == null) return;
final uri = Uri.tryParse(rawValue);
if (uri == null || uri.scheme.isEmpty) return;
_scanned = true;
await _applySettings(uri);
if (mounted) Navigator.pop(context);
}
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text(AppLocalizations.of(context)!.settingsTitle),
),
body: MobileScanner(
fit: BoxFit.cover,
onDetect: _onDetect,
),
);
}
}

View file

@ -3,6 +3,7 @@ import 'dart:io';
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:flutter_background_geolocation/flutter_background_geolocation.dart' as bg;
import 'package:traccar_client/qr_code_screen.dart';
import 'package:wakelock_partial_android/wakelock_partial_android.dart';
import 'l10n/app_localizations.dart';
@ -17,23 +18,6 @@ class SettingsScreen extends StatefulWidget {
class _SettingsScreenState extends State<SettingsScreen> {
bool advanced = false;
bool buffering = true;
bool wakelock = false;
bool stopDetection = true;
@override
void initState() {
super.initState();
_initState();
}
void _initState() async {
setState(() {
buffering = Preferences.instance.getBool(Preferences.buffer) ?? true;
wakelock = Preferences.instance.getBool(Preferences.wakelock) ?? false;
stopDetection = Preferences.instance.getBool(Preferences.stopDetection) ?? true;
});
}
String _getAccuracyLabel(String? key) {
return switch (key) {
@ -148,7 +132,18 @@ class _SettingsScreenState extends State<SettingsScreen> {
final isHighestAccuracy = Preferences.instance.getString(Preferences.accuracy) == 'highest';
final distance = Preferences.instance.getInt(Preferences.distance);
return Scaffold(
appBar: AppBar(title: Text(AppLocalizations.of(context)!.settingsTitle)),
appBar: AppBar(
title: Text(AppLocalizations.of(context)!.settingsTitle),
actions: [
IconButton(
icon: const Icon(Icons.qr_code_scanner),
onPressed: () async {
await Navigator.push(context, MaterialPageRoute(builder: (_) => const QrCodeScreen()));
setState(() {});
},
),
],
),
body: ListView(
children: [
_buildListTile(AppLocalizations.of(context)!.idLabel, Preferences.id, false),
@ -172,17 +167,17 @@ class _SettingsScreenState extends State<SettingsScreen> {
if (advanced)
SwitchListTile(
title: Text(AppLocalizations.of(context)!.bufferLabel),
value: buffering,
value: Preferences.instance.getBool(Preferences.buffer) ?? true,
onChanged: (value) async {
await Preferences.instance.setBool(Preferences.buffer, value);
await bg.BackgroundGeolocation.setConfig(Preferences.geolocationConfig());
setState(() => buffering = value);
setState(() {});
},
),
if (advanced && Platform.isAndroid)
SwitchListTile(
title: Text(AppLocalizations.of(context)!.wakelockLabel),
value: wakelock,
value: Preferences.instance.getBool(Preferences.wakelock) ?? false,
onChanged: (value) async {
await Preferences.instance.setBool(Preferences.wakelock, value);
if (value) {
@ -193,17 +188,17 @@ class _SettingsScreenState extends State<SettingsScreen> {
} else {
WakelockPartialAndroid.release();
}
setState(() => wakelock = value);
setState(() {});
},
),
if (advanced)
SwitchListTile(
title: Text(AppLocalizations.of(context)!.stopDetectionLabel),
value: stopDetection,
value: Preferences.instance.getBool(Preferences.stopDetection) ?? true,
onChanged: (value) async {
await Preferences.instance.setBool(Preferences.stopDetection, value);
await bg.BackgroundGeolocation.setConfig(Preferences.geolocationConfig());
setState(() => stopDetection = value);
setState(() {});
},
),
],