SharedMapWidget és térkép nézet refraktorálása
This commit is contained in:
parent
f2457817b2
commit
d50a324e44
@ -7,6 +7,7 @@ import 'package:terepi_seged/routes/app_pages.dart';
|
||||
import 'package:terepi_seged/services/coord_converter_service.dart';
|
||||
import 'package:terepi_seged/services/gnss/gnss_device_service.dart';
|
||||
import 'package:terepi_seged/services/gnss/gnss_service.dart';
|
||||
import 'package:terepi_seged/services/ntrip_service.dart';
|
||||
|
||||
Future<void> main() async {
|
||||
WidgetsFlutterBinding.ensureInitialized();
|
||||
@ -21,6 +22,7 @@ Future<void> main() async {
|
||||
() => CoordConverterService().init());
|
||||
Get.put(GnssDeviceService());
|
||||
Get.put(GnssService());
|
||||
Get.put(NtripService());
|
||||
|
||||
runApp(const MyApp());
|
||||
}
|
||||
|
||||
File diff suppressed because it is too large
Load Diff
@ -84,6 +84,9 @@ class _StakeoutPanel extends GetView<MapSurveyController> {
|
||||
padding: const EdgeInsets.all(12),
|
||||
child: Obx(() {
|
||||
final onTarget = controller.isOnTarget;
|
||||
final dy = controller.deltaY;
|
||||
final dx = controller.deltaX;
|
||||
final dist = controller.distanceToTarget;
|
||||
return Column(
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
children: [
|
||||
@ -108,17 +111,17 @@ class _StakeoutPanel extends GetView<MapSurveyController> {
|
||||
children: [
|
||||
_DeltaCell(
|
||||
label: 'ΔY',
|
||||
value: controller.deltaY,
|
||||
value: dy,
|
||||
unit: 'm',
|
||||
),
|
||||
_DeltaCell(
|
||||
label: 'ΔX',
|
||||
value: controller.deltaX,
|
||||
value: dx,
|
||||
unit: 'm',
|
||||
),
|
||||
_DeltaCell(
|
||||
label: 'Táv',
|
||||
value: controller.distanceToTarget,
|
||||
value: dist,
|
||||
unit: 'm',
|
||||
alwaysPositive: true,
|
||||
),
|
||||
|
||||
@ -1,6 +1,7 @@
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:get/get.dart';
|
||||
import 'package:terepi_seged/pages/map_survey/presentations/controllers/map_survey_controller.dart';
|
||||
import 'package:terepi_seged/services/ntrip_service.dart';
|
||||
|
||||
class SettingsDialog extends StatelessWidget {
|
||||
final controller = Get.find<MapSurveyController>();
|
||||
@ -17,19 +18,19 @@ class SettingsDialog extends StatelessWidget {
|
||||
children: [
|
||||
IconButton(
|
||||
onPressed: () {
|
||||
if (controller.ntripUsernameController.text.isNotEmpty) {
|
||||
controller.ntripUserName.value =
|
||||
controller.ntripUsernameController.text;
|
||||
controller.saveNtripUserName(
|
||||
controller.ntripUsernameController.text);
|
||||
if (controller
|
||||
.ntripPasswordController.text.isNotEmpty) {
|
||||
controller.ntripPassword.value =
|
||||
controller.ntripPasswordController.text;
|
||||
controller.saveNtripPassword(
|
||||
controller.ntripPasswordController.text);
|
||||
}
|
||||
}
|
||||
// if (controller.ntripUsernameController.text.isNotEmpty) {
|
||||
// controller.ntripUserName.value =
|
||||
// controller.ntripUsernameController.text;
|
||||
// controller.saveNtripUserName(
|
||||
// controller.ntripUsernameController.text);
|
||||
// if (controller
|
||||
// .ntripPasswordController.text.isNotEmpty) {
|
||||
// controller.ntripPassword.value =
|
||||
// controller.ntripPasswordController.text;
|
||||
// controller.saveNtripPassword(
|
||||
// controller.ntripPasswordController.text);
|
||||
// }
|
||||
// }
|
||||
Get.back();
|
||||
},
|
||||
icon: const Icon(Icons.close)),
|
||||
@ -38,19 +39,19 @@ class SettingsDialog extends StatelessWidget {
|
||||
overlayColor:
|
||||
MaterialStateProperty.all(Colors.transparent)),
|
||||
onPressed: () {
|
||||
if (controller.ntripUsernameController.text.isNotEmpty) {
|
||||
controller.ntripUserName.value =
|
||||
controller.ntripUsernameController.text;
|
||||
controller.saveNtripUserName(
|
||||
controller.ntripUsernameController.text);
|
||||
if (controller
|
||||
.ntripPasswordController.text.isNotEmpty) {
|
||||
controller.ntripPassword.value =
|
||||
controller.ntripPasswordController.text;
|
||||
controller.saveNtripPassword(
|
||||
controller.ntripPasswordController.text);
|
||||
}
|
||||
}
|
||||
// if (controller.ntripUsernameController.text.isNotEmpty) {
|
||||
// controller.ntripUserName.value =
|
||||
// controller.ntripUsernameController.text;
|
||||
// controller.saveNtripUserName(
|
||||
// controller.ntripUsernameController.text);
|
||||
// if (controller
|
||||
// .ntripPasswordController.text.isNotEmpty) {
|
||||
// controller.ntripPassword.value =
|
||||
// controller.ntripPasswordController.text;
|
||||
// controller.saveNtripPassword(
|
||||
// controller.ntripPasswordController.text);
|
||||
// }
|
||||
// }
|
||||
Get.back();
|
||||
},
|
||||
child: const Text(
|
||||
@ -74,118 +75,118 @@ class SettingsDialog extends StatelessWidget {
|
||||
style: TextStyle(fontSize: 14.0, fontWeight: FontWeight.w500),
|
||||
),
|
||||
),
|
||||
Obx(() => Column(children: [
|
||||
RadioListTile(
|
||||
title: Text('TiGNSS Rover-BE6A'),
|
||||
value: '10:06:1C:97:BE:6A',
|
||||
groupValue: controller.gpsAddress.value,
|
||||
onChanged: (value) {
|
||||
controller.gpsAddress.value = value!;
|
||||
controller.gpsName.value = 'TiGNSS Rover-BE6A';
|
||||
controller.saveGpsAddress(value);
|
||||
controller.saveGpsName('TiGNSS Rover-BE6A');
|
||||
}),
|
||||
RadioListTile(
|
||||
title: Text('TiGNSS Rover-1DC6'),
|
||||
value: 'E8:31:CD:16:1D:C6',
|
||||
groupValue: controller.gpsAddress.value,
|
||||
onChanged: (value) {
|
||||
controller.gpsAddress.value = value!;
|
||||
controller.gpsName.value = 'TiGNSS Rover-1DC6';
|
||||
controller.saveGpsAddress(value);
|
||||
controller.saveGpsName('TiGNSS Rover-1DC6');
|
||||
}),
|
||||
RadioListTile(
|
||||
title: Text('TiGNSS Rover-9C3A'),
|
||||
value: '08:3A:8D:14:9C:3A',
|
||||
groupValue: controller.gpsAddress.value,
|
||||
onChanged: (value) {
|
||||
controller.gpsAddress.value = value!;
|
||||
controller.gpsName.value = 'TiGNSS Rover-9C3A';
|
||||
controller.saveGpsAddress(value);
|
||||
controller.saveGpsName('TiGNSS Rover-9C3A');
|
||||
}),
|
||||
RadioListTile(
|
||||
title: Text('TiGNSS Rover-72C2'),
|
||||
value: '10:06:1C:97:72:C2',
|
||||
groupValue: controller.gpsAddress.value,
|
||||
onChanged: (value) {
|
||||
controller.gpsAddress.value = value!;
|
||||
controller.gpsName.value = 'TiGNSS Rover-72C2';
|
||||
controller.saveGpsAddress(value);
|
||||
controller.saveGpsName('TiGNSS Rover-72C2');
|
||||
}),
|
||||
RadioListTile(
|
||||
title: Text('TiGNSS Rover-FE16'),
|
||||
value: '10:06:1C:9F:FE:16',
|
||||
groupValue: controller.gpsAddress.value,
|
||||
onChanged: (value) {
|
||||
controller.gpsAddress.value = value!;
|
||||
controller.gpsName.value = 'TiGNSS Rover-FE16';
|
||||
controller.saveGpsAddress(value);
|
||||
controller.saveGpsName('TiGNSS Rover-FE16');
|
||||
}),
|
||||
RadioListTile(
|
||||
title: Text('TiGNSS Rover-3B0A'),
|
||||
value: '10:C6:1C:9E:3B:0A',
|
||||
groupValue: controller.gpsAddress.value,
|
||||
onChanged: (value) {
|
||||
controller.gpsAddress.value = value!;
|
||||
controller.gpsName.value = 'TiGNSS Rover-3B0A';
|
||||
controller.saveGpsAddress(value);
|
||||
controller.saveGpsName('TiGNSS Rover-3B0A');
|
||||
}),
|
||||
RadioListTile(
|
||||
title: Text('TiGNSS Rover-7FEA'),
|
||||
value: '10:06:1C:9C:7F:EA',
|
||||
groupValue: controller.gpsAddress.value,
|
||||
onChanged: (value) {
|
||||
controller.gpsAddress.value = value!;
|
||||
controller.gpsName.value = 'TiGNSS Rover-7FEA';
|
||||
controller.saveGpsAddress(value);
|
||||
controller.saveGpsName('TiGNSS Rover-7FEA');
|
||||
}),
|
||||
RadioListTile(
|
||||
title: Text('TiGNSS Rover-A39E'),
|
||||
value: '10:06:1C:97:A3:9E',
|
||||
groupValue: controller.gpsAddress.value,
|
||||
onChanged: (value) {
|
||||
controller.gpsAddress.value = value!;
|
||||
controller.gpsName.value = 'TiGNSS Rover-A39E';
|
||||
controller.saveGpsAddress(value);
|
||||
controller.saveGpsName('TiGNSS Rover-A39E');
|
||||
}),
|
||||
RadioListTile(
|
||||
title: Text('TiGNSS Rover-FF4E'),
|
||||
value: '98:CD:AC:62:FF:4E',
|
||||
groupValue: controller.gpsAddress.value,
|
||||
onChanged: (value) {
|
||||
controller.gpsAddress.value = value!;
|
||||
controller.gpsName.value = 'TiGNSS Rover-FF4E';
|
||||
controller.saveGpsAddress(value);
|
||||
controller.saveGpsName('TiGNSS Rover-FF4E');
|
||||
}),
|
||||
RadioListTile(
|
||||
title: Text('TiGNSS Rover-8BB2'),
|
||||
value: 'E8:31:CD:14:8B:B2',
|
||||
groupValue: controller.gpsAddress.value,
|
||||
onChanged: (value) {
|
||||
controller.gpsAddress.value = value!;
|
||||
controller.gpsName.value = 'TiGNSS Rover-8BB2';
|
||||
controller.saveGpsAddress(value);
|
||||
controller.saveGpsName('TiGNSS Rover-8BB2');
|
||||
}),
|
||||
RadioListTile(
|
||||
title: Text('TiGNSS Rover-FF36'),
|
||||
value: '98:CD:AC:62:FF:36',
|
||||
groupValue: controller.gpsAddress.value,
|
||||
onChanged: (value) {
|
||||
controller.gpsAddress.value = value!;
|
||||
controller.gpsName.value = 'TiGNSS Rover-FF36';
|
||||
controller.saveGpsAddress(value);
|
||||
controller.saveGpsName('TiGNSS Rover-FF36');
|
||||
})
|
||||
])),
|
||||
// Obx(() => Column(children: [
|
||||
// RadioListTile(
|
||||
// title: Text('TiGNSS Rover-BE6A'),
|
||||
// value: '10:06:1C:97:BE:6A',
|
||||
// groupValue: controller.gpsAddress.value,
|
||||
// onChanged: (value) {
|
||||
// controller.gpsAddress.value = value!;
|
||||
// controller.gpsName.value = 'TiGNSS Rover-BE6A';
|
||||
// controller.saveGpsAddress(value);
|
||||
// controller.saveGpsName('TiGNSS Rover-BE6A');
|
||||
// }),
|
||||
// RadioListTile(
|
||||
// title: Text('TiGNSS Rover-1DC6'),
|
||||
// value: 'E8:31:CD:16:1D:C6',
|
||||
// groupValue: controller.gpsAddress.value,
|
||||
// onChanged: (value) {
|
||||
// controller.gpsAddress.value = value!;
|
||||
// controller.gpsName.value = 'TiGNSS Rover-1DC6';
|
||||
// controller.saveGpsAddress(value);
|
||||
// controller.saveGpsName('TiGNSS Rover-1DC6');
|
||||
// }),
|
||||
// RadioListTile(
|
||||
// title: Text('TiGNSS Rover-9C3A'),
|
||||
// value: '08:3A:8D:14:9C:3A',
|
||||
// groupValue: controller.gpsAddress.value,
|
||||
// onChanged: (value) {
|
||||
// controller.gpsAddress.value = value!;
|
||||
// controller.gpsName.value = 'TiGNSS Rover-9C3A';
|
||||
// controller.saveGpsAddress(value);
|
||||
// controller.saveGpsName('TiGNSS Rover-9C3A');
|
||||
// }),
|
||||
// RadioListTile(
|
||||
// title: Text('TiGNSS Rover-72C2'),
|
||||
// value: '10:06:1C:97:72:C2',
|
||||
// groupValue: controller.gpsAddress.value,
|
||||
// onChanged: (value) {
|
||||
// controller.gpsAddress.value = value!;
|
||||
// controller.gpsName.value = 'TiGNSS Rover-72C2';
|
||||
// controller.saveGpsAddress(value);
|
||||
// controller.saveGpsName('TiGNSS Rover-72C2');
|
||||
// }),
|
||||
// RadioListTile(
|
||||
// title: Text('TiGNSS Rover-FE16'),
|
||||
// value: '10:06:1C:9F:FE:16',
|
||||
// groupValue: controller.gpsAddress.value,
|
||||
// onChanged: (value) {
|
||||
// controller.gpsAddress.value = value!;
|
||||
// controller.gpsName.value = 'TiGNSS Rover-FE16';
|
||||
// controller.saveGpsAddress(value);
|
||||
// controller.saveGpsName('TiGNSS Rover-FE16');
|
||||
// }),
|
||||
// RadioListTile(
|
||||
// title: Text('TiGNSS Rover-3B0A'),
|
||||
// value: '10:C6:1C:9E:3B:0A',
|
||||
// groupValue: controller.gpsAddress.value,
|
||||
// onChanged: (value) {
|
||||
// controller.gpsAddress.value = value!;
|
||||
// controller.gpsName.value = 'TiGNSS Rover-3B0A';
|
||||
// controller.saveGpsAddress(value);
|
||||
// controller.saveGpsName('TiGNSS Rover-3B0A');
|
||||
// }),
|
||||
// RadioListTile(
|
||||
// title: Text('TiGNSS Rover-7FEA'),
|
||||
// value: '10:06:1C:9C:7F:EA',
|
||||
// groupValue: controller.gpsAddress.value,
|
||||
// onChanged: (value) {
|
||||
// controller.gpsAddress.value = value!;
|
||||
// controller.gpsName.value = 'TiGNSS Rover-7FEA';
|
||||
// controller.saveGpsAddress(value);
|
||||
// controller.saveGpsName('TiGNSS Rover-7FEA');
|
||||
// }),
|
||||
// RadioListTile(
|
||||
// title: Text('TiGNSS Rover-A39E'),
|
||||
// value: '10:06:1C:97:A3:9E',
|
||||
// groupValue: controller.gpsAddress.value,
|
||||
// onChanged: (value) {
|
||||
// controller.gpsAddress.value = value!;
|
||||
// controller.gpsName.value = 'TiGNSS Rover-A39E';
|
||||
// controller.saveGpsAddress(value);
|
||||
// controller.saveGpsName('TiGNSS Rover-A39E');
|
||||
// }),
|
||||
// RadioListTile(
|
||||
// title: Text('TiGNSS Rover-FF4E'),
|
||||
// value: '98:CD:AC:62:FF:4E',
|
||||
// groupValue: controller.gpsAddress.value,
|
||||
// onChanged: (value) {
|
||||
// controller.gpsAddress.value = value!;
|
||||
// controller.gpsName.value = 'TiGNSS Rover-FF4E';
|
||||
// controller.saveGpsAddress(value);
|
||||
// controller.saveGpsName('TiGNSS Rover-FF4E');
|
||||
// }),
|
||||
// RadioListTile(
|
||||
// title: Text('TiGNSS Rover-8BB2'),
|
||||
// value: 'E8:31:CD:14:8B:B2',
|
||||
// groupValue: controller.gpsAddress.value,
|
||||
// onChanged: (value) {
|
||||
// controller.gpsAddress.value = value!;
|
||||
// controller.gpsName.value = 'TiGNSS Rover-8BB2';
|
||||
// controller.saveGpsAddress(value);
|
||||
// controller.saveGpsName('TiGNSS Rover-8BB2');
|
||||
// }),
|
||||
// RadioListTile(
|
||||
// title: Text('TiGNSS Rover-FF36'),
|
||||
// value: '98:CD:AC:62:FF:36',
|
||||
// groupValue: controller.gpsAddress.value,
|
||||
// onChanged: (value) {
|
||||
// controller.gpsAddress.value = value!;
|
||||
// controller.gpsName.value = 'TiGNSS Rover-FF36';
|
||||
// controller.saveGpsAddress(value);
|
||||
// controller.saveGpsName('TiGNSS Rover-FF36');
|
||||
// })
|
||||
// ])),
|
||||
const Padding(
|
||||
padding: EdgeInsets.symmetric(horizontal: 20),
|
||||
child: Divider(
|
||||
@ -329,7 +330,7 @@ class SettingsDialog extends StatelessWidget {
|
||||
SizedBox(
|
||||
height: 40,
|
||||
child: TextField(
|
||||
controller: controller.ntripUsernameController,
|
||||
controller: NtripService.to.usernameController,
|
||||
enableSuggestions: false,
|
||||
autocorrect: false,
|
||||
decoration: InputDecoration(
|
||||
@ -351,10 +352,10 @@ class SettingsDialog extends StatelessWidget {
|
||||
child: TextField(
|
||||
keyboardType: TextInputType.visiblePassword,
|
||||
obscureText: controller.isShowPassword.value,
|
||||
focusNode: controller.passwordFieldFocusNode,
|
||||
//focusNode: controller.passwordFieldFocusNode,
|
||||
enableSuggestions: false,
|
||||
autocorrect: false,
|
||||
controller: controller.ntripPasswordController,
|
||||
controller: NtripService.to.passwordController,
|
||||
decoration: InputDecoration(
|
||||
floatingLabelBehavior: FloatingLabelBehavior.never,
|
||||
isDense: true,
|
||||
@ -371,7 +372,7 @@ class SettingsDialog extends StatelessWidget {
|
||||
suffixIcon: Padding(
|
||||
padding: const EdgeInsets.fromLTRB(0, 0, 4, 0),
|
||||
child: GestureDetector(
|
||||
onTap: controller.toggleShowPassword,
|
||||
//onTap: controller.toggleShowPassword,
|
||||
child: Icon(
|
||||
controller.isShowPassword.value
|
||||
? Icons.visibility_rounded
|
||||
|
||||
@ -11,7 +11,11 @@ class ShellBinding extends Bindings {
|
||||
// TODO: implement dependencies
|
||||
Get.put(ShellController());
|
||||
Get.put(HomeViewController());
|
||||
Get.put(MapViewController());
|
||||
Get.put(MapSurveyController());
|
||||
// Get.put(MapViewController());
|
||||
// Get.put(MapSurveyController());
|
||||
Get.lazyPut<MapSurveyController>(
|
||||
() => MapSurveyController(),
|
||||
fenix: true,
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
@ -2,6 +2,7 @@ import 'package:flutter/material.dart';
|
||||
import 'package:get/get.dart';
|
||||
import 'package:terepi_seged/pages/home/presentation/views/home_view.dart';
|
||||
import 'package:terepi_seged/pages/map_survey/presentations/controllers/map_survey_controller.dart';
|
||||
import 'package:terepi_seged/services/ntrip_service.dart';
|
||||
import 'package:terepi_seged/widgets/gnss_status_chip.dart';
|
||||
|
||||
import '../../../../widgets/app_drawer.dart';
|
||||
@ -30,12 +31,11 @@ class ShellView extends GetView<ShellController> {
|
||||
const SizedBox(width: 6),
|
||||
Obx(() => controller.currentIndex.value == 1
|
||||
? NtripStatusChip(
|
||||
isConnected: MapSurveyController.to.ntripIsConnected,
|
||||
isConnected: NtripService.to.isConnected,
|
||||
onToggle: () {
|
||||
final c = MapSurveyController.to;
|
||||
c.ntripIsConnected.value
|
||||
? c.disconnectFromNtripServer()
|
||||
: c.connectToNtripServer();
|
||||
NtripService.to.isConnected.value
|
||||
? NtripService.to.disconnect()
|
||||
: NtripService.to.connect();
|
||||
},
|
||||
)
|
||||
: const SizedBox.shrink())
|
||||
|
||||
@ -92,4 +92,8 @@ class BtSerialGnssConnection implements GnssConnection {
|
||||
_nmeaController.close();
|
||||
_stateController.close();
|
||||
}
|
||||
|
||||
void sendData(Uint8List data) {
|
||||
_connection?.output.add(data);
|
||||
}
|
||||
}
|
||||
|
||||
@ -1,43 +1,68 @@
|
||||
// lib/services/gnss/gnss_service.dart
|
||||
import 'dart:async';
|
||||
import 'dart:typed_data';
|
||||
|
||||
import 'package:get/get.dart';
|
||||
import 'package:nmea/nmea.dart';
|
||||
import 'package:terepi_seged/services/gnss/gnss_device_service.dart';
|
||||
|
||||
import '../../gnss_sentences/gngga.dart';
|
||||
import 'gnss_connection.dart';
|
||||
import '../../gnss_sentences/gngst.dart';
|
||||
import '../../gnss_sentences/gnrmc.dart';
|
||||
import 'bt_serial_gnss_connection.dart';
|
||||
import 'ble_gnss_connection.dart';
|
||||
import 'gnss_device_service.dart';
|
||||
import 'gnss_connection.dart';
|
||||
|
||||
class GnssService extends GetxService {
|
||||
static GnssService get to => Get.find();
|
||||
|
||||
GnssConnection? _connection;
|
||||
|
||||
// Reaktív állapot — a controllerek Obx-szel figyelhetik
|
||||
// ── Kapcsolat állapot ─────────────────────────────────────────────
|
||||
final connectionState = GnssConnectionState.disconnected.obs;
|
||||
final activeConnectionType = Rxn<GnssConnectionType>();
|
||||
|
||||
// Parsed NMEA adatok
|
||||
// ── GGA adatok ────────────────────────────────────────────────────
|
||||
final latitude = 0.0.obs;
|
||||
final longitude = 0.0.obs;
|
||||
final altitude = 0.0.obs;
|
||||
final geoidSeparation = 0.0.obs;
|
||||
final altitude = 0.0.obs; // MSL (ortometrikus)
|
||||
final geoidSeparation = 0.0.obs; // N — geoid undulációja
|
||||
final gpsQuality = 0.obs;
|
||||
final utcFix = ''.obs;
|
||||
final satelliteCount = 0.obs;
|
||||
final hdop = 0.0.obs;
|
||||
|
||||
// Utolsó nyers GGA sor — NtripService küldi vissza a casternek
|
||||
final lastGgaLine = ''.obs;
|
||||
|
||||
// ── GST adatok (pontossági hibák) ─────────────────────────────────
|
||||
final latitudeError = 0.0.obs;
|
||||
final longitudeError = 0.0.obs;
|
||||
final altitudeError = 0.0.obs;
|
||||
|
||||
// ── RMC adatok (dátum/idő) ────────────────────────────────────────
|
||||
final gpsDateTime = DateTime(2000).obs;
|
||||
|
||||
// Segédmező: van-e érvényes adat
|
||||
bool get hasValidData => gpsQuality.value > 0;
|
||||
|
||||
// ── Belső ─────────────────────────────────────────────────────────
|
||||
final NmeaDecoder _decoder = NmeaDecoder();
|
||||
StreamSubscription? _nmeaSub;
|
||||
StreamSubscription? _stateSub;
|
||||
String _utcTime = '';
|
||||
String _utcDate = '';
|
||||
|
||||
@override
|
||||
void onInit() {
|
||||
super.onInit();
|
||||
_decoder.registerTalkerSentence('GGA', (l) => Gngga(raw: l));
|
||||
_decoder
|
||||
..registerTalkerSentence('GGA', (l) => Gngga(raw: l))
|
||||
..registerTalkerSentence('GST', (l) => Gngst(raw: l))
|
||||
..registerTalkerSentence('RMC', (l) => Gnrmc(raw: l));
|
||||
}
|
||||
|
||||
// ── Kapcsolódás ──────────────────────────────────────────────────
|
||||
// ── Kapcsolódás ───────────────────────────────────────────────────
|
||||
|
||||
Future<void> connectBtSerial(String macAddress) async {
|
||||
await _disconnect();
|
||||
@ -60,6 +85,27 @@ class GnssService extends GetxService {
|
||||
await _doConnect(deviceId);
|
||||
}
|
||||
|
||||
/// Eszközváltás — GnssDevicePicker hívja.
|
||||
Future<void> onDeviceChanged(GnssDevice device) async {
|
||||
switch (device.type) {
|
||||
case GnssConnectionType.btSerial:
|
||||
await connectBtSerial(device.address);
|
||||
case GnssConnectionType.ble:
|
||||
await connectBle(device.address);
|
||||
case GnssConnectionType.phoneGps:
|
||||
await _disconnect();
|
||||
connectionState.value = GnssConnectionState.disconnected;
|
||||
activeConnectionType.value = GnssConnectionType.phoneGps;
|
||||
}
|
||||
}
|
||||
|
||||
Future<void> reconnect() async {
|
||||
final device = GnssDeviceService.to.selectedDevice.value;
|
||||
if (device == null) return;
|
||||
await _disconnect();
|
||||
await onDeviceChanged(device);
|
||||
}
|
||||
|
||||
Future<void> _doConnect(String address) async {
|
||||
_stateSub = _connection!.connectionState.listen((s) {
|
||||
connectionState.value = s;
|
||||
@ -74,26 +120,74 @@ class GnssService extends GetxService {
|
||||
await _connection?.disconnect();
|
||||
_connection?.dispose();
|
||||
_connection = null;
|
||||
connectionState.value = GnssConnectionState.disconnected;
|
||||
}
|
||||
|
||||
// ── NMEA parsing — egy helyen, nem háromban ──────────────────────
|
||||
Future<void> disconnect() => _disconnect();
|
||||
|
||||
/// RTCM adat továbbítása a GNSS vevőnek (NtripService hívja).
|
||||
void sendToReceiver(Uint8List data) {
|
||||
if (_connection == null) return;
|
||||
if (connectionState.value != GnssConnectionState.connected) return;
|
||||
(_connection as BtSerialGnssConnection?)?.sendData(data);
|
||||
}
|
||||
|
||||
// ── NMEA parsing ──────────────────────────────────────────────────
|
||||
|
||||
void _parseNmea(String line) {
|
||||
if (!line.startsWith('\$GNGGA') && !line.startsWith('\$GPGGA')) return;
|
||||
if (line.startsWith('\$GNGGA') || line.startsWith('\$GPGGA')) {
|
||||
_parseGga(line);
|
||||
} else if (line.startsWith('\$GNGST') && hasValidData) {
|
||||
_parseGst(line);
|
||||
} else if (line.startsWith('\$GNRMC') && hasValidData) {
|
||||
_parseRmc(line);
|
||||
}
|
||||
}
|
||||
|
||||
void _parseGga(String line) {
|
||||
try {
|
||||
final sentence = _decoder.decode(line);
|
||||
if (sentence == null || !sentence.valid || sentence is! Gngga) return;
|
||||
if (sentence.gpsQualityIndicator == 0) return;
|
||||
final s = _decoder.decode(line);
|
||||
if (s == null || !s.valid || s is! Gngga) return;
|
||||
if (s.gpsQualityIndicator == 0) return;
|
||||
|
||||
latitude.value = sentence.latitude;
|
||||
longitude.value = sentence.longitude;
|
||||
altitude.value = sentence.altitudeAboveMeanSeaLevel;
|
||||
geoidSeparation.value = sentence.geoidSeparation;
|
||||
gpsQuality.value = sentence.gpsQualityIndicator;
|
||||
utcFix.value = sentence.utcOfPositionFix;
|
||||
satelliteCount.value = sentence.numberOfSvsInUse;
|
||||
hdop.value = sentence.hdop;
|
||||
latitude.value = s.latitude;
|
||||
longitude.value = s.longitude;
|
||||
altitude.value = s.altitudeAboveMeanSeaLevel;
|
||||
geoidSeparation.value = s.geoidSeparation;
|
||||
gpsQuality.value = s.gpsQualityIndicator;
|
||||
utcFix.value = s.utcOfPositionFix;
|
||||
satelliteCount.value = s.numberOfSvsInUse;
|
||||
hdop.value = s.hdop;
|
||||
lastGgaLine.value = line;
|
||||
_utcTime = s.utcOfPositionFix;
|
||||
} catch (_) {}
|
||||
}
|
||||
|
||||
void _parseGst(String line) {
|
||||
try {
|
||||
final s = _decoder.decode(line);
|
||||
if (s == null || !s.valid || s is! Gngst) return;
|
||||
latitudeError.value = s.latitudeError;
|
||||
longitudeError.value = s.longitudeError;
|
||||
altitudeError.value = s.heightError;
|
||||
} catch (_) {}
|
||||
}
|
||||
|
||||
void _parseRmc(String line) {
|
||||
try {
|
||||
final s = _decoder.decode(line);
|
||||
if (s == null || !s.valid || s is! Gnrmc) return;
|
||||
_utcDate = s.date;
|
||||
if (_utcDate.length >= 6 && _utcTime.length >= 6) {
|
||||
gpsDateTime.value = DateTime(
|
||||
2000 + int.parse('${_utcDate[4]}${_utcDate[5]}'),
|
||||
int.parse('${_utcDate[2]}${_utcDate[3]}'),
|
||||
int.parse('${_utcDate[0]}${_utcDate[1]}'),
|
||||
int.parse('${_utcTime[0]}${_utcTime[1]}'),
|
||||
int.parse('${_utcTime[2]}${_utcTime[3]}'),
|
||||
int.parse('${_utcTime[4]}${_utcTime[5]}'),
|
||||
);
|
||||
}
|
||||
} catch (_) {}
|
||||
}
|
||||
|
||||
@ -102,24 +196,4 @@ class GnssService extends GetxService {
|
||||
_disconnect();
|
||||
super.onClose();
|
||||
}
|
||||
|
||||
// ── Eszközváltás — a GnssDevicePickerDialog hívja ───────────────
|
||||
|
||||
Future<void> onDeviceChanged(GnssDevice device) async {
|
||||
switch (device.type) {
|
||||
case GnssConnectionType.btSerial:
|
||||
await connectBtSerial(device.address);
|
||||
|
||||
case GnssConnectionType.ble:
|
||||
await connectBle(device.address);
|
||||
|
||||
case GnssConnectionType.phoneGps:
|
||||
// Külső GPS kapcsolat lezárása ha volt
|
||||
await _disconnect();
|
||||
connectionState.value = GnssConnectionState.disconnected;
|
||||
activeConnectionType.value = GnssConnectionType.phoneGps;
|
||||
// A telefon GPS kezelése a map_controller _startPhoneGps()-ben van
|
||||
// itt csak jelezzük hogy phone módra váltottunk
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
248
lib/services/ntrip_service.dart
Normal file
248
lib/services/ntrip_service.dart
Normal file
@ -0,0 +1,248 @@
|
||||
// lib/services/ntrip_service.dart
|
||||
|
||||
import 'dart:async';
|
||||
import 'dart:convert';
|
||||
import 'dart:io';
|
||||
import 'dart:typed_data';
|
||||
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:get/get.dart';
|
||||
import 'package:shared_preferences/shared_preferences.dart';
|
||||
|
||||
/// NTRIP kapcsolatot kezelő singleton service.
|
||||
///
|
||||
/// Felelőssége:
|
||||
/// - Socket kapcsolat az NTRIP casterhez
|
||||
/// - RTCM adatok fogadása és továbbítása a GNSS vevőnek
|
||||
/// - GGA mondatok küldése a casternek (5 másodpercenként)
|
||||
/// - Beállítások tárolása SharedPreferences-ben
|
||||
///
|
||||
/// Használat:
|
||||
/// ```dart
|
||||
/// // Csatlakozás előtt add meg a callback-et:
|
||||
/// NtripService.to.onRtcmData = (data) => connection.output.add(data);
|
||||
/// await NtripService.to.connect();
|
||||
/// ```
|
||||
class NtripService extends GetxService {
|
||||
static NtripService get to => Get.find();
|
||||
|
||||
// ── Reaktív állapot ───────────────────────────────────────────────
|
||||
final isConnected = false.obs;
|
||||
final receivedBytes = 0.obs;
|
||||
final packetCount = 0.obs;
|
||||
final ggaSentCount = 0.obs;
|
||||
final ggaLastSentTime = ''.obs;
|
||||
|
||||
// ── Beállítások ───────────────────────────────────────────────────
|
||||
final host = '84.206.45.44'.obs; // gnssnet.hu IP
|
||||
final port = 2101.obs;
|
||||
final mountpoint = 'SGO_RTK3.2'.obs;
|
||||
final username = ''.obs;
|
||||
final password = ''.obs;
|
||||
|
||||
// ── UI controllerek (beállítás dialóghoz) ────────────────────────
|
||||
final hostController = TextEditingController();
|
||||
final portController = TextEditingController();
|
||||
final mountpointController = TextEditingController();
|
||||
final usernameController = TextEditingController();
|
||||
final passwordController = TextEditingController();
|
||||
|
||||
// ── Belső állapot ────────────────────────────────────────────────
|
||||
Socket? _socket;
|
||||
StreamSubscription? _socketSub;
|
||||
String _lastGgaMessage = '';
|
||||
DateTime _lastGgaSentTime =
|
||||
DateTime.now().subtract(const Duration(seconds: 30));
|
||||
|
||||
/// Callback: RTCM adat érkezett → a controller továbbítja a GNSS vevőnek.
|
||||
/// Beállítás: `NtripService.to.onRtcmData = (data) => connection.output.add(data);`
|
||||
Function(Uint8List)? onRtcmData;
|
||||
|
||||
// ── Inicializálás ────────────────────────────────────────────────
|
||||
|
||||
@override
|
||||
Future<void> onInit() async {
|
||||
super.onInit();
|
||||
await _loadSettings();
|
||||
_syncControllersFromValues();
|
||||
}
|
||||
|
||||
@override
|
||||
void onClose() {
|
||||
disconnect();
|
||||
hostController.dispose();
|
||||
portController.dispose();
|
||||
mountpointController.dispose();
|
||||
usernameController.dispose();
|
||||
passwordController.dispose();
|
||||
super.onClose();
|
||||
}
|
||||
|
||||
// ── Kapcsolat ────────────────────────────────────────────────────
|
||||
|
||||
Future<void> connect() async {
|
||||
if (isConnected.value) return;
|
||||
|
||||
try {
|
||||
_socket = await Socket.connect(
|
||||
InternetAddress(host.value),
|
||||
port.value,
|
||||
timeout: const Duration(seconds: 5),
|
||||
);
|
||||
|
||||
_socket!.encoding = ascii;
|
||||
isConnected.value = true;
|
||||
receivedBytes.value = 0;
|
||||
packetCount.value = 0;
|
||||
|
||||
// HTTP fejléc összeállítása
|
||||
final header = _buildNtripHeader();
|
||||
_socket!.add(_toUint8List(header));
|
||||
|
||||
// Adatfogadás
|
||||
_socketSub = _socket!.listen(
|
||||
_onData,
|
||||
onError: _onError,
|
||||
onDone: _onDone,
|
||||
);
|
||||
} catch (e) {
|
||||
isConnected.value = false;
|
||||
Get.snackbar(
|
||||
'NTRIP hiba',
|
||||
'Nem sikerült csatlakozni: $e',
|
||||
backgroundColor: const Color(0xFFB71C1C),
|
||||
colorText: const Color(0xFFFFFFFF),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
Future<void> disconnect() async {
|
||||
if (!isConnected.value && _socket == null) return;
|
||||
await _socketSub?.cancel();
|
||||
await _socket?.flush();
|
||||
_socket?.close();
|
||||
_socket?.destroy();
|
||||
_socket = null;
|
||||
isConnected.value = false;
|
||||
receivedBytes.value = 0;
|
||||
}
|
||||
|
||||
void reconnect() async {
|
||||
await disconnect();
|
||||
await Future.delayed(const Duration(seconds: 1));
|
||||
await connect();
|
||||
}
|
||||
|
||||
// ── GGA küldés ───────────────────────────────────────────────────
|
||||
|
||||
/// Az NMEA feldolgozó hívja minden GGA mondatnál.
|
||||
/// 5 másodpercenként küld egyet az NTRIP casternek.
|
||||
void onGgaReceived(String ggaLine, String utcTime) {
|
||||
_lastGgaMessage = ggaLine;
|
||||
|
||||
if (!isConnected.value) return;
|
||||
if (ggaLine.isEmpty) return;
|
||||
|
||||
final elapsed = DateTime.now().difference(_lastGgaSentTime).inSeconds;
|
||||
if (elapsed < 5) return;
|
||||
|
||||
_sendGga(ggaLine);
|
||||
ggaSentCount.value++;
|
||||
ggaLastSentTime.value = utcTime;
|
||||
_lastGgaSentTime = DateTime.now();
|
||||
}
|
||||
|
||||
void _sendGga(String ggaMessage) {
|
||||
if (_socket == null || !isConnected.value) return;
|
||||
_socket!.add(_toUint8List('$ggaMessage\r\n'));
|
||||
}
|
||||
|
||||
// ── Belső adatfogadás ────────────────────────────────────────────
|
||||
|
||||
void _onData(Uint8List data) {
|
||||
receivedBytes.value = data.length;
|
||||
packetCount.value++;
|
||||
|
||||
// Csak RTCM adat (>14 byte) kerül a GNSS vevőhöz
|
||||
if (data.length > 14) {
|
||||
onRtcmData?.call(data);
|
||||
}
|
||||
}
|
||||
|
||||
void _onError(dynamic error) {
|
||||
_socket?.destroy();
|
||||
isConnected.value = false;
|
||||
Get.snackbar(
|
||||
'NTRIP kapcsolat hiba',
|
||||
error.toString(),
|
||||
backgroundColor: const Color(0xFFB71C1C),
|
||||
colorText: const Color(0xFFFFFFFF),
|
||||
);
|
||||
}
|
||||
|
||||
void _onDone() async {
|
||||
await _socketSub?.cancel();
|
||||
await _socket?.flush();
|
||||
_socket?.destroy();
|
||||
_socket = null;
|
||||
isConnected.value = false;
|
||||
receivedBytes.value = 0;
|
||||
}
|
||||
|
||||
// ── HTTP fejléc összeállítás ─────────────────────────────────────
|
||||
|
||||
String _buildNtripHeader() {
|
||||
final auth = _toBase64('${username.value}:${password.value}');
|
||||
final host = '${this.host.value}:${port.value}';
|
||||
|
||||
return 'GET /${mountpoint.value} HTTP/1.1\r\n'
|
||||
'User-Agent: SharpGps iter.dk\r\n'
|
||||
'Accept: */*\r\n'
|
||||
'Connection: close\r\n'
|
||||
'Authorization: Basic $auth\r\n'
|
||||
'Host:$host\r\n'
|
||||
'Ntrip-Version:Ntrip/2.0\r\n'
|
||||
'\r\n';
|
||||
}
|
||||
|
||||
// ── Beállítások mentése / betöltése ──────────────────────────────
|
||||
|
||||
Future<void> saveSettings() async {
|
||||
// Szinkronizálás a controllerektől az Rx értékekbe
|
||||
host.value = hostController.text.trim();
|
||||
port.value = int.tryParse(portController.text) ?? 2101;
|
||||
mountpoint.value = mountpointController.text.trim();
|
||||
username.value = usernameController.text.trim();
|
||||
password.value = passwordController.text;
|
||||
|
||||
final prefs = await SharedPreferences.getInstance();
|
||||
await prefs.setString('ntrip_host', host.value);
|
||||
await prefs.setInt('ntrip_port', port.value);
|
||||
await prefs.setString('ntrip_mountpoint', mountpoint.value);
|
||||
await prefs.setString('ntrip_username', username.value);
|
||||
await prefs.setString('ntrip_password', password.value);
|
||||
}
|
||||
|
||||
Future<void> _loadSettings() async {
|
||||
final prefs = await SharedPreferences.getInstance();
|
||||
host.value = prefs.getString('ntrip_host') ?? '84.206.45.44';
|
||||
port.value = prefs.getInt('ntrip_port') ?? 2101;
|
||||
mountpoint.value = prefs.getString('ntrip_mountpoint') ?? 'SGO_RTK3.2';
|
||||
username.value = prefs.getString('ntrip_username') ?? '';
|
||||
password.value = prefs.getString('ntrip_password') ?? '';
|
||||
}
|
||||
|
||||
void _syncControllersFromValues() {
|
||||
hostController.text = host.value;
|
||||
portController.text = port.value.toString();
|
||||
mountpointController.text = mountpoint.value;
|
||||
usernameController.text = username.value;
|
||||
// Jelszót nem pre-töltjük biztonsági okokból
|
||||
}
|
||||
|
||||
// ── Segédfüggvények ──────────────────────────────────────────────
|
||||
|
||||
String _toBase64(String str) => base64.encode(ascii.encode(str));
|
||||
|
||||
Uint8List _toUint8List(String str) => Uint8List.fromList(str.codeUnits);
|
||||
}
|
||||
@ -4,6 +4,7 @@ import 'dart:math';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:get/get.dart';
|
||||
import 'package:intl/intl.dart';
|
||||
import 'package:terepi_seged/services/ntrip_service.dart';
|
||||
|
||||
import '../services/gnss/gnss_service.dart';
|
||||
|
||||
@ -49,10 +50,10 @@ class CoordinatePanel extends StatelessWidget {
|
||||
vertError: ctrl.gpsAltitudeError as RxDouble,
|
||||
altitudeMsl: ctrl.gpsAltitude as RxDouble,
|
||||
geoidSeparation: ctrl.gpsGeoidSeparation as RxDouble,
|
||||
ntripConnected: ctrl.ntripIsConnected as RxBool,
|
||||
ntripBytes: ctrl.ntripReceivedData as RxInt,
|
||||
ntripPackets: ctrl.ntripDataPacketNumbers as RxInt,
|
||||
ggaPackets: ctrl.ggaSenDataPacketNumber as RxInt,
|
||||
ntripConnected: NtripService.to.isConnected,
|
||||
ntripBytes: NtripService.to.receivedBytes,
|
||||
ntripPackets: NtripService.to.packetCount,
|
||||
ggaPackets: NtripService.to.ggaSentCount,
|
||||
);
|
||||
}
|
||||
|
||||
|
||||
@ -1,5 +1,6 @@
|
||||
// lib/widgets/shared_map_widget.dart
|
||||
import 'dart:math' as math;
|
||||
import 'dart:math';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_map/flutter_map.dart';
|
||||
import 'package:get/get.dart';
|
||||
@ -139,13 +140,17 @@ class _SharedMapWidgetState extends State<SharedMapWidget> {
|
||||
// 2. Extra rétegek (terepbejárás elemei)
|
||||
...widget.extraLayers,
|
||||
// 3. Bemért pontok
|
||||
if (Get.isRegistered<MapSurveyController>())
|
||||
Obx(() => MarkerLayer(
|
||||
markers: _buildMeasuredPointMarkers(),
|
||||
)),
|
||||
// if (Get.isRegistered<MapSurveyController>())
|
||||
// Obx(() => MarkerLayer(
|
||||
// markers: _buildMeasuredPointMarkers(),
|
||||
// )),
|
||||
// 4. Kitűzési célpont + vonal
|
||||
if (Get.isRegistered<MapSurveyController>())
|
||||
Obx(() => _buildStakeoutLayer(lat, lon)),
|
||||
Obx(() {
|
||||
final lat = GnssService.to.latitude.value;
|
||||
final lon = GnssService.to.longitude.value;
|
||||
return _buildStakeoutLayer(lat, lon);
|
||||
}),
|
||||
// 5. GPS pozíció
|
||||
Obx(() {
|
||||
final lat = GnssService.to.latitude.value;
|
||||
@ -187,19 +192,7 @@ class _SharedMapWidgetState extends State<SharedMapWidget> {
|
||||
|
||||
List<Marker> _buildMeasuredPointMarkers() {
|
||||
if (!Get.isRegistered<MapSurveyController>()) return [];
|
||||
return MapSurveyController.to.measuredPoints1
|
||||
.map((point) => Marker(
|
||||
point: LatLng(point.latitude, point.longitude),
|
||||
width: 100,
|
||||
height: 56,
|
||||
alignment: Alignment.bottomCenter,
|
||||
child: _LabeledMarker(
|
||||
label: point.name,
|
||||
icon: Icons.location_on,
|
||||
color: Colors.blue,
|
||||
),
|
||||
))
|
||||
.toList();
|
||||
return MapSurveyController.to.pointNotesMarker;
|
||||
}
|
||||
|
||||
Widget _buildStakeoutLayer(double lat, double lon) {
|
||||
@ -214,6 +207,10 @@ class _SharedMapWidgetState extends State<SharedMapWidget> {
|
||||
final wgs = CoordConverterService.to
|
||||
.eovToWgsPoint(ctrl.targetEovY.value, ctrl.targetEovX.value);
|
||||
final targetLatLng = LatLng(wgs.y, wgs.x);
|
||||
final dx = ctrl.eov.value.X - ctrl.targetEovX.value;
|
||||
final dy = ctrl.eov.value.Y - ctrl.targetEovY.value;
|
||||
final dist = sqrt(dx * dx + dy * dy);
|
||||
final onTarget = dist < 0.05;
|
||||
|
||||
return Stack(children: [
|
||||
// Szaggatott vonal
|
||||
@ -235,10 +232,8 @@ class _SharedMapWidgetState extends State<SharedMapWidget> {
|
||||
label: ctrl.targetName.value,
|
||||
icon: Icons.flag,
|
||||
color: Colors.orange,
|
||||
activeColor: ctrl.isOnTarget ? Colors.green : null,
|
||||
sublabel: ctrl.isOnTarget
|
||||
? '✓ Célponton'
|
||||
: '${ctrl.distanceToTarget.toStringAsFixed(3)} m',
|
||||
activeColor: onTarget ? Colors.green : null,
|
||||
sublabel: onTarget ? '✓ Célponton' : '${dist.toStringAsFixed(3)} m',
|
||||
),
|
||||
),
|
||||
]),
|
||||
|
||||
Loading…
Reference in New Issue
Block a user