Implement app link support (fix #39)
This commit is contained in:
parent
e508f50054
commit
66155fb273
7 changed files with 136 additions and 50 deletions
|
|
@ -29,6 +29,12 @@
|
|||
<action android:name="android.intent.action.MAIN"/>
|
||||
<category android:name="android.intent.category.LAUNCHER"/>
|
||||
</intent-filter>
|
||||
<intent-filter>
|
||||
<action android:name="android.intent.action.VIEW"/>
|
||||
<category android:name="android.intent.category.DEFAULT"/>
|
||||
<category android:name="android.intent.category.BROWSABLE"/>
|
||||
<data android:scheme="org.traccar.client"/>
|
||||
</intent-filter>
|
||||
</activity>
|
||||
<!-- Don't delete the meta-data below.
|
||||
This is used by the Flutter tool to generate GeneratedPluginRegistrant.java -->
|
||||
|
|
|
|||
|
|
@ -28,6 +28,15 @@
|
|||
<string>????</string>
|
||||
<key>CFBundleVersion</key>
|
||||
<string>$(FLUTTER_BUILD_NUMBER)</string>
|
||||
<key>CFBundleURLTypes</key>
|
||||
<array>
|
||||
<dict>
|
||||
<key>CFBundleURLSchemes</key>
|
||||
<array>
|
||||
<string>org.traccar.client</string>
|
||||
</array>
|
||||
</dict>
|
||||
</array>
|
||||
<key>LSRequiresIPhoneOS</key>
|
||||
<true/>
|
||||
<key>NSAppTransportSecurity</key>
|
||||
|
|
|
|||
63
lib/configuration_service.dart
Normal file
63
lib/configuration_service.dart
Normal file
|
|
@ -0,0 +1,63 @@
|
|||
import 'package:flutter_background_geolocation/flutter_background_geolocation.dart' as bg;
|
||||
|
||||
import 'preferences.dart';
|
||||
|
||||
class ConfigurationService {
|
||||
static Future<void> applyUri(Uri uri) async {
|
||||
if (uri.scheme == 'http' || uri.scheme == 'https') {
|
||||
await Preferences.instance.setString(Preferences.url, '${uri.origin}${uri.path}');
|
||||
} else {
|
||||
final url = uri.queryParameters['url'];
|
||||
if (url != null) {
|
||||
await Preferences.instance.setString(Preferences.url, url);
|
||||
}
|
||||
}
|
||||
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());
|
||||
}
|
||||
|
||||
static Future<void> _applyStringParameter(
|
||||
Map<String, String> parameters, String key) async {
|
||||
final value = parameters[key];
|
||||
if (value != null) {
|
||||
await Preferences.instance.setString(key, value);
|
||||
}
|
||||
}
|
||||
|
||||
static 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);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static 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;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
|
@ -1,8 +1,10 @@
|
|||
import 'dart:async';
|
||||
import 'dart:developer' as developer;
|
||||
|
||||
import 'package:firebase_core/firebase_core.dart';
|
||||
import 'package:firebase_crashlytics/firebase_crashlytics.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:app_links/app_links.dart';
|
||||
import 'package:rate_my_app/rate_my_app.dart';
|
||||
import 'package:traccar_client/geolocation_service.dart';
|
||||
import 'package:traccar_client/push_service.dart';
|
||||
|
|
@ -11,6 +13,7 @@ import 'package:traccar_client/quick_actions.dart';
|
|||
import 'l10n/app_localizations.dart';
|
||||
import 'main_screen.dart';
|
||||
import 'preferences.dart';
|
||||
import 'configuration_service.dart';
|
||||
|
||||
final messengerKey = GlobalKey<ScaffoldMessengerState>();
|
||||
|
||||
|
|
@ -38,6 +41,7 @@ class _MainAppState extends State<MainApp> {
|
|||
@override
|
||||
void initState() {
|
||||
super.initState();
|
||||
_initLinks();
|
||||
WidgetsBinding.instance.addPostFrameCallback((_) async {
|
||||
await rateMyApp.init();
|
||||
if (mounted && rateMyApp.shouldOpenDialog) {
|
||||
|
|
@ -50,6 +54,17 @@ class _MainAppState extends State<MainApp> {
|
|||
});
|
||||
}
|
||||
|
||||
Future<void> _initLinks() async {
|
||||
final appLinks = AppLinks();
|
||||
final uri = await appLinks.getInitialLink();
|
||||
if (uri != null) {
|
||||
await ConfigurationService.applyUri(uri);
|
||||
}
|
||||
appLinks.uriLinkStream.listen((uri) async {
|
||||
await ConfigurationService.applyUri(uri);
|
||||
});
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return MaterialApp(
|
||||
|
|
|
|||
|
|
@ -1,8 +1,7 @@
|
|||
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 'configuration_service.dart';
|
||||
import 'l10n/app_localizations.dart';
|
||||
|
||||
class QrCodeScreen extends StatefulWidget {
|
||||
|
|
@ -15,53 +14,6 @@ class QrCodeScreen extends StatefulWidget {
|
|||
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;
|
||||
|
|
@ -70,7 +22,7 @@ class _QrCodeScreenState extends State<QrCodeScreen> {
|
|||
final uri = Uri.tryParse(rawValue);
|
||||
if (uri == null || uri.scheme.isEmpty) return;
|
||||
_scanned = true;
|
||||
await _applySettings(uri);
|
||||
await ConfigurationService.applyUri(uri);
|
||||
if (mounted) Navigator.pop(context);
|
||||
}
|
||||
|
||||
|
|
|
|||
40
pubspec.lock
40
pubspec.lock
|
|
@ -9,6 +9,38 @@ packages:
|
|||
url: "https://pub.dev"
|
||||
source: hosted
|
||||
version: "1.3.60"
|
||||
app_links:
|
||||
dependency: "direct main"
|
||||
description:
|
||||
name: app_links
|
||||
sha256: "5f88447519add627fe1cbcab4fd1da3d4fed15b9baf29f28b22535c95ecee3e8"
|
||||
url: "https://pub.dev"
|
||||
source: hosted
|
||||
version: "6.4.1"
|
||||
app_links_linux:
|
||||
dependency: transitive
|
||||
description:
|
||||
name: app_links_linux
|
||||
sha256: f5f7173a78609f3dfd4c2ff2c95bd559ab43c80a87dc6a095921d96c05688c81
|
||||
url: "https://pub.dev"
|
||||
source: hosted
|
||||
version: "1.0.3"
|
||||
app_links_platform_interface:
|
||||
dependency: transitive
|
||||
description:
|
||||
name: app_links_platform_interface
|
||||
sha256: "05f5379577c513b534a29ddea68176a4d4802c46180ee8e2e966257158772a3f"
|
||||
url: "https://pub.dev"
|
||||
source: hosted
|
||||
version: "2.0.2"
|
||||
app_links_web:
|
||||
dependency: transitive
|
||||
description:
|
||||
name: app_links_web
|
||||
sha256: af060ed76183f9e2b87510a9480e56a5352b6c249778d07bd2c95fc35632a555
|
||||
url: "https://pub.dev"
|
||||
source: hosted
|
||||
version: "1.0.4"
|
||||
async:
|
||||
dependency: transitive
|
||||
description:
|
||||
|
|
@ -261,6 +293,14 @@ packages:
|
|||
description: flutter
|
||||
source: sdk
|
||||
version: "0.0.0"
|
||||
gtk:
|
||||
dependency: transitive
|
||||
description:
|
||||
name: gtk
|
||||
sha256: e8ce9ca4b1df106e4d72dad201d345ea1a036cc12c360f1a7d5a758f78ffa42c
|
||||
url: "https://pub.dev"
|
||||
source: hosted
|
||||
version: "2.1.0"
|
||||
intl:
|
||||
dependency: "direct main"
|
||||
description:
|
||||
|
|
|
|||
|
|
@ -24,6 +24,7 @@ dependencies:
|
|||
wakelock_partial_android: ^1.0.1
|
||||
mobile_scanner: ^7.0.1
|
||||
flutter_secure_storage: ^9.2.4
|
||||
app_links: ^6.4.1
|
||||
|
||||
dev_dependencies:
|
||||
flutter_test:
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue