feat: Add logger service and improve expense dialog with enhanced receipt management and calculation logic.

This commit is contained in:
Van Leemput Dayron
2025-11-28 12:54:54 +01:00
parent cad9d42128
commit fd710b8cb8
35 changed files with 2148 additions and 1296 deletions

View File

@@ -3,10 +3,12 @@ import 'package:http/http.dart' as http;
import 'package:flutter_dotenv/flutter_dotenv.dart';
import '../models/trip.dart';
import 'error_service.dart';
import 'logger_service.dart';
/// Service pour géocoder les destinations des voyages
class TripGeocodingService {
static final TripGeocodingService _instance = TripGeocodingService._internal();
static final TripGeocodingService _instance =
TripGeocodingService._internal();
factory TripGeocodingService() => _instance;
TripGeocodingService._internal();
@@ -16,35 +18,41 @@ class TripGeocodingService {
/// Géocode la destination d'un voyage et retourne un Trip mis à jour
Future<Trip> geocodeTrip(Trip trip) async {
try {
print('🌍 [TripGeocoding] Géocodage de "${trip.location}"');
LoggerService.info('🌍 [TripGeocoding] Géocodage de "${trip.location}"');
// Vérifier si on a déjà des coordonnées récentes
if (trip.hasRecentCoordinates) {
print('✅ [TripGeocoding] Coordonnées récentes trouvées, pas de géocodage nécessaire');
LoggerService.info(
'✅ [TripGeocoding] Coordonnées récentes trouvées, pas de géocodage nécessaire',
);
return trip;
}
if (_apiKey.isEmpty) {
print('❌ [TripGeocoding] Clé API Google Maps manquante');
LoggerService.error('❌ [TripGeocoding] Clé API Google Maps manquante');
throw Exception('Clé API Google Maps non configurée');
}
final coordinates = await _geocodeDestination(trip.location);
if (coordinates != null) {
print('✅ [TripGeocoding] Coordonnées trouvées: ${coordinates['lat']}, ${coordinates['lng']}');
LoggerService.info(
'✅ [TripGeocoding] Coordonnées trouvées: ${coordinates['lat']}, ${coordinates['lng']}',
);
return trip.copyWith(
latitude: coordinates['lat'],
longitude: coordinates['lng'],
lastGeocodingUpdate: DateTime.now(),
);
} else {
print('⚠️ [TripGeocoding] Impossible de géocoder "${trip.location}"');
LoggerService.warning(
'⚠️ [TripGeocoding] Impossible de géocoder "${trip.location}"',
);
return trip;
}
} catch (e) {
print('❌ [TripGeocoding] Erreur lors du géocodage: $e');
LoggerService.error('❌ [TripGeocoding] Erreur lors du géocodage: $e');
_errorService.logError('trip_geocoding_service', e);
return trip; // Retourner le voyage original en cas d'erreur
}
@@ -53,18 +61,23 @@ class TripGeocodingService {
/// Géocode une destination et retourne les coordonnées
Future<Map<String, double>?> _geocodeDestination(String destination) async {
try {
final url = 'https://maps.googleapis.com/maps/api/geocode/json'
final url =
'https://maps.googleapis.com/maps/api/geocode/json'
'?address=${Uri.encodeComponent(destination)}'
'&key=$_apiKey';
print('🌐 [TripGeocoding] URL = $url');
LoggerService.info('🌐 [TripGeocoding] URL = $url');
final response = await http.get(Uri.parse(url));
print('📡 [TripGeocoding] Status code = ${response.statusCode}');
LoggerService.info(
'📡 [TripGeocoding] Status code = ${response.statusCode}',
);
if (response.statusCode == 200) {
final data = json.decode(response.body);
print('📋 [TripGeocoding] Réponse géocodage = ${data['status']}');
LoggerService.info(
'📋 [TripGeocoding] Réponse géocodage = ${data['status']}',
);
if (data['status'] == 'OK' && data['results'].isNotEmpty) {
final location = data['results'][0]['geometry']['location'];
@@ -72,18 +85,24 @@ class TripGeocodingService {
'lat': (location['lat'] as num).toDouble(),
'lng': (location['lng'] as num).toDouble(),
};
print('📍 [TripGeocoding] Coordonnées trouvées = $coordinates');
LoggerService.info(
'📍 [TripGeocoding] Coordonnées trouvées = $coordinates',
);
return coordinates;
} else {
print('⚠️ [TripGeocoding] Erreur API = ${data['error_message'] ?? data['status']}');
LoggerService.warning(
'⚠️ [TripGeocoding] Erreur API = ${data['error_message'] ?? data['status']}',
);
return null;
}
} else {
print('❌ [TripGeocoding] Erreur HTTP ${response.statusCode}');
LoggerService.error(
'❌ [TripGeocoding] Erreur HTTP ${response.statusCode}',
);
return null;
}
} catch (e) {
print('❌ [TripGeocoding] Exception lors du géocodage: $e');
LoggerService.error('❌ [TripGeocoding] Exception lors du géocodage: $e');
_errorService.logError('trip_geocoding_service', e);
return null;
}
@@ -96,23 +115,25 @@ class TripGeocodingService {
/// Géocode plusieurs voyages en batch
Future<List<Trip>> geocodeTrips(List<Trip> trips) async {
print('🔄 [TripGeocoding] Géocodage de ${trips.length} voyages');
LoggerService.info(
'🔄 [TripGeocoding] Géocodage de ${trips.length} voyages',
);
final List<Trip> geocodedTrips = [];
for (final trip in trips) {
if (needsGeocoding(trip)) {
final geocodedTrip = await geocodeTrip(trip);
geocodedTrips.add(geocodedTrip);
// Petit délai pour éviter de saturer l'API Google
await Future.delayed(const Duration(milliseconds: 200));
} else {
geocodedTrips.add(trip);
}
}
print('✅ [TripGeocoding] Géocodage terminé');
LoggerService.info('✅ [TripGeocoding] Géocodage terminé');
return geocodedTrips;
}
}
}