import 'package:flutter/material.dart'; import 'package:provider/provider.dart'; import '../app_state.dart'; import '../dev_config.dart'; import '../models/node_profile.dart'; import '../models/operator_profile.dart'; import '../services/localization_service.dart'; import '../services/node_cache.dart'; import '../services/changelog_service.dart'; import '../state/settings_state.dart'; import 'refine_tags_sheet.dart'; import 'advanced_edit_options_sheet.dart'; import 'proximity_warning_dialog.dart'; import 'submission_guide_dialog.dart'; import 'positioning_tutorial_overlay.dart'; class EditNodeSheet extends StatefulWidget { const EditNodeSheet({super.key, required this.session}); final EditNodeSession session; @override State createState() => _EditNodeSheetState(); } class _EditNodeSheetState extends State { bool _showTutorial = false; bool _isCheckingTutorial = true; @override void initState() { super.initState(); _checkTutorialStatus(); } Future _checkTutorialStatus() async { final hasCompleted = await ChangelogService().hasCompletedPositioningTutorial(); if (mounted) { setState(() { _showTutorial = !hasCompleted; _isCheckingTutorial = false; }); // If tutorial should be shown, register callback with AppState if (_showTutorial) { final appState = context.read(); appState.registerTutorialCallback(_hideTutorial); } } } void _hideTutorial() { if (mounted && _showTutorial) { setState(() { _showTutorial = false; }); } } @override void dispose() { // Clear tutorial callback when widget is disposed if (_showTutorial) { try { context.read().clearTutorialCallback(); } catch (e) { // Context might be unavailable during disposal, ignore } } super.dispose(); } void _checkProximityAndCommit(BuildContext context, AppState appState, LocalizationService locService) { _checkSubmissionGuideAndProceed(context, appState, locService); } void _checkSubmissionGuideAndProceed(BuildContext context, AppState appState, LocalizationService locService) async { // Check if user has seen the submission guide final hasSeenGuide = await ChangelogService().hasSeenSubmissionGuide(); if (!hasSeenGuide) { // Show submission guide dialog first final shouldProceed = await showDialog( context: context, barrierDismissible: false, builder: (context) => const SubmissionGuideDialog(), ); // If user canceled the submission guide, don't proceed with submission if (shouldProceed != true) { return; } } // Now proceed with proximity check _checkProximityOnly(context, appState, locService); } void _checkProximityOnly(BuildContext context, AppState appState, LocalizationService locService) { // Check for nearby nodes within the configured distance, excluding the node being edited final nearbyNodes = NodeCache.instance.findNodesWithinDistance( widget.session.target, kNodeProximityWarningDistance, excludeNodeId: widget.session.originalNode.id, ); if (nearbyNodes.isNotEmpty) { // Show proximity warning dialog showDialog( context: context, builder: (context) => ProximityWarningDialog( nearbyNodes: nearbyNodes, distance: kNodeProximityWarningDistance, onGoBack: () { Navigator.of(context).pop(); // Close dialog }, onSubmitAnyway: () { Navigator.of(context).pop(); // Close dialog _commitWithoutCheck(context, appState, locService); }, ), ); } else { // No nearby nodes, proceed with commit _commitWithoutCheck(context, appState, locService); } } void _commitWithoutCheck(BuildContext context, AppState appState, LocalizationService locService) { appState.commitEditSession(); Navigator.pop(context); ScaffoldMessenger.of(context).showSnackBar( SnackBar(content: Text(locService.t('node.editQueuedForUpload'))), ); } Widget _buildDirectionControls(BuildContext context, AppState appState, EditNodeSession session, LocalizationService locService) { final requiresDirection = session.profile != null && session.profile!.requiresDirection; // Format direction display text with bold for current direction String directionsText = ''; if (requiresDirection) { final directionsWithBold = []; for (int i = 0; i < session.directions.length; i++) { final dirStr = session.directions[i].round().toString(); if (i == session.currentDirectionIndex) { directionsWithBold.add('**$dirStr**'); // Mark for bold formatting } else { directionsWithBold.add(dirStr); } } directionsText = directionsWithBold.join(', '); } return Column( children: [ ListTile( title: requiresDirection ? RichText( text: TextSpan( style: Theme.of(context).textTheme.titleMedium, children: [ const TextSpan(text: 'Directions: '), if (directionsText.isNotEmpty) ...directionsText.split('**').asMap().entries.map((entry) { final isEven = entry.key % 2 == 0; return TextSpan( text: entry.value, style: TextStyle( fontWeight: isEven ? FontWeight.normal : FontWeight.bold, ), ); }), ], ), ) : Text(locService.t('editNode.direction', params: [session.directionDegrees.round().toString()])), subtitle: Row( children: [ // Slider takes most of the space Expanded( child: Slider( min: 0, max: 359, divisions: 359, value: session.directionDegrees, label: session.directionDegrees.round().toString(), onChanged: requiresDirection ? (v) => appState.updateEditSession(directionDeg: v) : null, ), ), // Direction control buttons - always show but grey out when direction not required const SizedBox(width: 8), // Remove button IconButton( icon: Icon( Icons.remove, size: 20, color: requiresDirection ? null : Theme.of(context).disabledColor, ), onPressed: requiresDirection && session.directions.length > 1 ? () => appState.removeDirection() : null, tooltip: requiresDirection ? 'Remove current direction' : 'Direction not required for this profile', padding: EdgeInsets.zero, constraints: const BoxConstraints(minWidth: kDirectionButtonMinWidth, minHeight: kDirectionButtonMinHeight), ), // Add button IconButton( icon: Icon( Icons.add, size: 20, color: requiresDirection && session.directions.length < 8 ? null : Theme.of(context).disabledColor, ), onPressed: requiresDirection && session.directions.length < 8 ? () => appState.addDirection() : null, tooltip: requiresDirection ? (session.directions.length >= 8 ? 'Maximum 8 directions allowed' : 'Add new direction') : 'Direction not required for this profile', padding: EdgeInsets.zero, constraints: const BoxConstraints(minWidth: kDirectionButtonMinWidth, minHeight: kDirectionButtonMinHeight), ), // Cycle button IconButton( icon: Icon( Icons.repeat, size: 20, color: requiresDirection ? null : Theme.of(context).disabledColor, ), onPressed: requiresDirection && session.directions.length > 1 ? () => appState.cycleDirection() : null, tooltip: requiresDirection ? 'Cycle through directions' : 'Direction not required for this profile', padding: EdgeInsets.zero, constraints: const BoxConstraints(minWidth: kDirectionButtonMinWidth, minHeight: kDirectionButtonMinHeight), ), ], ), ), // Show info text when profile doesn't require direction if (!requiresDirection) Padding( padding: const EdgeInsets.fromLTRB(16, 0, 16, 12), child: Row( children: [ const Icon(Icons.info_outline, color: Colors.grey, size: 16), const SizedBox(width: 6), Expanded( child: Text( 'This profile does not require a direction.', style: const TextStyle(color: Colors.grey, fontSize: 12), ), ), ], ), ), ], ); } @override Widget build(BuildContext context) { return AnimatedBuilder( animation: LocalizationService.instance, builder: (context, child) { final locService = LocalizationService.instance; final appState = context.watch(); void _commit() { _checkProximityAndCommit(context, appState, locService); } void _cancel() { appState.cancelEditSession(); Navigator.pop(context); } final session = widget.session; final submittableProfiles = appState.enabledProfiles.where((p) => p.isSubmittable).toList(); final isSandboxMode = appState.uploadMode == UploadMode.sandbox; final allowSubmit = kEnableNodeEdits && appState.isLoggedIn && submittableProfiles.isNotEmpty && session.profile != null && session.profile!.isSubmittable; void _navigateToLogin() { Navigator.pushNamed(context, '/settings/osm-account'); } void _openRefineTags() async { final result = await Navigator.push( context, MaterialPageRoute( builder: (context) => RefineTagsSheet( selectedOperatorProfile: session.operatorProfile, selectedProfile: session.profile, currentRefinedTags: session.refinedTags, ), fullscreenDialog: true, ), ); if (result != null) { appState.updateEditSession( operatorProfile: result.operatorProfile, refinedTags: result.refinedTags, ); } } return Stack( clipBehavior: Clip.none, fit: StackFit.loose, children: [ Column( mainAxisSize: MainAxisSize.min, children: [ const SizedBox(height: 12), Container( width: 40, height: 4, decoration: BoxDecoration( color: Colors.grey.shade400, borderRadius: BorderRadius.circular(2), ), ), const SizedBox(height: 8), Text( locService.t('editNode.title', params: [session.originalNode.id.toString()]), style: Theme.of(context).textTheme.titleMedium, ), const SizedBox(height: 16), ListTile( title: Text(locService.t('editNode.profile')), trailing: DropdownButton( value: session.profile, hint: Text(locService.t('editNode.selectProfile')), items: submittableProfiles .map((p) => DropdownMenuItem(value: p, child: Text(p.name))) .toList(), onChanged: (p) => appState.updateEditSession(profile: p), ), ), // Direction controls _buildDirectionControls(context, appState, session, locService), // Constraint message for nodes that cannot be moved if (session.originalNode.isConstrained) Padding( padding: const EdgeInsets.fromLTRB(16, 0, 16, 12), child: Column( children: [ // Extract from way checkbox (only show if enabled in dev config) if (kEnableNodeExtraction) ...[ CheckboxListTile( title: Text(locService.t('editNode.extractFromWay')), subtitle: Text(locService.t('editNode.extractFromWaySubtitle')), value: session.extractFromWay, onChanged: (value) { appState.updateEditSession(extractFromWay: value); }, controlAffinity: ListTileControlAffinity.leading, contentPadding: EdgeInsets.zero, ), const SizedBox(height: 8), ], // Constraint info message (only show if extract is not checked or not enabled) if (!kEnableNodeExtraction || !session.extractFromWay) ...[ Row( children: [ const Icon(Icons.info_outline, size: 20), const SizedBox(width: 8), Expanded( child: Text( locService.t('editNode.cannotMoveConstrainedNode'), style: Theme.of(context).textTheme.bodyMedium, ), ), ], ), const SizedBox(height: 8), ], Row( mainAxisAlignment: MainAxisAlignment.end, children: [ OutlinedButton.icon( onPressed: () => _openAdvancedEdit(context), icon: const Icon(Icons.open_in_new, size: 16), label: Text(locService.t('actions.useAdvancedEditor')), style: OutlinedButton.styleFrom( minimumSize: const Size(0, 32), ), ), ], ), ], ), ), if (!kEnableNodeEdits) Padding( padding: const EdgeInsets.fromLTRB(16, 0, 16, 12), child: Row( children: [ const Icon(Icons.construction, color: Colors.orange, size: 20), const SizedBox(width: 6), Expanded( child: Text( locService.t('editNode.temporarilyDisabled'), style: const TextStyle(color: Colors.orange, fontSize: 13), ), ), ], ), ) else if (!appState.isLoggedIn) Padding( padding: const EdgeInsets.fromLTRB(16, 0, 16, 12), child: Row( children: [ const Icon(Icons.info_outline, color: Colors.red, size: 20), const SizedBox(width: 6), Expanded( child: Text( locService.t('editNode.mustBeLoggedIn'), style: const TextStyle(color: Colors.red, fontSize: 13), ), ), ], ), ) else if (submittableProfiles.isEmpty) Padding( padding: const EdgeInsets.fromLTRB(16, 0, 16, 12), child: Row( children: [ const Icon(Icons.info_outline, color: Colors.red, size: 20), const SizedBox(width: 6), Expanded( child: Text( locService.t('editNode.enableSubmittableProfile'), style: const TextStyle(color: Colors.red, fontSize: 13), ), ), ], ), ) else if (session.profile == null) Padding( padding: const EdgeInsets.fromLTRB(16, 0, 16, 12), child: Row( children: [ const Icon(Icons.info_outline, color: Colors.orange, size: 20), const SizedBox(width: 6), Expanded( child: Text( locService.t('editNode.profileRequired'), style: const TextStyle(color: Colors.orange, fontSize: 13), ), ), ], ), ) else if (!session.profile!.isSubmittable) Padding( padding: const EdgeInsets.fromLTRB(16, 0, 16, 12), child: Row( children: [ const Icon(Icons.info_outline, color: Colors.orange, size: 20), const SizedBox(width: 6), Expanded( child: Text( locService.t('editNode.profileViewOnlyWarning'), style: const TextStyle(color: Colors.orange, fontSize: 13), ), ), ], ), ), const SizedBox(height: 8), Padding( padding: const EdgeInsets.symmetric(horizontal: 16.0), child: SizedBox( width: double.infinity, child: OutlinedButton.icon( onPressed: session.profile != null ? _openRefineTags : null, // Disabled when no profile selected icon: const Icon(Icons.tune), label: Text(session.operatorProfile != null ? locService.t('editNode.refineTagsWithProfile', params: [session.operatorProfile!.name]) : locService.t('editNode.refineTags')), ), ), ), const SizedBox(height: 8), Padding( padding: const EdgeInsets.symmetric(horizontal: 16.0), child: Row( children: [ Expanded( child: OutlinedButton( onPressed: _cancel, child: Text(locService.cancel), ), ), const SizedBox(width: 12), Expanded( child: ElevatedButton( onPressed: !appState.isLoggedIn ? _navigateToLogin : (allowSubmit ? _commit : null), child: Text(!appState.isLoggedIn ? locService.t('actions.logIn') : locService.t('actions.saveEdit')), ), ), ], ), ), const SizedBox(height: 20), ], ), // Tutorial overlay - show only if tutorial should be shown and we're done checking if (!_isCheckingTutorial && _showTutorial) Positioned.fill( child: PositioningTutorialOverlay(), ), ], ); }, ); } void _openAdvancedEdit(BuildContext context) { showModalBottomSheet( context: context, isScrollControlled: true, builder: (context) => AdvancedEditOptionsSheet(node: widget.session.originalNode), ); } }