2026-03-07 17:11:02 -08:00
|
|
|
import 'dart:typed_data';
|
Add TWP Softphone Flutter app and complete mobile backend API
Backend: Add /voice/token endpoint with AccessToken + VoiceGrant for
mobile VoIP, implement unhold_call() with call leg detection, wire FCM
push notifications into call queue and webhook missed call handlers,
add data-only FCM message support for Android background wake, and add
Twilio API Key / Push Credential settings fields.
Flutter app: Full softphone with Twilio Voice SDK integration, JWT auth
with auto-refresh, SSE real-time queue updates, FCM push notifications,
Material 3 UI with dashboard, active call screen, dialpad, and call
controls (mute/speaker/hold/transfer).
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-06 13:01:23 -08:00
|
|
|
import 'package:firebase_core/firebase_core.dart';
|
|
|
|
|
import 'package:firebase_messaging/firebase_messaging.dart';
|
2026-03-07 17:11:02 -08:00
|
|
|
import 'package:flutter/foundation.dart';
|
Add TWP Softphone Flutter app and complete mobile backend API
Backend: Add /voice/token endpoint with AccessToken + VoiceGrant for
mobile VoIP, implement unhold_call() with call leg detection, wire FCM
push notifications into call queue and webhook missed call handlers,
add data-only FCM message support for Android background wake, and add
Twilio API Key / Push Credential settings fields.
Flutter app: Full softphone with Twilio Voice SDK integration, JWT auth
with auto-refresh, SSE real-time queue updates, FCM push notifications,
Material 3 UI with dashboard, active call screen, dialpad, and call
controls (mute/speaker/hold/transfer).
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-06 13:01:23 -08:00
|
|
|
import 'package:flutter_local_notifications/flutter_local_notifications.dart';
|
|
|
|
|
import 'api_client.dart';
|
|
|
|
|
|
2026-03-07 17:11:02 -08:00
|
|
|
/// Notification ID for queue alerts (fixed so we can cancel it).
|
|
|
|
|
const int _queueAlertNotificationId = 9001;
|
|
|
|
|
|
|
|
|
|
/// Background handler — must be top-level function.
|
Add TWP Softphone Flutter app and complete mobile backend API
Backend: Add /voice/token endpoint with AccessToken + VoiceGrant for
mobile VoIP, implement unhold_call() with call leg detection, wire FCM
push notifications into call queue and webhook missed call handlers,
add data-only FCM message support for Android background wake, and add
Twilio API Key / Push Credential settings fields.
Flutter app: Full softphone with Twilio Voice SDK integration, JWT auth
with auto-refresh, SSE real-time queue updates, FCM push notifications,
Material 3 UI with dashboard, active call screen, dialpad, and call
controls (mute/speaker/hold/transfer).
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-06 13:01:23 -08:00
|
|
|
@pragma('vm:entry-point')
|
|
|
|
|
Future<void> _firebaseMessagingBackgroundHandler(RemoteMessage message) async {
|
|
|
|
|
await Firebase.initializeApp();
|
2026-03-07 17:11:02 -08:00
|
|
|
final data = message.data;
|
|
|
|
|
final type = data['type'];
|
|
|
|
|
|
|
|
|
|
if (type == 'queue_alert') {
|
|
|
|
|
await _showQueueAlertNotification(data);
|
|
|
|
|
} else if (type == 'queue_alert_cancel') {
|
|
|
|
|
final plugin = FlutterLocalNotificationsPlugin();
|
|
|
|
|
await plugin.cancel(_queueAlertNotificationId);
|
|
|
|
|
}
|
|
|
|
|
// VoIP pushes handled natively by twilio_voice plugin.
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Show an insistent queue alert notification (works from background handler too).
|
|
|
|
|
Future<void> _showQueueAlertNotification(Map<String, dynamic> data) async {
|
|
|
|
|
final plugin = FlutterLocalNotificationsPlugin();
|
|
|
|
|
|
|
|
|
|
final title = data['title'] ?? 'Call Waiting';
|
|
|
|
|
final body = data['body'] ?? 'New call in queue';
|
|
|
|
|
|
|
|
|
|
final androidDetails = AndroidNotificationDetails(
|
|
|
|
|
'twp_queue_alerts',
|
|
|
|
|
'Queue Alerts',
|
|
|
|
|
channelDescription: 'Alerts when calls are waiting in queue',
|
|
|
|
|
importance: Importance.max,
|
|
|
|
|
priority: Priority.max,
|
|
|
|
|
playSound: true,
|
|
|
|
|
sound: const RawResourceAndroidNotificationSound('queue_alert'),
|
|
|
|
|
enableVibration: true,
|
|
|
|
|
vibrationPattern: Int64List.fromList([0, 500, 200, 500, 200, 500]),
|
|
|
|
|
ongoing: true,
|
|
|
|
|
autoCancel: false,
|
|
|
|
|
category: AndroidNotificationCategory.alarm,
|
|
|
|
|
additionalFlags: Int32List.fromList([4]), // FLAG_INSISTENT = 4
|
|
|
|
|
fullScreenIntent: true,
|
|
|
|
|
visibility: NotificationVisibility.public,
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
await plugin.show(
|
|
|
|
|
_queueAlertNotificationId,
|
|
|
|
|
title,
|
|
|
|
|
body,
|
|
|
|
|
NotificationDetails(android: androidDetails),
|
|
|
|
|
);
|
Add TWP Softphone Flutter app and complete mobile backend API
Backend: Add /voice/token endpoint with AccessToken + VoiceGrant for
mobile VoIP, implement unhold_call() with call leg detection, wire FCM
push notifications into call queue and webhook missed call handlers,
add data-only FCM message support for Android background wake, and add
Twilio API Key / Push Credential settings fields.
Flutter app: Full softphone with Twilio Voice SDK integration, JWT auth
with auto-refresh, SSE real-time queue updates, FCM push notifications,
Material 3 UI with dashboard, active call screen, dialpad, and call
controls (mute/speaker/hold/transfer).
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-06 13:01:23 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class PushNotificationService {
|
|
|
|
|
final ApiClient _api;
|
|
|
|
|
final FirebaseMessaging _messaging = FirebaseMessaging.instance;
|
|
|
|
|
final FlutterLocalNotificationsPlugin _localNotifications =
|
|
|
|
|
FlutterLocalNotificationsPlugin();
|
2026-03-07 17:11:02 -08:00
|
|
|
String? _fcmToken;
|
|
|
|
|
|
|
|
|
|
String? get fcmToken => _fcmToken;
|
Add TWP Softphone Flutter app and complete mobile backend API
Backend: Add /voice/token endpoint with AccessToken + VoiceGrant for
mobile VoIP, implement unhold_call() with call leg detection, wire FCM
push notifications into call queue and webhook missed call handlers,
add data-only FCM message support for Android background wake, and add
Twilio API Key / Push Credential settings fields.
Flutter app: Full softphone with Twilio Voice SDK integration, JWT auth
with auto-refresh, SSE real-time queue updates, FCM push notifications,
Material 3 UI with dashboard, active call screen, dialpad, and call
controls (mute/speaker/hold/transfer).
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-06 13:01:23 -08:00
|
|
|
|
|
|
|
|
PushNotificationService(this._api);
|
|
|
|
|
|
|
|
|
|
Future<void> initialize() async {
|
|
|
|
|
FirebaseMessaging.onBackgroundMessage(_firebaseMessagingBackgroundHandler);
|
|
|
|
|
|
|
|
|
|
await _messaging.requestPermission(
|
|
|
|
|
alert: true,
|
|
|
|
|
badge: true,
|
|
|
|
|
sound: true,
|
|
|
|
|
criticalAlert: true,
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Initialize local notifications
|
|
|
|
|
const androidSettings =
|
|
|
|
|
AndroidInitializationSettings('@mipmap/ic_launcher');
|
|
|
|
|
const initSettings = InitializationSettings(android: androidSettings);
|
|
|
|
|
await _localNotifications.initialize(initSettings);
|
|
|
|
|
|
|
|
|
|
// Get and register FCM token
|
|
|
|
|
final token = await _messaging.getToken();
|
2026-03-07 17:11:02 -08:00
|
|
|
debugPrint('FCM token: ${token != null ? "${token.substring(0, 20)}..." : "NULL"}');
|
Add TWP Softphone Flutter app and complete mobile backend API
Backend: Add /voice/token endpoint with AccessToken + VoiceGrant for
mobile VoIP, implement unhold_call() with call leg detection, wire FCM
push notifications into call queue and webhook missed call handlers,
add data-only FCM message support for Android background wake, and add
Twilio API Key / Push Credential settings fields.
Flutter app: Full softphone with Twilio Voice SDK integration, JWT auth
with auto-refresh, SSE real-time queue updates, FCM push notifications,
Material 3 UI with dashboard, active call screen, dialpad, and call
controls (mute/speaker/hold/transfer).
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-06 13:01:23 -08:00
|
|
|
if (token != null) {
|
2026-03-07 17:11:02 -08:00
|
|
|
_fcmToken = token;
|
Add TWP Softphone Flutter app and complete mobile backend API
Backend: Add /voice/token endpoint with AccessToken + VoiceGrant for
mobile VoIP, implement unhold_call() with call leg detection, wire FCM
push notifications into call queue and webhook missed call handlers,
add data-only FCM message support for Android background wake, and add
Twilio API Key / Push Credential settings fields.
Flutter app: Full softphone with Twilio Voice SDK integration, JWT auth
with auto-refresh, SSE real-time queue updates, FCM push notifications,
Material 3 UI with dashboard, active call screen, dialpad, and call
controls (mute/speaker/hold/transfer).
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-06 13:01:23 -08:00
|
|
|
await _registerToken(token);
|
2026-03-07 17:11:02 -08:00
|
|
|
} else {
|
|
|
|
|
debugPrint('FCM: Failed to get token - Firebase may not be configured correctly');
|
Add TWP Softphone Flutter app and complete mobile backend API
Backend: Add /voice/token endpoint with AccessToken + VoiceGrant for
mobile VoIP, implement unhold_call() with call leg detection, wire FCM
push notifications into call queue and webhook missed call handlers,
add data-only FCM message support for Android background wake, and add
Twilio API Key / Push Credential settings fields.
Flutter app: Full softphone with Twilio Voice SDK integration, JWT auth
with auto-refresh, SSE real-time queue updates, FCM push notifications,
Material 3 UI with dashboard, active call screen, dialpad, and call
controls (mute/speaker/hold/transfer).
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-06 13:01:23 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Listen for token refresh
|
|
|
|
|
_messaging.onTokenRefresh.listen(_registerToken);
|
|
|
|
|
|
|
|
|
|
// Handle foreground messages (non-VoIP)
|
|
|
|
|
FirebaseMessaging.onMessage.listen(_handleForegroundMessage);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Future<void> _registerToken(String token) async {
|
|
|
|
|
try {
|
|
|
|
|
await _api.dio.post('/fcm/register', data: {'fcm_token': token});
|
|
|
|
|
} catch (_) {}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void _handleForegroundMessage(RemoteMessage message) {
|
|
|
|
|
final data = message.data;
|
|
|
|
|
final type = data['type'];
|
|
|
|
|
|
|
|
|
|
// VoIP incoming_call is handled by twilio_voice natively
|
|
|
|
|
if (type == 'incoming_call') return;
|
|
|
|
|
|
2026-03-07 17:11:02 -08:00
|
|
|
// Queue alert — show insistent notification
|
|
|
|
|
if (type == 'queue_alert') {
|
|
|
|
|
_showQueueAlertNotification(data);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Queue alert cancel — dismiss notification
|
|
|
|
|
if (type == 'queue_alert_cancel') {
|
|
|
|
|
_localNotifications.cancel(_queueAlertNotificationId);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Show local notification for other types (missed call, etc.)
|
Add TWP Softphone Flutter app and complete mobile backend API
Backend: Add /voice/token endpoint with AccessToken + VoiceGrant for
mobile VoIP, implement unhold_call() with call leg detection, wire FCM
push notifications into call queue and webhook missed call handlers,
add data-only FCM message support for Android background wake, and add
Twilio API Key / Push Credential settings fields.
Flutter app: Full softphone with Twilio Voice SDK integration, JWT auth
with auto-refresh, SSE real-time queue updates, FCM push notifications,
Material 3 UI with dashboard, active call screen, dialpad, and call
controls (mute/speaker/hold/transfer).
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-06 13:01:23 -08:00
|
|
|
_localNotifications.show(
|
|
|
|
|
message.hashCode,
|
|
|
|
|
data['title'] ?? 'TWP Softphone',
|
|
|
|
|
data['body'] ?? '',
|
|
|
|
|
const NotificationDetails(
|
|
|
|
|
android: AndroidNotificationDetails(
|
|
|
|
|
'twp_general',
|
|
|
|
|
'General Notifications',
|
|
|
|
|
importance: Importance.high,
|
|
|
|
|
priority: Priority.high,
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
);
|
|
|
|
|
}
|
2026-03-07 17:11:02 -08:00
|
|
|
|
|
|
|
|
/// Cancel any active queue alert (called when agent accepts a call in-app).
|
|
|
|
|
void cancelQueueAlert() {
|
|
|
|
|
_localNotifications.cancel(_queueAlertNotificationId);
|
|
|
|
|
}
|
Add TWP Softphone Flutter app and complete mobile backend API
Backend: Add /voice/token endpoint with AccessToken + VoiceGrant for
mobile VoIP, implement unhold_call() with call leg detection, wire FCM
push notifications into call queue and webhook missed call handlers,
add data-only FCM message support for Android background wake, and add
Twilio API Key / Push Credential settings fields.
Flutter app: Full softphone with Twilio Voice SDK integration, JWT auth
with auto-refresh, SSE real-time queue updates, FCM push notifications,
Material 3 UI with dashboard, active call screen, dialpad, and call
controls (mute/speaker/hold/transfer).
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-06 13:01:23 -08:00
|
|
|
}
|