2026-05-10 02:31:27 +02:00
|
|
|
|
import 'dart:async';
|
2026-06-24 11:24:19 +02:00
|
|
|
|
import 'package:firebase_core/firebase_core.dart';
|
2026-05-10 02:31:27 +02:00
|
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
|
|
import 'package:flutter_foreground_task/flutter_foreground_task.dart';
|
2026-06-24 11:24:19 +02:00
|
|
|
|
import 'package:geolocator/geolocator.dart';
|
2025-10-05 14:34:46 +02:00
|
|
|
|
import 'package:get/get.dart';
|
2026-05-10 02:31:27 +02:00
|
|
|
|
import 'package:intl/intl.dart';
|
|
|
|
|
|
import 'package:latlong2/latlong.dart';
|
|
|
|
|
|
import 'package:share_plus/share_plus.dart';
|
2026-06-24 11:24:19 +02:00
|
|
|
|
import 'package:terepi_seged/services/app_logger.dart';
|
|
|
|
|
|
import 'package:terepi_seged/services/firebase_logger.dart';
|
2026-06-12 13:41:36 +02:00
|
|
|
|
import 'package:terepi_seged/services/project_service.dart';
|
2026-06-23 15:21:20 +02:00
|
|
|
|
import 'package:terepi_seged/services/track_sync_service.dart';
|
2025-10-05 14:34:46 +02:00
|
|
|
|
|
2026-05-10 02:31:27 +02:00
|
|
|
|
import '../../../../services/location_source.dart';
|
|
|
|
|
|
import '../../../../services/phone_gps_source.dart';
|
|
|
|
|
|
import '../../../../services/track_database.dart';
|
|
|
|
|
|
import '../../../../services/gpx_exporter.dart';
|
|
|
|
|
|
import '../../../../models/track.dart';
|
|
|
|
|
|
|
|
|
|
|
|
// ─── Foreground task handler ─────────────────────────────────────────────────
|
|
|
|
|
|
// Ez a függvény a háttér-isolate-ban fut. Csak a meglevő pozíció-streamet
|
|
|
|
|
|
// tartja fenn, a tényleges adatfeldolgozás a főszálban történik.
|
|
|
|
|
|
|
|
|
|
|
|
@pragma('vm:entry-point')
|
|
|
|
|
|
void startTrackingCallback() {
|
|
|
|
|
|
FlutterForegroundTask.setTaskHandler(_TrackingTaskHandler());
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
class _TrackingTaskHandler extends TaskHandler {
|
|
|
|
|
|
@override
|
|
|
|
|
|
Future<void> onStart(DateTime timestamp, TaskStarter starter) async {}
|
|
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
|
void onRepeatEvent(DateTime timestamp) {
|
|
|
|
|
|
// A Geolocator stream a főizoláton folyik tovább, itt nincs teendő.
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
|
Future<void> onDestroy(DateTime timestamp, bool isTimeout) async {}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ─── TrackingController ──────────────────────────────────────────────────────
|
|
|
|
|
|
|
|
|
|
|
|
class TrackingController extends GetxController {
|
2026-06-11 01:20:55 +02:00
|
|
|
|
static TrackingController get to => Get.find();
|
|
|
|
|
|
|
2026-05-10 02:31:27 +02:00
|
|
|
|
// ── Állapot ────────────────────────────────────────────────────────────────
|
|
|
|
|
|
final isRecording = false.obs;
|
|
|
|
|
|
final isPaused = false.obs;
|
|
|
|
|
|
final currentTrack = Rxn<Track>();
|
|
|
|
|
|
|
|
|
|
|
|
/// Aktuális session pontjai a térképhez (csak a memóriában).
|
|
|
|
|
|
final livePoints = <LatLng>[].obs;
|
|
|
|
|
|
|
|
|
|
|
|
/// Aktuális sebesség [km/h].
|
|
|
|
|
|
final currentSpeedKmh = 0.0.obs;
|
|
|
|
|
|
|
|
|
|
|
|
/// Megtett távolság [m] az aktuális sessionben.
|
|
|
|
|
|
final sessionDistance = 0.0.obs;
|
|
|
|
|
|
|
|
|
|
|
|
/// Eltelt idő formátuma óó:pp:mm.
|
|
|
|
|
|
final elapsedFormatted = '00:00:00'.obs;
|
|
|
|
|
|
|
|
|
|
|
|
/// Előtér/háttér állapot jelzése.
|
|
|
|
|
|
final isInBackground = false.obs;
|
|
|
|
|
|
|
|
|
|
|
|
// ── Mentett track-ek listája ────────────────────────────────────────────────
|
|
|
|
|
|
final savedTracks = <Track>[].obs;
|
|
|
|
|
|
|
2026-06-11 01:20:55 +02:00
|
|
|
|
// Melyik track-ek látszanak overlay-ként a térképen
|
|
|
|
|
|
final overlayTrackIds = <int>[].obs;
|
|
|
|
|
|
|
|
|
|
|
|
// Betöltött koordináták cache
|
|
|
|
|
|
final Map<int, List<LatLng>> _trackCoords = {};
|
|
|
|
|
|
|
|
|
|
|
|
int get livePointCount => livePoints.length;
|
|
|
|
|
|
|
2026-05-10 02:31:27 +02:00
|
|
|
|
// ── Belső állapot ──────────────────────────────────────────────────────────
|
|
|
|
|
|
LocationSource? _source;
|
|
|
|
|
|
StreamSubscription<SourcePosition>? _positionSub;
|
|
|
|
|
|
Timer? _elapsedTimer;
|
|
|
|
|
|
|
|
|
|
|
|
TrackPoint? _lastPoint;
|
|
|
|
|
|
DateTime? _sessionStart;
|
|
|
|
|
|
double _accumulatedDistance = 0;
|
|
|
|
|
|
|
|
|
|
|
|
final _db = TrackDatabase.instance;
|
|
|
|
|
|
final _exporter = GpxExporter();
|
|
|
|
|
|
|
|
|
|
|
|
// ── Inicializálás ──────────────────────────────────────────────────────────
|
|
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
|
void onInit() {
|
|
|
|
|
|
super.onInit();
|
|
|
|
|
|
_initForegroundTask();
|
|
|
|
|
|
loadSavedTracks();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void _initForegroundTask() {
|
|
|
|
|
|
FlutterForegroundTask.init(
|
|
|
|
|
|
androidNotificationOptions: AndroidNotificationOptions(
|
|
|
|
|
|
channelId: 'tracking_channel',
|
|
|
|
|
|
channelName: 'Nyomvonal rögzítés',
|
|
|
|
|
|
channelDescription: 'Aktív track rögzítése folyamatban',
|
|
|
|
|
|
channelImportance: NotificationChannelImportance.LOW,
|
|
|
|
|
|
priority: NotificationPriority.LOW,
|
|
|
|
|
|
),
|
|
|
|
|
|
iosNotificationOptions: const IOSNotificationOptions(
|
|
|
|
|
|
showNotification: true,
|
|
|
|
|
|
playSound: false,
|
|
|
|
|
|
),
|
|
|
|
|
|
foregroundTaskOptions: ForegroundTaskOptions(
|
|
|
|
|
|
eventAction: ForegroundTaskEventAction.repeat(1000),
|
|
|
|
|
|
autoRunOnBoot: false,
|
|
|
|
|
|
allowWakeLock: true,
|
|
|
|
|
|
allowWifiLock: false,
|
|
|
|
|
|
),
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ── Publikus API ───────────────────────────────────────────────────────────
|
|
|
|
|
|
|
|
|
|
|
|
/// Elindítja a rögzítést a megadott forrással (alapértelmezett: telefon GPS).
|
2026-06-12 13:41:36 +02:00
|
|
|
|
Future<void> startRecording(
|
|
|
|
|
|
{LocationSource? source,
|
|
|
|
|
|
String? name,
|
|
|
|
|
|
TrackSyncMode syncMode = TrackSyncMode.online}) async {
|
2026-05-10 02:31:27 +02:00
|
|
|
|
if (isRecording.value) return;
|
|
|
|
|
|
|
|
|
|
|
|
_source = source ?? PhoneGpsSource(intervalMs: 1000, distanceFilter: 2.0);
|
|
|
|
|
|
|
|
|
|
|
|
// Track létrehozása az adatbázisban
|
|
|
|
|
|
final now = DateTime.now();
|
2026-06-12 13:41:36 +02:00
|
|
|
|
final trackName = name ?? DateFormat('yyyy-MM-dd HH:mm').format(now);
|
2026-05-10 02:31:27 +02:00
|
|
|
|
final trackId = await _db.insertTrack(Track(
|
2026-06-12 13:41:36 +02:00
|
|
|
|
name: trackName,
|
|
|
|
|
|
startTime: now,
|
|
|
|
|
|
source: _source!.displayName,
|
|
|
|
|
|
projectId: ProjectService.to.activeProjectId,
|
|
|
|
|
|
syncMode: syncMode));
|
2026-05-10 02:31:27 +02:00
|
|
|
|
currentTrack.value = await _db.getTrack(trackId);
|
|
|
|
|
|
|
|
|
|
|
|
// Állapot reset
|
|
|
|
|
|
livePoints.clear();
|
|
|
|
|
|
sessionDistance.value = 0;
|
|
|
|
|
|
_accumulatedDistance = 0;
|
|
|
|
|
|
_lastPoint = null;
|
|
|
|
|
|
_sessionStart = now;
|
|
|
|
|
|
|
|
|
|
|
|
// Foreground service indítása (háttér-működéshez)
|
|
|
|
|
|
await FlutterForegroundTask.startService(
|
|
|
|
|
|
notificationTitle: 'Track rögzítése',
|
2026-06-12 13:41:36 +02:00
|
|
|
|
notificationText: trackName,
|
2026-05-10 02:31:27 +02:00
|
|
|
|
callback: startTrackingCallback,
|
|
|
|
|
|
);
|
|
|
|
|
|
|
2026-06-23 15:21:20 +02:00
|
|
|
|
if (!(currentTrack.value?.isLocalOnly ?? true)) {
|
|
|
|
|
|
TrackSyncService.to
|
|
|
|
|
|
.createRemoteTrack(currentTrack.value!)
|
|
|
|
|
|
.then((supabaseId) async {
|
|
|
|
|
|
if (supabaseId == null) return;
|
|
|
|
|
|
final updated = currentTrack.value!.copyWith(supabaseId: supabaseId);
|
|
|
|
|
|
await _db.updateTrack(updated);
|
|
|
|
|
|
currentTrack.value = updated;
|
|
|
|
|
|
TrackSyncService.to.startSession(updated);
|
|
|
|
|
|
});
|
|
|
|
|
|
} else {
|
|
|
|
|
|
TrackSyncService.to.startSession(currentTrack.value!);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-05-10 02:31:27 +02:00
|
|
|
|
// GPS stream feliratkozás
|
|
|
|
|
|
_positionSub = _source!.positionStream.listen(
|
|
|
|
|
|
_onPosition,
|
|
|
|
|
|
onError: (e) {
|
2026-06-24 11:24:19 +02:00
|
|
|
|
AppLogger.e('positionStream', 'Stream hiba', error: e);
|
|
|
|
|
|
FirebaseLogger.e('positionStream', 'Stream hiba', error: e);
|
|
|
|
|
|
if (e is LocationServiceDisabledException ||
|
|
|
|
|
|
e is PermissionDeniedException) {
|
|
|
|
|
|
Get.snackbar('GPS hiba', e.toString(),
|
|
|
|
|
|
backgroundColor: Colors.red, colorText: Colors.white);
|
|
|
|
|
|
stopRecording();
|
|
|
|
|
|
}
|
2026-05-10 02:31:27 +02:00
|
|
|
|
},
|
|
|
|
|
|
);
|
2026-06-24 11:24:19 +02:00
|
|
|
|
AppLogger.e('track_started', 'name:$trackName');
|
|
|
|
|
|
FirebaseLogger.event('track_started', {'name': trackName});
|
2026-05-10 02:31:27 +02:00
|
|
|
|
|
|
|
|
|
|
// Időmérő
|
|
|
|
|
|
_startElapsedTimer();
|
|
|
|
|
|
|
|
|
|
|
|
isRecording.value = true;
|
|
|
|
|
|
isPaused.value = false;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void pauseRecording() {
|
|
|
|
|
|
if (!isRecording.value || isPaused.value) return;
|
|
|
|
|
|
_positionSub?.pause();
|
|
|
|
|
|
_elapsedTimer?.cancel();
|
|
|
|
|
|
isPaused.value = true;
|
|
|
|
|
|
FlutterForegroundTask.updateService(
|
|
|
|
|
|
notificationTitle: 'Track szüneteltetve',
|
|
|
|
|
|
notificationText: currentTrack.value?.name ?? '',
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void resumeRecording() {
|
|
|
|
|
|
if (!isPaused.value) return;
|
|
|
|
|
|
_positionSub?.resume();
|
|
|
|
|
|
_startElapsedTimer();
|
|
|
|
|
|
isPaused.value = false;
|
|
|
|
|
|
FlutterForegroundTask.updateService(
|
|
|
|
|
|
notificationTitle: 'Track rögzítése',
|
|
|
|
|
|
notificationText: currentTrack.value?.name ?? '',
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Future<void> stopRecording() async {
|
|
|
|
|
|
if (!isRecording.value) return;
|
|
|
|
|
|
|
|
|
|
|
|
await _positionSub?.cancel();
|
|
|
|
|
|
_positionSub = null;
|
|
|
|
|
|
_elapsedTimer?.cancel();
|
|
|
|
|
|
_elapsedTimer = null;
|
|
|
|
|
|
|
|
|
|
|
|
// Track lezárása az adatbázisban
|
|
|
|
|
|
final track = currentTrack.value;
|
|
|
|
|
|
if (track?.id != null) {
|
|
|
|
|
|
final finished = track!.copyWith(
|
|
|
|
|
|
status: TrackStatus.finished,
|
|
|
|
|
|
endTime: DateTime.now(),
|
2026-06-11 01:20:55 +02:00
|
|
|
|
distanceMeters: _accumulatedDistance,
|
|
|
|
|
|
pointCount: livePoints.length,
|
2026-05-10 02:31:27 +02:00
|
|
|
|
);
|
|
|
|
|
|
await _db.updateTrack(finished);
|
2026-06-23 15:21:20 +02:00
|
|
|
|
await TrackSyncService.to.stopSession(finished);
|
2026-05-10 02:31:27 +02:00
|
|
|
|
currentTrack.value = finished;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
await FlutterForegroundTask.stopService();
|
|
|
|
|
|
await _source?.dispose();
|
|
|
|
|
|
_source = null;
|
|
|
|
|
|
|
|
|
|
|
|
isRecording.value = false;
|
|
|
|
|
|
isPaused.value = false;
|
|
|
|
|
|
|
|
|
|
|
|
await loadSavedTracks();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Future<void> loadSavedTracks() async {
|
|
|
|
|
|
savedTracks.value = await _db.listTracks();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Future<void> deleteTrack(int id) async {
|
|
|
|
|
|
await _db.deleteTrack(id);
|
2026-06-11 01:20:55 +02:00
|
|
|
|
overlayTrackIds.remove(id);
|
|
|
|
|
|
_trackCoords.remove(id);
|
2026-05-10 02:31:27 +02:00
|
|
|
|
await loadSavedTracks();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// GPX export + rendszer megosztás dialóg.
|
|
|
|
|
|
Future<void> exportTrack(Track track) async {
|
|
|
|
|
|
try {
|
|
|
|
|
|
final path = await _exporter.export(track);
|
|
|
|
|
|
await Share.shareXFiles([XFile(path)],
|
2026-06-23 15:21:20 +02:00
|
|
|
|
subject: 'Nyomvonal: ${track.name}.gpx');
|
2026-05-10 02:31:27 +02:00
|
|
|
|
} catch (e) {
|
|
|
|
|
|
Get.snackbar('Export hiba', e.toString(),
|
|
|
|
|
|
backgroundColor: Colors.red, colorText: Colors.white);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// Visszatölt egy mentett track pontjait a térképre (megtekintés).
|
|
|
|
|
|
Future<List<LatLng>> loadTrackPoints(int trackId) async {
|
|
|
|
|
|
final pts = await _db.getLatLons(trackId);
|
|
|
|
|
|
return pts.map((p) => LatLng(p.lat, p.lon)).toList();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ── Belső logika ───────────────────────────────────────────────────────────
|
|
|
|
|
|
|
|
|
|
|
|
Future<void> _onPosition(SourcePosition pos) async {
|
2026-06-24 11:24:19 +02:00
|
|
|
|
try {
|
|
|
|
|
|
if (isPaused.value) return;
|
|
|
|
|
|
final sw = Stopwatch()..start();
|
|
|
|
|
|
|
|
|
|
|
|
final trackId = currentTrack.value?.id;
|
|
|
|
|
|
if (trackId == null) return;
|
|
|
|
|
|
|
|
|
|
|
|
// Távolság a legutóbbi ponttól
|
|
|
|
|
|
double segmentDist = 0;
|
|
|
|
|
|
if (_lastPoint != null) {
|
|
|
|
|
|
segmentDist = haversineMeters(
|
|
|
|
|
|
_lastPoint!.latitude,
|
|
|
|
|
|
_lastPoint!.longitude,
|
|
|
|
|
|
pos.latitude,
|
|
|
|
|
|
pos.longitude,
|
|
|
|
|
|
);
|
|
|
|
|
|
// Szűrés: ugrásszerű változás (pl. GPS lock elvesztése) ignorálása
|
|
|
|
|
|
if (segmentDist > 100) return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
_accumulatedDistance += segmentDist;
|
|
|
|
|
|
sessionDistance.value = _accumulatedDistance;
|
|
|
|
|
|
|
|
|
|
|
|
// Sebesség km/h-ban
|
|
|
|
|
|
currentSpeedKmh.value = (pos.speed ?? 0) * 3.6;
|
|
|
|
|
|
|
|
|
|
|
|
// Pont mentése
|
|
|
|
|
|
final point = TrackPoint(
|
|
|
|
|
|
trackId: trackId,
|
|
|
|
|
|
latitude: pos.latitude,
|
|
|
|
|
|
longitude: pos.longitude,
|
|
|
|
|
|
altitude: pos.altitude,
|
|
|
|
|
|
accuracy: pos.accuracy,
|
|
|
|
|
|
speed: pos.speed,
|
|
|
|
|
|
heading: pos.heading,
|
|
|
|
|
|
timestamp: pos.timestamp,
|
2026-05-10 02:31:27 +02:00
|
|
|
|
);
|
2026-06-24 11:24:19 +02:00
|
|
|
|
await _db.addPoint(point, _accumulatedDistance);
|
|
|
|
|
|
sw.stop();
|
|
|
|
|
|
|
|
|
|
|
|
if (sw.elapsedMilliseconds > 300) {
|
|
|
|
|
|
AppLogger.w(
|
|
|
|
|
|
'on_Position',
|
|
|
|
|
|
'Lassú addPoint: ${sw.elapsedMilliseconds}ms, '
|
|
|
|
|
|
'pts: ${livePoints.length}');
|
|
|
|
|
|
FirebaseLogger.w(
|
|
|
|
|
|
'on_Position',
|
|
|
|
|
|
'Lassú addPoint: ${sw.elapsedMilliseconds}ms, '
|
|
|
|
|
|
'pts: ${livePoints.length}');
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
TrackSyncService.to.onNewPoint(point);
|
|
|
|
|
|
|
|
|
|
|
|
_lastPoint = point;
|
|
|
|
|
|
|
|
|
|
|
|
// UI frissítés
|
|
|
|
|
|
livePoints.add(LatLng(pos.latitude, pos.longitude));
|
|
|
|
|
|
|
|
|
|
|
|
// Értesítés frissítése
|
|
|
|
|
|
if (livePoints.length % 10 == 0) {
|
|
|
|
|
|
final dist = _formatDistance(_accumulatedDistance);
|
|
|
|
|
|
FlutterForegroundTask.updateService(
|
|
|
|
|
|
notificationTitle: 'Track rögzítése – $dist',
|
|
|
|
|
|
notificationText: elapsedFormatted.value,
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
} catch (e, stack) {
|
|
|
|
|
|
AppLogger.e('_onPosition', 'pont feldolgozási hiba - track folytatódik',
|
|
|
|
|
|
error: e, stack: stack);
|
2026-05-10 02:31:27 +02:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void _startElapsedTimer() {
|
|
|
|
|
|
_elapsedTimer = Timer.periodic(const Duration(seconds: 1), (_) {
|
|
|
|
|
|
if (_sessionStart == null) return;
|
|
|
|
|
|
final d = DateTime.now().difference(_sessionStart!);
|
|
|
|
|
|
final h = d.inHours.toString().padLeft(2, '0');
|
|
|
|
|
|
final m = (d.inMinutes % 60).toString().padLeft(2, '0');
|
|
|
|
|
|
final s = (d.inSeconds % 60).toString().padLeft(2, '0');
|
|
|
|
|
|
elapsedFormatted.value = '$h:$m:$s';
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String _formatDistance(double m) {
|
|
|
|
|
|
if (m < 1000) return '${m.toStringAsFixed(0)} m';
|
|
|
|
|
|
return '${(m / 1000).toStringAsFixed(2)} km';
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
|
void onClose() {
|
|
|
|
|
|
stopRecording();
|
|
|
|
|
|
if (!isRecording.value) {
|
|
|
|
|
|
_positionSub?.cancel();
|
|
|
|
|
|
_elapsedTimer?.cancel();
|
|
|
|
|
|
}
|
|
|
|
|
|
super.onClose();
|
|
|
|
|
|
}
|
2026-06-11 01:20:55 +02:00
|
|
|
|
|
|
|
|
|
|
// Overlay ki/be kapcsolása — ha bekapcsol, betölti a koordinátákat
|
|
|
|
|
|
void toggleTrackOverlay(int trackId) {
|
|
|
|
|
|
if (overlayTrackIds.contains(trackId)) {
|
|
|
|
|
|
overlayTrackIds.remove(trackId);
|
|
|
|
|
|
overlayTrackIds.refresh();
|
|
|
|
|
|
} else {
|
|
|
|
|
|
overlayTrackIds.add(trackId);
|
|
|
|
|
|
overlayTrackIds.refresh();
|
|
|
|
|
|
_loadTrackCoords(trackId);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// Koordináták visszaadása — üres lista ha még nincs betöltve
|
|
|
|
|
|
List<LatLng> getCoordsFor(int trackId) => _trackCoords[trackId] ?? [];
|
|
|
|
|
|
|
|
|
|
|
|
Future<void> _loadTrackCoords(int trackId) async {
|
|
|
|
|
|
if (_trackCoords.containsKey(trackId)) return;
|
|
|
|
|
|
final pts = await _db.getLatLons(trackId);
|
|
|
|
|
|
_trackCoords[trackId] = pts.map((p) => LatLng(p.lat, p.lon)).toList();
|
|
|
|
|
|
overlayTrackIds.refresh(); // térkép frissítés
|
|
|
|
|
|
}
|
2026-05-10 02:31:27 +02:00
|
|
|
|
}
|