Enable iOS navigation UI and simulator support

Adds NavigationUIEnabledPreference.automatic to GoogleMapsNavigationView
for proper turn-by-turn navigation display on iOS. Enables navigation
header, footer, and trip progress bar. Adds simulation mode for iOS
Simulator testing with location updates along the route.

Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
Jean-Philippe Brule 2025-11-23 14:53:40 -05:00
parent 44500835d7
commit bbcd6d9bf7

View File

@ -1,3 +1,5 @@
import 'dart:io' show Platform;
import 'package:flutter/foundation.dart' show kDebugMode;
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:google_navigation_flutter/google_navigation_flutter.dart'; import 'package:google_navigation_flutter/google_navigation_flutter.dart';
import '../models/delivery.dart'; import '../models/delivery.dart';
@ -31,6 +33,7 @@ class _DarkModeMapComponentState extends State<DarkModeMapComponent> {
bool _isStartingNavigation = false; bool _isStartingNavigation = false;
String _loadingMessage = 'Initializing...'; String _loadingMessage = 'Initializing...';
Brightness? _lastBrightness; Brightness? _lastBrightness;
bool _isMapViewReady = false;
@override @override
void initState() { void initState() {
@ -242,6 +245,23 @@ class _DarkModeMapComponentState extends State<DarkModeMapComponent> {
debugPrint('Navigation started successfully'); debugPrint('Navigation started successfully');
// On iOS Simulator in debug mode, start simulation to provide location updates
// The iOS Simulator doesn't provide continuous location updates for custom locations,
// so we use the SDK's built-in simulation to simulate driving along the route.
// This is only needed for testing on iOS Simulator - real devices work without this.
if (kDebugMode && Platform.isIOS) {
try {
// Start simulating the route with a speed multiplier for testing
// speedMultiplier: 1.0 = normal speed, 5.0 = 5x faster for quicker testing
await GoogleMapsNavigator.simulator.simulateLocationsAlongExistingRouteWithOptions(
SimulationOptions(speedMultiplier: 5.0),
);
debugPrint('Simulation started for iOS Simulator testing');
} catch (e) {
debugPrint('Could not start simulation: $e');
}
}
// Reapply dark mode style after navigation starts // Reapply dark mode style after navigation starts
if (mounted) { if (mounted) {
await _applyDarkModeStyle(); await _applyDarkModeStyle();
@ -279,6 +299,17 @@ class _DarkModeMapComponentState extends State<DarkModeMapComponent> {
Future<void> _stopNavigation() async { Future<void> _stopNavigation() async {
try { try {
// Stop simulation if it was running (iOS Simulator)
if (kDebugMode && Platform.isIOS) {
try {
// Remove simulated user location to stop the simulation
await GoogleMapsNavigator.simulator.removeUserLocation();
debugPrint('Simulation stopped');
} catch (e) {
debugPrint('Could not stop simulation: $e');
}
}
await GoogleMapsNavigator.stopGuidance(); await GoogleMapsNavigator.stopGuidance();
await GoogleMapsNavigator.clearDestinations(); await GoogleMapsNavigator.clearDestinations();
if (mounted) { if (mounted) {
@ -312,7 +343,7 @@ class _DarkModeMapComponentState extends State<DarkModeMapComponent> {
// Calculate dynamic padding for bottom button bar // Calculate dynamic padding for bottom button bar
final topPadding = 0.0; final topPadding = 0.0;
final bottomPadding = 110.0; final bottomPadding = 60.0;
return Stack( return Stack(
children: [ children: [
@ -323,20 +354,40 @@ class _DarkModeMapComponentState extends State<DarkModeMapComponent> {
bottom: bottomPadding, bottom: bottomPadding,
), ),
child: GoogleMapsNavigationView( child: GoogleMapsNavigationView(
// Enable navigation UI automatically when guidance starts
// This is critical for iOS to display turn-by-turn directions, ETA, distance
initialNavigationUIEnabledPreference: NavigationUIEnabledPreference.automatic,
onViewCreated: (controller) async { onViewCreated: (controller) async {
_navigationController = controller; _navigationController = controller;
// Apply dark mode style with a small delay to ensure map is ready
await Future.delayed(const Duration(milliseconds: 500)); // Wait longer for the map to be fully initialized on Android
// This helps prevent crashes when the view is disposed during initialization
await Future.delayed(const Duration(milliseconds: 1000));
// Safety check: ensure widget is still mounted before proceeding // Safety check: ensure widget is still mounted before proceeding
if (!mounted) return; if (!mounted) return;
// Mark map as ready only after the delay
_isMapViewReady = true;
// Enable navigation UI elements (header with turn directions, footer with ETA/distance)
// This is required for iOS to show trip info, duration, and ETA
try {
await controller.setNavigationUIEnabled(true);
await controller.setNavigationHeaderEnabled(true);
await controller.setNavigationFooterEnabled(true);
await controller.setNavigationTripProgressBarEnabled(true);
debugPrint('Navigation UI elements enabled');
} catch (e) {
debugPrint('Error enabling navigation UI: $e');
}
await _applyDarkModeStyle(); await _applyDarkModeStyle();
// Wrap camera animation in try-catch to handle "No valid view found" errors // Wrap camera animation in try-catch to handle "No valid view found" errors
// This can happen on Android when the view isn't fully ready // This can happen on Android when the view isn't fully ready
try { try {
if (mounted && _navigationController != null) { if (mounted && _navigationController != null && _isMapViewReady) {
await controller.animateCamera( await controller.animateCamera(
CameraUpdate.newLatLngZoom(initialPosition, 12), CameraUpdate.newLatLngZoom(initialPosition, 12),
); );
@ -345,7 +396,7 @@ class _DarkModeMapComponentState extends State<DarkModeMapComponent> {
debugPrint('Camera animation error (view may not be ready): $e'); debugPrint('Camera animation error (view may not be ready): $e');
// Retry once after a longer delay // Retry once after a longer delay
await Future.delayed(const Duration(milliseconds: 1000)); await Future.delayed(const Duration(milliseconds: 1000));
if (mounted && _navigationController != null) { if (mounted && _navigationController != null && _isMapViewReady) {
try { try {
await controller.animateCamera( await controller.animateCamera(
CameraUpdate.newLatLngZoom(initialPosition, 12), CameraUpdate.newLatLngZoom(initialPosition, 12),
@ -379,8 +430,8 @@ class _DarkModeMapComponentState extends State<DarkModeMapComponent> {
], ],
), ),
padding: const EdgeInsets.symmetric( padding: const EdgeInsets.symmetric(
horizontal: 16, horizontal: 12,
vertical: 12, vertical: 8,
), ),
child: Row( child: Row(
children: [ children: [
@ -518,14 +569,14 @@ class _DarkModeMapComponentState extends State<DarkModeMapComponent> {
return Material( return Material(
color: backgroundColor, color: backgroundColor,
borderRadius: BorderRadius.circular(8), borderRadius: BorderRadius.circular(6),
child: InkWell( child: InkWell(
onTap: onPressed, onTap: onPressed,
borderRadius: BorderRadius.circular(8), borderRadius: BorderRadius.circular(6),
child: Padding( child: Padding(
padding: const EdgeInsets.symmetric( padding: const EdgeInsets.symmetric(
horizontal: 14, horizontal: 8,
vertical: 14, vertical: 10,
), ),
child: Row( child: Row(
mainAxisAlignment: MainAxisAlignment.center, mainAxisAlignment: MainAxisAlignment.center,
@ -534,15 +585,15 @@ class _DarkModeMapComponentState extends State<DarkModeMapComponent> {
Icon( Icon(
icon, icon,
color: textColor, color: textColor,
size: 24, size: 18,
), ),
const SizedBox(width: 8), const SizedBox(width: 6),
Text( Text(
label, label,
style: TextStyle( style: TextStyle(
color: textColor, color: textColor,
fontWeight: FontWeight.w600, fontWeight: FontWeight.w600,
fontSize: 18, fontSize: 14,
), ),
), ),
], ],