import 'dart:async'; import 'dart:convert'; import 'dart:io'; import 'dart:math'; import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:get/get.dart'; import 'package:get_storage/get_storage.dart'; import 'package:intl/date_symbol_data_local.dart'; import 'package:wakelock_plus/wakelock_plus.dart'; import 'package:firebase_core/firebase_core.dart'; import 'package:firebase_messaging/firebase_messaging.dart'; import 'package:flutter_local_notifications/flutter_local_notifications.dart'; import 'package:flutter_overlay_window/flutter_overlay_window.dart'; import 'package:flutter_secure_storage/flutter_secure_storage.dart'; import 'package:flutter_stripe/flutter_stripe.dart'; import 'constant/api_key.dart'; import 'constant/info.dart'; import 'constant/notification.dart'; import 'controller/firebase/firbase_messge.dart'; import 'controller/firebase/local_notification.dart'; import 'controller/functions/add_error.dart'; import 'controller/functions/battery_status.dart'; import 'controller/functions/crud.dart'; import 'controller/functions/encrypt_decrypt.dart'; import 'controller/functions/secure_storage.dart'; import 'controller/local/local_controller.dart'; import 'controller/local/translations.dart'; import 'firebase_options.dart'; import 'models/db_sql.dart'; import 'print.dart'; import 'splash_screen_page.dart'; import 'views/home/Captin/orderCaptin/order_request_page.dart'; import 'views/home/Captin/driver_map_page.dart'; import 'views/home/Captin/orderCaptin/order_over_lay.dart'; final box = GetStorage(); const storage = FlutterSecureStorage(); DbSql sql = DbSql.instance; final GlobalKey navigatorKey = GlobalKey(); const platform = MethodChannel('com.example.intaleq_driver/app_control'); /// تهيئة Firebase بوعي لمنع تهيئة مكرّرة على أندرويد (isolates متعددة) Future initFirebaseIfNeeded() async { if (Firebase.apps.isEmpty) { await Firebase.initializeApp( options: DefaultFirebaseOptions.currentPlatform); } else { Firebase.app(); } } /// ============ Handlers: Background ============ @pragma('vm:entry-point') Future backgroundMessageHandler(RemoteMessage message) async { WidgetsFlutterBinding.ensureInitialized(); await initFirebaseIfNeeded(); await GetStorage.init(); if (!Get.isRegistered()) { Get.put(NotificationController()); } if (!Get.isRegistered()) { Get.put(FirebaseMessagesController()); } if (!await FlutterOverlayWindow.isPermissionGranted()) { Log.print("Overlay permission not granted; showing only notification."); } if (Platform.isAndroid) { if (message.notification != null && message.notification!.title != null) { Log.print('message.notification!.title: ${message.notification!.title}'); if (message.notification?.title == 'Order' || message.notification?.title == 'OrderSpeed') { final myListString = message.data['DriverList'] ?? '[]'; Log.print('myListString: $myListString'); List myList; try { myList = jsonDecode(myListString) as List; } catch (e) { Log.print('Error decoding JSON: $e'); myList = []; } final isOverlayActive = await FlutterOverlayWindow.isActive(); if (isOverlayActive) { await FlutterOverlayWindow.shareData(myList); } else { await FlutterOverlayWindow.showOverlay( enableDrag: true, flag: OverlayFlag.focusPointer, positionGravity: PositionGravity.auto, height: 1400, width: WindowSize.matchParent, startPosition: const OverlayPosition(0, -30), ); await FlutterOverlayWindow.shareData(myList); } NotificationController().showNotification( message.notification!.title.toString(), message.notification!.body.toString(), 'order', myListString, ); } else { FirebaseMessagesController().fireBaseTitles(message); } } } } @pragma('vm:entry-point') void notificationTapBackground(NotificationResponse notificationResponse) { Log.print('Notification tapped in background!'); NotificationController().handleNotificationResponse(notificationResponse); } /// ============ Entrypoint: Overlay ============ @pragma('vm:entry-point') void overlayMain() async { WidgetsFlutterBinding.ensureInitialized(); await GetStorage.init(); await initFirebaseIfNeeded(); if (!Get.isRegistered()) { Get.put(NotificationController()); } runApp(const MaterialApp( debugShowCheckedModeBanner: false, home: OrderOverlay(), )); } /// إغلاق الـ Overlay عند الحاجة Future closeOverLay() async { final isOverlayActive = await FlutterOverlayWindow.isActive(); if (isOverlayActive) { await FlutterOverlayWindow.closeOverlay(); } } /// ============ Entrypoint: App ============ void main() { runZonedGuarded(() async { WidgetsFlutterBinding.ensureInitialized(); await initFirebaseIfNeeded(); await WakelockPlus.enable(); await GetStorage.init(); await initializeDateFormatting(); Stripe.publishableKey = AK.publishableKeyStripe; await SystemChrome.setPreferredOrientations([ DeviceOrientation.portraitUp, DeviceOrientation.portraitDown, ]); // سجل الهاندلر تبع رسائل الخلفية (لازم يكون Top-Level ومع @pragma) FirebaseMessaging.onBackgroundMessage(backgroundMessageHandler); runApp(const MyApp()); }, (error, stack) { // ==== START: ERROR FILTER ==== final errorString = error.toString(); // اطبع كل شيء محلياً // (يمكنك استبدال print بـ Log.print إن رغبت) print("Caught Dart error: $error"); print(stack); // تجاهُل بعض الأخطاء المعروفة final isIgnoredError = errorString.contains('PERMISSION_DENIED') || errorString.contains('FormatException') || errorString.contains('Null check operator used on a null value'); if (!isIgnoredError) { // أرسل فقط ما ليس ضمن قائمة التجاهل CRUD.addError(error.toString(), stack.toString(), 'main'); } else { print("Ignoring error and not sending to server: $errorString"); } // ==== END: ERROR FILTER ==== }); } class MyApp extends StatefulWidget { const MyApp({Key? key}) : super(key: key); @override _MyAppState createState() => _MyAppState(); } class _MyAppState extends State with WidgetsBindingObserver { @override void initState() { super.initState(); WidgetsBinding.instance.addObserver(this); _initApp(); } @override void dispose() { WidgetsBinding.instance.removeObserver(this); super.dispose(); } Future _initApp() async { try { final AppInitializer initializer = AppInitializer(); await initializer.initializeApp(); await EncryptionHelper.initialize(); if (!Get.isRegistered()) { Get.put(NotificationController()); } if (!Get.isRegistered()) { Get.put(FirebaseMessagesController()); } await FirebaseMessaging.instance.requestPermission(); // يمكن أيضاً تفعيل foreground presentation options هنا لو احتجت await NotificationController().initNotifications(); // Generate a random index to pick a message final random = Random(); final randomMessage = syrianDriverMessages[random.nextInt(syrianDriverMessages.length)]; // Schedule the notification with the random message NotificationController().scheduleNotificationsForSevenDays( randomMessage.split(':')[0], randomMessage.split(':')[1], "tone1", ); } catch (e) { Log.print("Error during _initApp: $e"); } } @override Widget build(BuildContext context) { final LocaleController localController = Get.put(LocaleController()); return GetMaterialApp( navigatorKey: navigatorKey, title: AppInformation.appName, translations: MyTranslation(), debugShowCheckedModeBanner: false, locale: localController.language, theme: localController.appTheme, initialRoute: '/', getPages: [ GetPage(name: '/', page: () => SplashScreen()), GetPage(name: '/order-page', page: () => OrderRequestPage()), GetPage( name: '/passenger-location-map', page: () => PassengerLocationMapPage(), ), ], ); } }