import 'dart:async'; import 'dart:convert'; import 'dart:io'; import 'dart:typed_data'; import 'package:SEFER/constant/colors.dart'; import 'package:SEFER/views/home/Captin/orderCaptin/order_request_page.dart'; import 'package:SEFER/views/home/Captin/orderCaptin/order_speed_request.dart'; import 'package:flutter_local_notifications/flutter_local_notifications.dart'; import 'package:get/get.dart'; import 'package:permission_handler/permission_handler.dart'; import 'package:timezone/data/latest.dart' as tz; import 'package:timezone/timezone.dart' as tz; import '../../constant/box_name.dart'; import '../../main.dart'; import '../../print.dart'; import '../../views/notification/notification_captain.dart'; import '../home/captin/home_captain_controller.dart'; class NotificationController extends GetxController { final FlutterLocalNotificationsPlugin _flutterLocalNotificationsPlugin = FlutterLocalNotificationsPlugin(); Future initNotifications() async { const AndroidInitializationSettings android = AndroidInitializationSettings('@mipmap/launcher_icon'); DarwinInitializationSettings ios = DarwinInitializationSettings( requestAlertPermission: true, requestBadgePermission: true, requestSoundPermission: true, onDidReceiveLocalNotification: (int id, String? title, String? body, String? payload) async {}, ); InitializationSettings initializationSettings = InitializationSettings(android: android, iOS: ios); tz.initializeTimeZones(); print('Notifications initialized'); await _flutterLocalNotificationsPlugin.initialize( initializationSettings, onDidReceiveNotificationResponse: onDidReceiveNotificationResponse, onDidReceiveBackgroundNotificationResponse: notificationTapBackground, ); // Create a notification channel const AndroidNotificationChannel channel = AndroidNotificationChannel( 'high_importance_channel', // Use the same ID as in strings.xml 'High Importance Notifications', description: 'This channel is used for important notifications.', importance: Importance.high, ); // Register the channel with the system await _flutterLocalNotificationsPlugin .resolvePlatformSpecificImplementation< AndroidFlutterLocalNotificationsPlugin>() ?.createNotificationChannel(channel); } // Displays a notification with the given title and message void showNotification( String title, String message, String tone, String payLoad) async { BigTextStyleInformation bigTextStyleInformation = BigTextStyleInformation( message, contentTitle: title.tr, htmlFormatContent: true, htmlFormatContentTitle: true, ); AndroidNotificationDetails android = AndroidNotificationDetails( 'high_importance_channel', // Use the same ID as before 'High Importance Notifications', importance: Importance.high, priority: Priority.high, styleInformation: bigTextStyleInformation, playSound: true, sound: RawResourceAndroidNotificationSound(tone), // audioAttributesUsage: AudioAttributesUsage.alarm, visibility: NotificationVisibility.public, autoCancel: false, color: AppColor.primaryColor, showProgress: true, showWhen: true, ongoing: true, enableVibration: true, vibrationPattern: Int64List.fromList([0, 1000, 500, 1000]), timeoutAfter: 14500, setAsGroupSummary: true, subText: message, fullScreenIntent: true, actions: [ AndroidNotificationAction( allowGeneratedReplies: true, 'id', title.tr, titleColor: AppColor.bronze, showsUserInterface: true, ) ], category: AndroidNotificationCategory.progress, ); DarwinNotificationDetails ios = const DarwinNotificationDetails( sound: 'default', presentAlert: true, presentBadge: true, presentSound: true, ); NotificationDetails details = NotificationDetails(android: android, iOS: ios); await _flutterLocalNotificationsPlugin.show(0, title, message, details, payload: jsonEncode({'title': title, 'data': payLoad})); } void scheduleNotificationAtSpecificTime( String title, String message, String tone, int hour, int minute) async { // Initialize and set Cairo time zone tz.initializeTimeZones(); var cairoLocation; if (box.read(BoxName.countryCode).toString() == 'Egypt') { cairoLocation = tz.getLocation('Africa/Cairo'); } else {} // todo get for location country final AndroidNotificationDetails android = AndroidNotificationDetails( 'high_importance_channel', 'High Importance Notifications', importance: Importance.max, priority: Priority.high, showWhen: false, sound: RawResourceAndroidNotificationSound(tone), ); const DarwinNotificationDetails ios = DarwinNotificationDetails( sound: 'default', presentAlert: true, presentBadge: true, presentSound: true, ); final NotificationDetails details = NotificationDetails(android: android, iOS: ios); final now = tz.TZDateTime.now(cairoLocation); // Use Cairo timezone for current time tz.TZDateTime scheduledTime = tz.TZDateTime( cairoLocation, now.year, now.month, now.day, hour, minute); // If the scheduled time has already passed for today, schedule it for the next day if (scheduledTime.isBefore(now)) { scheduledTime = scheduledTime.add(const Duration(days: 1)); } if (Platform.isAndroid) { if (await Permission.scheduleExactAlarm.isDenied) { if (await Permission.scheduleExactAlarm.request().isGranted) { print('SCHEDULE_EXACT_ALARM permission granted'); } else { print('SCHEDULE_EXACT_ALARM permission denied'); return; } } } print('Current time: $now'); print('Scheduling notification for: $scheduledTime'); await _flutterLocalNotificationsPlugin.zonedSchedule( 0, title, message, scheduledTime, details, androidAllowWhileIdle: true, uiLocalNotificationDateInterpretation: UILocalNotificationDateInterpretation.absoluteTime, matchDateTimeComponents: DateTimeComponents.time, // Triggers daily at the same time ); print('Notification scheduled successfully'); } void scheduleNotificationAfter1Minute( String title, String message, String tone) async { final AndroidNotificationDetails android = AndroidNotificationDetails( 'high_importance_channel', 'High Importance Notifications', importance: Importance.max, priority: Priority.high, showWhen: false, sound: RawResourceAndroidNotificationSound(tone), ); const DarwinNotificationDetails ios = DarwinNotificationDetails( sound: 'default', presentAlert: true, presentBadge: true, presentSound: true, ); final NotificationDetails details = NotificationDetails(android: android, iOS: ios); // Schedule the notification to be shown after 1 minute Timer.periodic(const Duration(seconds: 15), (timer) async { final now = tz.TZDateTime.now(tz.local); final scheduledTime = now.add(const Duration(seconds: 10)); Log.print('scheduledTime: ${scheduledTime}'); if (Platform.isAndroid) { if (await Permission.scheduleExactAlarm.isDenied) { if (await Permission.scheduleExactAlarm.request().isGranted) { print('SCHEDULE_EXACT_ALARM permission granted'); } else { print('SCHEDULE_EXACT_ALARM permission denied'); return; } } } print('Scheduling notification for: $scheduledTime'); await _flutterLocalNotificationsPlugin.zonedSchedule( 0, title, message, scheduledTime, details, androidAllowWhileIdle: true, uiLocalNotificationDateInterpretation: UILocalNotificationDateInterpretation.absoluteTime, matchDateTimeComponents: DateTimeComponents.time, ); print('Notification scheduled successfully'); }); } void scheduleDailyNotifications( String title, String message, String tone) async { final AndroidNotificationDetails android = AndroidNotificationDetails( 'high_importance_channel', 'High Importance Notifications', importance: Importance.max, priority: Priority.high, sound: RawResourceAndroidNotificationSound(tone), ); const DarwinNotificationDetails ios = DarwinNotificationDetails( sound: 'default', presentAlert: true, presentBadge: true, presentSound: true, ); final NotificationDetails details = NotificationDetails(android: android, iOS: ios); // Check for the exact alarm permission on Android 12 and above if (Platform.isAndroid) { if (await Permission.scheduleExactAlarm.isDenied) { if (await Permission.scheduleExactAlarm.request().isGranted) { print('SCHEDULE_EXACT_ALARM permission granted'); } else { print('SCHEDULE_EXACT_ALARM permission denied'); return; } } } // Schedule notifications for 10:00 AM and 3:00 PM daily await _scheduleNotificationForTime(7, 0, title, message, details); await _scheduleNotificationForTime(13, 0, title, message, details); await _scheduleNotificationForTime(18, 0, title, message, details); // await _scheduleNotificationForTime(0, 22, title, message, details); print('Daily notifications scheduled successfully'); } // Helper function to get the next instance of a specific hour and minute Future _scheduleNotificationForTime(int hour, int minute, String title, String message, NotificationDetails details) async { // Initialize and set Cairo timezone tz.initializeTimeZones(); final cairoLocation = tz.getLocation('Africa/Cairo'); // Set Cairo timezone final now = tz.TZDateTime.now( cairoLocation); // Use Cairo timezone for the current time tz.TZDateTime scheduledDate = tz.TZDateTime( cairoLocation, now.year, now.month, now.day, hour, minute); // If scheduled time is already past today, schedule it for the next day if (scheduledDate.isBefore(now)) { scheduledDate = scheduledDate.add(const Duration(days: 1)); } print('Current time (Cairo): $now'); print('Scheduling notification for: $scheduledDate'); await _flutterLocalNotificationsPlugin.zonedSchedule( 0, // Use unique IDs if you want to manage each notification separately title, message, scheduledDate, details, androidAllowWhileIdle: true, uiLocalNotificationDateInterpretation: UILocalNotificationDateInterpretation.absoluteTime, matchDateTimeComponents: DateTimeComponents.time, ); print('Notification scheduled successfully for Cairo timezone'); } void scheduleNotificationEvery10Hours( String title, String message, String tone) async { final AndroidNotificationDetails android = AndroidNotificationDetails( 'high_importance_channel', 'High Importance Notifications', importance: Importance.max, priority: Priority.high, showWhen: false, sound: RawResourceAndroidNotificationSound(tone), ); const DarwinNotificationDetails ios = DarwinNotificationDetails( sound: 'default', presentAlert: true, presentBadge: true, presentSound: true, ); final NotificationDetails details = NotificationDetails(android: android, iOS: ios); if (Platform.isAndroid) { if (await Permission.scheduleExactAlarm.isDenied) { if (await Permission.scheduleExactAlarm.request().isGranted) { print('SCHEDULE_EXACT_ALARM permission granted'); } else { print('SCHEDULE_EXACT_ALARM permission denied'); return; } } } Timer.periodic(const Duration(hours: 10), (timer) async { final now = tz.TZDateTime.now(tz.local); final scheduledTime = now.add(const Duration(minutes: 10)); print('Scheduling notification for: $scheduledTime'); await _flutterLocalNotificationsPlugin.zonedSchedule( 0, title.tr, message.tr, scheduledTime, details, androidAllowWhileIdle: true, uiLocalNotificationDateInterpretation: UILocalNotificationDateInterpretation.absoluteTime, matchDateTimeComponents: DateTimeComponents.time, ); }); print('Notifications scheduled every 5 seconds'); } // Callback when the notification is tapped void onDidReceiveNotificationResponse(NotificationResponse response) { handleNotificationResponse(response); } // Callback when the notification is tapped while the app is in the background void onDidReceiveBackgroundNotificationResponse( NotificationResponse response) { handleNotificationResponse(response); } // Handle notification response for both foreground and background void handleNotificationResponse(NotificationResponse response) { print('Notification tapped!'); Log.print('response.payload: ${response.payload}'); if (response.payload != null) { print('Notification payload: ${response.payload}'); var payloadData = jsonDecode(response.payload.toString()); if (payloadData is Map) { String title = payloadData['title']; var data = payloadData['data']; switch (title) { case 'Order': _handleOrderNotification(data); break; case 'OrderSpeed': _handleOrderSpeedNotification(data); break; case 'ADS': _handleADSNotification(); break; default: Log.print('Unknown notification type'); } } else { Log.print('Invalid payload format'); } } else { Log.print('Payload is null'); } } void _handleOrderNotification(dynamic data) { if (data is String) { var orderData = jsonDecode(data); if (orderData is List && orderData.length == 34) { closeOverLay(); Get.put(HomeCaptainController()).changeRideId(); Get.to(() => OrderRequestPage(), arguments: {'myListString': data}); } else { Log.print('Invalid order data'); } } else { Log.print('Invalid order payload'); } } void _handleOrderSpeedNotification(dynamic data) { if (data is String) { var orderData = jsonDecode(data); if (orderData is List && orderData.length == 34) { closeOverLay(); Get.put(HomeCaptainController()).changeRideId(); Get.to(() => OrderSpeedRequest(), arguments: {'myListString': data}); } else { Log.print('Invalid order data'); } } else { Log.print('Invalid order payload'); } } void _handleADSNotification() { // var orderData = jsonDecode(data); closeOverLay(); Get.to( () => const NotificationCaptain(), ); } void onDidReceiveLocalNotification( int id, String? title, String? body, String? payload) async { // display a dialog with the notification details, tap ok to go to another page } } class NotificationController1 extends GetxController { final FlutterLocalNotificationsPlugin _flutterLocalNotificationsPlugin = FlutterLocalNotificationsPlugin(); // Initializes the local notifications plugin Future initNotifications() async { const AndroidInitializationSettings android = AndroidInitializationSettings('@mipmap/launcher_icon'); const InitializationSettings initializationSettings = InitializationSettings(android: android); await _flutterLocalNotificationsPlugin.initialize(initializationSettings); } // Displays a notification with the given title and message void showNotification( String title, String message, String tone, String payLoad) async { AndroidNotificationDetails android = AndroidNotificationDetails( 'your channel id', 'your channel name', importance: Importance.max, priority: Priority.high, showWhen: false, sound: RawResourceAndroidNotificationSound(tone)); NotificationDetails details = NotificationDetails(android: android); await _flutterLocalNotificationsPlugin.show(0, title, message, details); } }