Update: 2026-06-21 18:58:05
This commit is contained in:
@@ -0,0 +1,187 @@
|
||||
import 'dart:async';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:get/get.dart';
|
||||
import 'package:intaleq_maps/intaleq_maps.dart';
|
||||
import 'package:siro_driver/constant/links.dart';
|
||||
import 'package:siro_driver/controller/functions/crud.dart';
|
||||
import 'package:siro_driver/controller/functions/tts.dart';
|
||||
import 'package:siro_driver/views/widgets/error_snakbar.dart';
|
||||
|
||||
class DestinationController extends GetxController {
|
||||
bool isLoading = false;
|
||||
bool isSelectingDestinationOnMap = false;
|
||||
|
||||
// Active destination details
|
||||
LatLng? activeLatLng;
|
||||
String activeName = "";
|
||||
|
||||
final TextEditingController searchController = TextEditingController();
|
||||
List<dynamic> placesSuggestions = [];
|
||||
Timer? _debounce;
|
||||
|
||||
@override
|
||||
void onInit() {
|
||||
super.onInit();
|
||||
fetchActiveDestination();
|
||||
}
|
||||
|
||||
@override
|
||||
void onClose() {
|
||||
searchController.dispose();
|
||||
_debounce?.cancel();
|
||||
super.onClose();
|
||||
}
|
||||
|
||||
// --- 1. Get current active destination from backend ---
|
||||
Future<void> fetchActiveDestination() async {
|
||||
isLoading = true;
|
||||
update();
|
||||
|
||||
try {
|
||||
final response = await CRUD().post(
|
||||
link: AppLink.saveDriverDestination,
|
||||
payload: {'action': 'get'},
|
||||
);
|
||||
|
||||
if (response != null && response is Map && response['status'] == 'success') {
|
||||
final data = response['message'];
|
||||
if (data != null) {
|
||||
final lat = double.tryParse(data['target_latitude']?.toString() ?? '0') ?? 0.0;
|
||||
final lng = double.tryParse(data['target_longitude']?.toString() ?? '0') ?? 0.0;
|
||||
activeLatLng = LatLng(lat, lng);
|
||||
activeName = data['destination_name']?.toString() ?? "";
|
||||
} else {
|
||||
activeLatLng = null;
|
||||
activeName = "";
|
||||
}
|
||||
}
|
||||
} catch (e) {
|
||||
print("[DestinationController] fetchActiveDestination error: $e");
|
||||
} finally {
|
||||
isLoading = false;
|
||||
update();
|
||||
}
|
||||
}
|
||||
|
||||
// --- 2. Autocomplete search suggestions ---
|
||||
void onSearchChanged(String query) {
|
||||
if (_debounce?.isActive ?? false) _debounce!.cancel();
|
||||
_debounce = Timer(const Duration(milliseconds: 500), () => getPlacesSuggestions(query));
|
||||
}
|
||||
|
||||
Future<void> getPlacesSuggestions(String query) async {
|
||||
if (query.trim().length < 3) {
|
||||
placesSuggestions = [];
|
||||
update();
|
||||
return;
|
||||
}
|
||||
|
||||
try {
|
||||
final url = '${AppLink.mapSaasPlaces}?q=${Uri.encodeComponent(query)}';
|
||||
final response = await CRUD().getMapSaas(link: url);
|
||||
|
||||
if (response != null && response is List) {
|
||||
placesSuggestions = response;
|
||||
} else {
|
||||
placesSuggestions = [];
|
||||
}
|
||||
update();
|
||||
} catch (e) {
|
||||
print("[DestinationController] getPlacesSuggestions error: $e");
|
||||
}
|
||||
}
|
||||
|
||||
// --- 3. Save destination to database ---
|
||||
Future<void> saveDestination(LatLng position, String name) async {
|
||||
isLoading = true;
|
||||
update();
|
||||
|
||||
try {
|
||||
final response = await CRUD().post(
|
||||
link: AppLink.saveDriverDestination,
|
||||
payload: {
|
||||
'action': 'set',
|
||||
'destination_lat': position.latitude.toString(),
|
||||
'destination_lng': position.longitude.toString(),
|
||||
'destination_name': name,
|
||||
},
|
||||
);
|
||||
|
||||
if (response != null && response is Map) {
|
||||
if (response['status'] == 'success') {
|
||||
activeLatLng = position;
|
||||
activeName = name;
|
||||
mySnackbarSuccess('Destination set successfully!'.tr);
|
||||
} else {
|
||||
final msg = response['message']?.toString() ?? "";
|
||||
if (msg.contains("الحد الأقصى") || msg.contains("limit")) {
|
||||
mySnackbarWarning('You have reached the daily limit of 2 destination settings.'.tr);
|
||||
} else {
|
||||
mySnackbarWarning('Failed to set destination. Please try again.'.tr);
|
||||
}
|
||||
}
|
||||
} else {
|
||||
mySnackbarWarning('Failed to set destination. Please try again.'.tr);
|
||||
}
|
||||
} catch (e) {
|
||||
print("[DestinationController] saveDestination error: $e");
|
||||
mySnackbarWarning('Failed to set destination. Please try again.'.tr);
|
||||
} finally {
|
||||
isLoading = false;
|
||||
update();
|
||||
}
|
||||
}
|
||||
|
||||
// --- 4. Clear/Deactivate destination ---
|
||||
Future<void> clearDestination() async {
|
||||
isLoading = true;
|
||||
update();
|
||||
|
||||
try {
|
||||
final response = await CRUD().post(
|
||||
link: AppLink.saveDriverDestination,
|
||||
payload: {'action': 'clear'},
|
||||
);
|
||||
|
||||
if (response != null && response is Map && response['status'] == 'success') {
|
||||
activeLatLng = null;
|
||||
activeName = "";
|
||||
mySnackbarSuccess('Destination cleared!'.tr);
|
||||
} else {
|
||||
mySnackbarWarning('Failed to set destination. Please try again.'.tr);
|
||||
}
|
||||
} catch (e) {
|
||||
print("[DestinationController] clearDestination error: $e");
|
||||
mySnackbarWarning('Failed to set destination. Please try again.'.tr);
|
||||
} finally {
|
||||
isLoading = false;
|
||||
update();
|
||||
}
|
||||
}
|
||||
|
||||
// --- 5. Reverse geocoding (for map picker) ---
|
||||
Future<String> reverseGeocode(LatLng location) async {
|
||||
final url = '${AppLink.reverseGeocoding}?lat=${location.latitude}&lng=${location.longitude}';
|
||||
try {
|
||||
final response = await CRUD().getMapSaas(link: url);
|
||||
if (response != null && response is List && response.isNotEmpty) {
|
||||
final data = response[0];
|
||||
return data['name_ar'] ?? data['name'] ?? 'Selected Location'.tr;
|
||||
}
|
||||
return 'Selected Location'.tr;
|
||||
} catch (e) {
|
||||
print("[DestinationController] reverseGeocode error: $e");
|
||||
return 'Selected Location'.tr;
|
||||
}
|
||||
}
|
||||
|
||||
// --- 6. TTS Voice Explanation Helper ---
|
||||
void toggleSpeaker() {
|
||||
final tts = Get.find<TextToSpeechController>();
|
||||
if (tts.isSpeaking) {
|
||||
tts.stop();
|
||||
} else {
|
||||
tts.speakText('Destination Tool Description'.tr);
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -2659,4 +2659,20 @@ final Map<String, String> ar_eg = {
|
||||
"🏆 \\\${'Maximum Level Reached!": "🏆 \\\${'تم الوصول إلى المستوى الأقصى!",
|
||||
"💰 Pay with Wallet": "💰 ادفع عبر المحفظة",
|
||||
"💳 Pay with Credit Card": "💳 ادفع ببطاقة ائتمان",
|
||||
"Set Destination": "تحديد الوجهة",
|
||||
"Personal Destination": "الوجهة الشخصية",
|
||||
"Set your destination to only receive orders heading in that direction.": "حدد وجهتك لتلقي الطلبات المتجهة في نفس المسار فقط.",
|
||||
"Explain Destination Tool": "شرح أداة تحديد الوجهة",
|
||||
"Destination Tool Description": "تتيح لك هذه الأداة تحديد وجهتك الشخصية التي تنوي الذهاب إليها (مثل طريق عودتك للمنزل). بمجرد تفعيلها، سيقوم النظام تلقائياً بفلترة طلبات الركاب وعرض الرحلات المتوافقة مع مسارك فقط.\n\nتنبيه هام:\n1. يمكنك تحديد الوجهة مرتين كحد أقصى يومياً.\n2. يتم مطابقة الرحلات التي تقع وجهتها في طريق مسارك وبفارق بسيط لضمان عدم الخروج عن مسارك.",
|
||||
"Search for area or city...": "ابحث عن منطقة أو مدينة...",
|
||||
"Select on Map": "تحديد من الخريطة",
|
||||
"Confirm Location": "تأكيد الموقع",
|
||||
"Choose Destination": "اختر الوجهة",
|
||||
"Daily Limit Reached": "تم الوصول للحد اليومي",
|
||||
"You have reached the daily limit of 2 destination settings.": "لقد وصلت للحد اليومي الأقصى المسموح به لتحديد الوجهة (مرتان يومياً).",
|
||||
"Destination set successfully!": "تم تحديد الوجهة بنجاح!",
|
||||
"Failed to set destination. Please try again.": "فشل تحديد الوجهة. يرجى المحاولة لاحقاً.",
|
||||
"Clear Destination": "مسح الوجهة",
|
||||
"Destination cleared!": "تم مسح الوجهة الشخصية!",
|
||||
"Move map to select destination": "حرك الخريطة لتحديد الوجهة الشخصية",
|
||||
};
|
||||
|
||||
@@ -2659,4 +2659,20 @@ final Map<String, String> ar_jo = {
|
||||
"🏆 \\\${'Maximum Level Reached!": "🏆 \\\${'تم الوصول إلى المستوى الأقصى!",
|
||||
"💰 Pay with Wallet": "💰 ادفع عبر المحفظة",
|
||||
"💳 Pay with Credit Card": "💳 ادفع ببطاقة ائتمان",
|
||||
"Set Destination": "تحديد الوجهة",
|
||||
"Personal Destination": "الوجهة الشخصية",
|
||||
"Set your destination to only receive orders heading in that direction.": "حدد وجهتك لتلقي الطلبات المتجهة في نفس المسار فقط.",
|
||||
"Explain Destination Tool": "شرح أداة تحديد الوجهة",
|
||||
"Destination Tool Description": "تتيح لك هذه الأداة تحديد وجهتك الشخصية التي تنوي الذهاب إليها (مثل طريق عودتك للمنزل). بمجرد تفعيلها، سيقوم النظام تلقائياً بفلترة طلبات الركاب وعرض الرحلات المتوافقة مع مسارك فقط.\n\nتنبيه هام:\n1. يمكنك تحديد الوجهة مرتين كحد أقصى يومياً.\n2. يتم مطابقة الرحلات التي تقع وجهتها في طريق مسارك وبفارق بسيط لضمان عدم الخروج عن مسارك.",
|
||||
"Search for area or city...": "ابحث عن منطقة أو مدينة...",
|
||||
"Select on Map": "تحديد من الخريطة",
|
||||
"Confirm Location": "تأكيد الموقع",
|
||||
"Choose Destination": "اختر الوجهة",
|
||||
"Daily Limit Reached": "تم الوصول للحد اليومي",
|
||||
"You have reached the daily limit of 2 destination settings.": "لقد وصلت للحد اليومي الأقصى المسموح به لتحديد الوجهة (مرتان يومياً).",
|
||||
"Destination set successfully!": "تم تحديد الوجهة بنجاح!",
|
||||
"Failed to set destination. Please try again.": "فشل تحديد الوجهة. يرجى المحاولة لاحقاً.",
|
||||
"Clear Destination": "مسح الوجهة",
|
||||
"Destination cleared!": "تم مسح الوجهة الشخصية!",
|
||||
"Move map to select destination": "حرك الخريطة لتحديد الوجهة الشخصية",
|
||||
};
|
||||
|
||||
@@ -2062,6 +2062,24 @@ final Map<String, String> ar_sy = {
|
||||
"You have gift 30000 EGP": "عندك هدية 30000 ج.م",
|
||||
"You have gift 30000 JOD": "عندك هدية 30000 د.أ",
|
||||
"You have gift 30000 SYP": "عندك هدية 30000 ل.س",
|
||||
"💰 Pay with Wallet": "💰 ادفع عبر المحفظة",
|
||||
"💳 Pay with Credit Card": "💳 ادفع ببطاقة ائتمان",
|
||||
"Set Destination": "تحديد الوجهة",
|
||||
"Personal Destination": "الوجهة الشخصية",
|
||||
"Set your destination to only receive orders heading in that direction.": "حدد وجهتك لتلقي الطلبات المتجهة في نفس المسار فقط.",
|
||||
"Explain Destination Tool": "شرح أداة تحديد الوجهة",
|
||||
"Destination Tool Description": "تتيح لك هذه الأداة تحديد وجهتك الشخصية التي تنوي الذهاب إليها (مثل طريق عودتك للمنزل). بمجرد تفعيلها، سيقوم النظام تلقائياً بفلترة طلبات الركاب وعرض الرحلات المتوافقة مع مسارك فقط.\n\nتنبيه هام:\n1. يمكنك تحديد الوجهة مرتين كحد أقصى يومياً.\n2. يتم مطابقة الرحلات التي تقع وجهتها في طريق مسارك وبفارق بسيط لضمان عدم الخروج عن مسارك.",
|
||||
"Search for area or city...": "ابحث عن منطقة أو مدينة...",
|
||||
"Select on Map": "تحديد من الخريطة",
|
||||
"Confirm Location": "تأكيد الموقع",
|
||||
"Choose Destination": "اختر الوجهة",
|
||||
"Daily Limit Reached": "تم الوصول للحد اليومي",
|
||||
"You have reached the daily limit of 2 destination settings.": "لقد وصلت للحد اليومي الأقصى المسموح به لتحديد الوجهة (مرتان يومياً).",
|
||||
"Destination set successfully!": "تم تحديد الوجهة بنجاح!",
|
||||
"Failed to set destination. Please try again.": "فشل تحديد الوجهة. يرجى المحاولة لاحقاً.",
|
||||
"Clear Destination": "مسح الوجهة",
|
||||
"Destination cleared!": "تم مسح الوجهة الشخصية!",
|
||||
"Move map to select destination": "حرك الخريطة لتحديد الوجهة الشخصية",
|
||||
"You have got a gift": "وصلتك هدية",
|
||||
"You have got a gift for invitation": "وصلتك هدية للدعوة",
|
||||
"You have in account": "لديك في الحساب",
|
||||
|
||||
@@ -2659,4 +2659,20 @@ final Map<String, String> en = {
|
||||
"🏆 \\\${'Maximum Level Reached!": "🏆 \\\${'Maximum Level Reached!",
|
||||
"💰 Pay with Wallet": "💰 Pay with Wallet",
|
||||
"💳 Pay with Credit Card": "💳 Pay with Credit Card",
|
||||
"Set Destination": "Set Destination",
|
||||
"Personal Destination": "Personal Destination",
|
||||
"Set your destination to only receive orders heading in that direction.": "Set your destination to only receive orders heading in that direction.",
|
||||
"Explain Destination Tool": "Explain Destination Tool",
|
||||
"Destination Tool Description": "This tool allows you to specify a personal destination (e.g. your way home). Once activated, the system filters passenger requests and only displays rides compatible with your route.\n\nImportant Notes:\n1. You can set the destination a maximum of 2 times daily.\n2. Matches are computed based on paths that align with your route to ensure minimal detours.",
|
||||
"Search for area or city...": "Search for area or city...",
|
||||
"Select on Map": "Select on Map",
|
||||
"Confirm Location": "Confirm Location",
|
||||
"Choose Destination": "Choose Destination",
|
||||
"Daily Limit Reached": "Daily Limit Reached",
|
||||
"You have reached the daily limit of 2 destination settings.": "You have reached the daily limit of 2 destination settings.",
|
||||
"Destination set successfully!": "Destination set successfully!",
|
||||
"Failed to set destination. Please try again.": "Failed to set destination. Please try again.",
|
||||
"Clear Destination": "Clear Destination",
|
||||
"Destination cleared!": "Destination cleared!",
|
||||
"Move map to select destination": "Move map to select destination",
|
||||
};
|
||||
|
||||
Reference in New Issue
Block a user