import 'package:animations/animations.dart'; import 'package:fforte/enums/databases.dart'; import 'package:fforte/screens/addCam/exceptions/location_disabled_exception.dart'; import 'package:fforte/screens/addCam/exceptions/location_forbidden_exception.dart'; import 'package:fforte/screens/addCam/services/geolocator_service.dart'; import 'package:fforte/screens/excursion/exceptions/need_always_location_exception.dart'; import 'package:fforte/screens/excursion/widgets/anzahlen.dart'; import 'package:fforte/screens/excursion/widgets/bima_nutzer.dart'; import 'package:fforte/screens/excursion/widgets/hinweise.dart'; import 'package:fforte/screens/excursion/widgets/hund_u_leine.dart'; import 'package:fforte/screens/excursion/widgets/letzter_niederschlag.dart'; import 'package:fforte/screens/excursion/widgets/spur_gefunden.dart'; import 'package:fforte/screens/excursion/widgets/strecke_u_spurbedingungen.dart'; import 'package:fforte/screens/excursion/widgets/tracking.dart'; import 'package:fforte/screens/helper/add_entries_dialog_helper.dart'; import 'package:fforte/screens/helper/snack_bar_helper.dart'; import 'package:fforte/screens/sharedMethods/check_required.dart'; import 'package:fforte/screens/sharedMethods/save_template.dart'; import 'package:fforte/screens/sharedWidgets/datum.dart'; import 'package:fforte/screens/sharedWidgets/var_text_field.dart'; import 'package:fforte/l10n/app_localizations.dart'; import 'package:fforte/services/tracking_service.dart'; import 'package:flutter/material.dart'; import 'package:geolocator/geolocator.dart'; import 'package:shared_preferences/shared_preferences.dart'; class ExcursionMain extends StatefulWidget { final bool isTemplate; final bool isSent; final Map? existingData; const ExcursionMain({ super.key, this.isTemplate = false, this.isSent = false, this.existingData, }); @override State createState() => _ExcursionMainState(); } class _ExcursionMainState extends State { int currentStep = 0; late bool isTemplate; Position? currentPosition; bool isLoadingPosition = true; late Future _positionFuture; bool bimaExtended = false; // all TextEditingController because its easier Map> rmap = { "ID": {"controller": TextEditingController(), "required": false}, // Step 1 "Datum": {"controller": TextEditingController(), "required": false}, "Rudel": {"controller": TextEditingController(), "required": false}, "Teilnehmer": {"controller": TextEditingController(), "required": false}, "Dauer": {"controller": TextEditingController(), "required": false}, "MHund": {"controller": TextEditingController(), "required": false}, "MLeine": {"controller": TextEditingController(), "required": false}, "BLand": {"controller": TextEditingController(), "required": false}, "Lkr": {"controller": TextEditingController(), "required": false}, "BeiOrt": {"controller": TextEditingController(), "required": false}, "BimaNr": {"controller": TextEditingController(), "required": false}, "BimaName": {"controller": TextEditingController(), "required": false}, "BimaNutzer": {"controller": TextEditingController(), "required": false}, "BimaAGV": {"controller": TextEditingController(), "required": false}, // Step 2 "Weg": {"controller": TextEditingController(), "required": false}, "Wetter": {"controller": TextEditingController(), "required": false}, "Temperat": {"controller": TextEditingController(), "required": false}, "RegenVor": {"controller": TextEditingController(), "required": false}, "KmAuto": {"controller": TextEditingController(), "required": false}, "KmFuss": {"controller": TextEditingController(), "required": false}, "KmRad": {"controller": TextEditingController(), "required": false}, "KmTotal": {"controller": TextEditingController(), "required": false}, "KmAuProz": {"controller": TextEditingController(), "required": false}, "KmFuProz": {"controller": TextEditingController(), "required": false}, "KmRaProz": {"controller": TextEditingController(), "required": false}, // Spur maybe own step? "SpGut": {"controller": TextEditingController(), "required": false}, "SpMittel": {"controller": TextEditingController(), "required": false}, "SpSchlecht": {"controller": TextEditingController(), "required": false}, "SpurFund": {"controller": TextEditingController(), "required": false}, "SpurLang": {"controller": TextEditingController(), "required": false}, "SpurTiere": {"controller": TextEditingController(), "required": false}, "SpSicher": {"controller": TextEditingController(), "required": false}, "WelpenSp": {"controller": TextEditingController(), "required": false}, "WelpenAnz": {"controller": TextEditingController(), "required": false}, "WpSicher": {"controller": TextEditingController(), "required": false}, "LosungGes": {"controller": TextEditingController(), "required": false}, "LosungAnz": {"controller": TextEditingController(), "required": false}, "LosungGen": {"controller": TextEditingController(), "required": false}, "UrinAnz": {"controller": TextEditingController(), "required": false}, "UrinGen": {"controller": TextEditingController(), "required": false}, "OestrAnz": {"controller": TextEditingController(), "required": false}, "OestrGen": {"controller": TextEditingController(), "required": false}, "HaarAnz": {"controller": TextEditingController(), "required": false}, "HaarGen": {"controller": TextEditingController(), "required": false}, "LosungKm": {"controller": TextEditingController(), "required": false}, "GenetiKm": {"controller": TextEditingController(), "required": false}, "Hinweise": {"controller": TextEditingController(), "required": false}, // Step 3 "Bemerk": {"controller": TextEditingController(), "required": false}, "IntKomm": {"controller": TextEditingController(), "required": false}, "FallNum": {"controller": TextEditingController(), "required": false}, "Sent": {"controller": TextEditingController(), "required": false}, }; @override void initState() { super.initState(); _positionFuture = _initializePosition(); if (widget.existingData?.isNotEmpty ?? false) { for (var key in widget.existingData!.keys) { rmap[key]!["controller"]!.text = widget.existingData?[key].toString() ?? ""; } } else { // Set BLand and default values if there is no existing data SharedPreferences.getInstance().then((prefs) { rmap["BLand"]!["controller"]!.text = prefs.getString('bLand') ?? ""; }); rmap["Datum"]!["controller"]!.text = DateTime.now().toString(); rmap["Sent"]!["controller"]!.text = "0"; } isTemplate = widget.isTemplate; } Future _initializePosition() async { try { final position = await GeolocatorService.deteterminePosition( alwaysOnNeeded: true, ); if (mounted) { setState(() { currentPosition = position; isLoadingPosition = false; }); } return position; } catch (error) { if (!mounted) { return _getDefaultPosition(); } if (error is LocationDisabledException) { SnackBarHelper.showSnackBarMessage( context, AppLocalizations.of(context)!.locationDisabled, ); } else if (error is LocationForbiddenException) { SnackBarHelper.showSnackBarMessage( context, AppLocalizations.of(context)!.locationForbidden, ); } else if (error is NeedAlwaysLocation) { AddEntriesDialogHelper.locationSettingsDialog(context); } // Return default position on any error final defaultPosition = _getDefaultPosition(); if (mounted) { setState(() { currentPosition = defaultPosition; isLoadingPosition = false; }); } return defaultPosition; } } Position _getDefaultPosition() { return Position( longitude: 10.0, latitude: 51.0, timestamp: DateTime.now(), accuracy: 0.0, altitude: 0.0, heading: 0.0, speed: 0.0, speedAccuracy: 0.0, altitudeAccuracy: 0.0, headingAccuracy: 0.0, ); } Future _refreshPosition() async { setState(() { isLoadingPosition = true; }); _positionFuture = _initializePosition(); try { final position = await _positionFuture; if (mounted) { setState(() { currentPosition = position; isLoadingPosition = false; }); } } catch (e) { // Error already handled in _initializePosition if (mounted) { setState(() { isLoadingPosition = false; }); } } } @override void dispose() { for (String key in rmap.keys) { rmap[key]!["controller"].dispose(); } super.dispose(); } Map getFieldsText() { Map puff = {}; for (var itemKey in rmap.keys) { puff[itemKey] = rmap[itemKey]!["controller"]!.text; } return puff; } @override Widget build(BuildContext context) { List getSteps() => [ Step( title: Text(AppLocalizations.of(context)!.dateandtime), content: Column( children: [ // ---------- Date Datum( initDatum: DateTime.now(), onDateChanged: (date) { rmap["Datum"]!["controller"]!.text = date.toString(); }, name: AppLocalizations.of(context)!.date, ), const SizedBox(height: 10), // ---------- Pack VarTextField( textController: rmap["Rudel"]!["controller"]!, localization: AppLocalizations.of(context)!.rudel, dbName: "Rudel", required: false, dbDesignation: DatabasesEnum.excursion, ), const SizedBox(height: 10), // ---------- Participants VarTextField( textController: rmap["Teilnehmer"]!["controller"]!, localization: AppLocalizations.of(context)!.teilnehmer, dbName: "Teilnehmer", required: false, dbDesignation: DatabasesEnum.excursion, ), const SizedBox(height: 10), // ---------- Duration VarTextField( textController: rmap["Dauer"]!["controller"]!, localization: AppLocalizations.of(context)!.dauer, dbName: "Dauer", required: false, dbDesignation: DatabasesEnum.excursion, ), const SizedBox(height: 20), // ---------- Dog(leash) HundULeine( mHund: rmap["MHund"]!["controller"]!, mLeine: rmap["MLeine"]!["controller"]!, ), const SizedBox(height: 10), // ---------- State VarTextField( textController: rmap["BLand"]!["controller"]!, localization: AppLocalizations.of(context)!.bland, dbName: "BLand", required: false, dbDesignation: DatabasesEnum.excursion, ), const SizedBox(height: 10), // ---------- Country VarTextField( textController: rmap["Lkr"]!["controller"]!, localization: AppLocalizations.of(context)!.lkr, dbName: "Lkr", required: false, dbDesignation: DatabasesEnum.excursion, ), const SizedBox(height: 10), // ---------- By State VarTextField( textController: rmap["BeiOrt"]!["controller"]!, localization: AppLocalizations.of(context)!.beiort, dbName: "BeiOrt", required: false, dbDesignation: DatabasesEnum.excursion, ), const SizedBox(height: 10), // ---------- Bima number const Divider(), const SizedBox(height: 10), ClipRRect( borderRadius: BorderRadius.all(Radius.circular(10)), child: ExpansionPanelList( expansionCallback: ((int index, bool isExpanded) => setState(() => bimaExtended = isExpanded)), expandedHeaderPadding: EdgeInsets.all(0), children: [ ExpansionPanel( isExpanded: bimaExtended, canTapOnHeader: true, headerBuilder: (context, bool isOpen) => Padding( padding: const EdgeInsets.only(left: 15), child: Align( alignment: Alignment.centerLeft, child: Text( "BImA", style: Theme.of(context).textTheme.bodyLarge, ), ), ), body: Padding( padding: const EdgeInsets.all(15), child: Column( children: [ const SizedBox(height: 10), VarTextField( textController: rmap["BimaNr"]!["controller"]!, localization: AppLocalizations.of(context)!.bimaNr, dbName: "BimaNr", required: false, dbDesignation: DatabasesEnum.excursion, ), const SizedBox(height: 10), // ---------- Bima name VarTextField( textController: rmap["BimaName"]!["controller"]!, localization: AppLocalizations.of(context)!.bimaName, dbName: "BimaName", required: false, dbDesignation: DatabasesEnum.excursion, ), const SizedBox(height: 10), // ---------- Bima user BimaNutzer( onBimaNutzerChanged: (value) { setState(() { rmap["BimaNutzer"]!["controller"]!.text = value; }); }, ), const SizedBox(height: 10), // ---------- Bima AGV VarTextField( textController: rmap["BimaAGV"]!["controller"]!, localization: AppLocalizations.of(context)!.bimaAGV, dbName: "BimaAGV", required: false, dbDesignation: DatabasesEnum.excursion, ), ], ), ), ), ], ), ), ], ), ), Step( title: Text(AppLocalizations.of(context)!.umstaendeUndAktionen), content: Column( children: [ // ---------- Tracking _buildTrackingButtons(), const SizedBox(height: 10), // ---------- Weather VarTextField( textController: rmap["Wetter"]!["controller"]!, localization: AppLocalizations.of(context)!.wetter, dbName: "Wetter", required: false, dbDesignation: DatabasesEnum.excursion, ), const SizedBox(height: 10), // ---------- Temperature VarTextField( textController: rmap["Temperat"]!["controller"]!, localization: AppLocalizations.of(context)!.temperatur, dbName: "Temperat", required: false, dbDesignation: DatabasesEnum.excursion, ), const SizedBox(height: 10), // ---------- Last precipitation LetzterNiederschlag( controller: rmap["RegenVor"]!["controller"]!), const SizedBox(height: 20), // ---------- Track conditions StreckeUSpurbedingungen( kmAutoController: rmap["KmAuto"]!["controller"]!, kmFussController: rmap["KmFuss"]!["controller"]!, kmRadController: rmap["KmRad"]!["controller"]!, spGutController: rmap["SpGut"]!["controller"]!, spMittelController: rmap["SpMittel"]!["controller"]!, spSchlechtController: rmap["SpSchlecht"]!["controller"]!, ), const SizedBox(height: 20), const Divider(), // ---------- Track found SpurGefunden( spurFund: rmap["SpurFund"]!["controller"]!, spurLang: rmap["SpurLang"]!["controller"]!, spurTiere: rmap["SpurTiere"]!["controller"]!, spSicher: rmap["SpSicher"]!["controller"]!, welpenSp: rmap["WelpenSp"]!["controller"]!, welpenAnz: rmap["WelpenAnz"]!["controller"]!, wpSicher: rmap["WpSicher"]!["controller"]!, ), const Divider(), const SizedBox(height: 20), // ---------- Counts Anzahlen( losungAnz: rmap["LosungAnz"]!["controller"]!, losungGes: rmap["LosungGes"]!["controller"]!, losungGen: rmap["LosungGen"]!["controller"]!, urinAnz: rmap["UrinAnz"]!["controller"]!, urinGen: rmap["UrinGen"]!["controller"]!, oestrAnz: rmap["OestrAnz"]!["controller"]!, oestrGen: rmap["OestrGen"]!["controller"]!, haarAnz: rmap["HaarAnz"]!["controller"]!, haarGen: rmap["HaarGen"]!["controller"]!, ), const SizedBox(height: 20), const Divider(), const SizedBox(height: 20), // ---------- Clues Align( alignment: Alignment.bottomLeft, child: Text( AppLocalizations.of(context)!.hinweise, style: Theme.of(context).textTheme.titleMedium, ), ), Hinweise(hinweise: rmap["Hinweise"]!["controller"]!), ], ), ), Step( title: Text(AppLocalizations.of(context)!.intkomm), content: Column( children: [ // ---------- Remarks VarTextField( textController: rmap["Bemerk"]!["controller"]!, localization: AppLocalizations.of(context)!.sonstbemerkungen, dbName: "Bemerk", required: false, dbDesignation: DatabasesEnum.excursion, ), const SizedBox(height: 20), // ---------- Internal communication VarTextField( textController: rmap["IntKomm"]!["controller"]!, localization: AppLocalizations.of(context)!.intkomm, dbName: "IntKomm", required: false, dbDesignation: DatabasesEnum.excursion, ), ], ), ), ]; return PopScope( canPop: false, onPopInvokedWithResult: (bool didPop, Object? res) async { if (didPop) { return; } // Show confirmation dialog final result = await showDialog( context: context, builder: (context) => AlertDialog( title: Text(AppLocalizations.of(context)!.leavePageTitle), content: Text(AppLocalizations.of(context)!.leavePageContent), actions: [ TextButton( onPressed: () => Navigator.of(context).pop(0), child: Text(AppLocalizations.of(context)!.nein), ), TextButton( onPressed: () { saveTemplate(getFieldsText(), DatabasesEnum.excursion); Navigator.of(context).pop(1); }, child: Text(AppLocalizations.of(context)!.leaveAndSaveTemplate), ), TextButton( onPressed: () => Navigator.of(context).pop(2), child: Text(AppLocalizations.of(context)!.leaveWithoutSaving), ), ], ), ); if (result == null || result == 0) { return; } else if (result == 1) { // Save as template and leave if (context.mounted) { saveTemplate( getFieldsText(), DatabasesEnum.excursion, ); } TrackingService.resetInstance(); if (context.mounted) { Navigator.of(context).pop(); } } else { // Just leave without saving TrackingService.resetInstance(); if (context.mounted) { Navigator.of(context).pop(); } } }, child: Scaffold( appBar: AppBar( title: Text(AppLocalizations.of(context)!.excursion), actions: [ // Text(TrackingService().isTracking ? "Tracking" : "Not tracking") Image.asset( TrackingService().isTracking ? "assets/icons/tracking_on.png" : "assets/icons/tracking_off.png", width: 40, ), ], ), body: PageTransitionSwitcher( duration: const Duration(microseconds: 800), transitionBuilder: ( Widget child, Animation animation, Animation secondaryAnimation, ) { return SharedAxisTransition( animation: animation, secondaryAnimation: secondaryAnimation, transitionType: SharedAxisTransitionType.vertical, child: child, ); }, child: Stepper( key: ValueKey(currentStep), steps: getSteps(), currentStep: currentStep, onStepTapped: (value) { setState(() { currentStep = value; }); }, onStepContinue: () async { final isLastStep = currentStep == getSteps().length - 1; if (!isLastStep) { var res = await saveTemplate( getFieldsText(), DatabasesEnum.excursion, ); isTemplate = true; setState(() { rmap["ID"]!["controller"]!.text = res.toString(); currentStep += 1; }); } else { if (widget.isSent) { Navigator.pushNamedAndRemoveUntil( context, '/home', (route) => false, ); return; } bool empty = CheckRequired.checkRequired(rmap); // for debugging always false // empty = false; if (empty) { AddEntriesDialogHelper.showTemplateDialog( context, getFieldsText(), DatabasesEnum.excursion, ); return; } else { bool pop = await AddEntriesDialogHelper.showSaveOptionsDialog( context, getFieldsText(), isTemplate, DatabasesEnum.excursion, ); if (pop && context.mounted) Navigator.of(context).pop(); } } }, onStepCancel: () { if (currentStep == 0) { Navigator.pop(context); } else { setState(() { currentStep -= 1; }); } }, ), ), ), ); } // Add a refresh position button next to the tracking button Widget _buildTrackingButtons() { return FutureBuilder( future: _positionFuture, builder: (context, snapshot) { final bool isLoading = snapshot.connectionState == ConnectionState.waiting && !snapshot.hasData; final Position position = snapshot.data ?? currentPosition ?? _getDefaultPosition(); return Row( children: [ Expanded( child: ElevatedButton( onPressed: isLoading ? null : () async { await Navigator.push(context, MaterialPageRoute( builder: (context) { return Tracking( weg: rmap["Weg"]!["controller"]!, startPosition: position, ); }, )); setState(() {}); }, child: isLoading ? Row( mainAxisSize: MainAxisSize.min, children: [ SizedBox( width: 16, height: 16, child: CircularProgressIndicator( strokeWidth: 2, color: Theme.of(context).colorScheme.onPrimary, ), ), const SizedBox(width: 8), Text(AppLocalizations.of(context)! .trackingAnAusschalten), ], ) : Text(AppLocalizations.of(context)!.trackingAnAusschalten), ), ), const SizedBox(width: 8), IconButton( onPressed: isLoading ? null : _refreshPosition, icon: const Icon(Icons.refresh), tooltip: 'Position aktualisieren', ), ], ); }, ); } }