我们正在开发一个加密的聊天应用程序,我们使用Firebase消息传递进行数据通知。某些客户端逻辑需要在收到数据通知后完成,然后再向用户显示实际通知。例如,必须将电话号码转换为本地联系人姓名。此转换是通过查找已在全球范围内可用的地图来完成的。
数据通知接收得很好,并且还调用了 onBackgroundMessage 回调。但是,似乎不可能从onBackgroundMessage函数访问任何类型的状态。例如,打印登录用户的电话号码将返回 null。 从onMessage回调打印相同的全局变量就可以了。
从onMessage运行flutter_local_notifications工作正常,但同样,从onBackgroundMessage根本不起作用,因为"找不到.show((方法的实现"。目前,它声称flutterLocalNotificationsPlugin为空,但事实并非如此。
在我们看来,一旦应用程序在后台运行,onBackgroundMessage 就无法访问应用程序提供的任何内容。必须执行一些操作才能使某些范围/上下文可用于后台进程。目前,这主要是整个flutter_local_notifications插件,以及将电话号码转换为姓名的本地联系人列表。
有没有人知道如何做到这一点?
以下是一些代码:
FlutterLocalNotificationsPlugin flutterLocalNotificationsPlugin;
final _chatRepository = ChatRepository();
Future<dynamic> backgroundMessageHandler(Map<String, dynamic> message) async {
if(message.containsKey('data')) {
await _showNotification(message);
return Future<void>.value();
}
}
Future _showNotification(message) async {
List<String> numbers = [];
numbers.add(message['data']['sender']);
var name = await _chatRepository.translatePhoneNumbersToChatName(numbers);
var androidPlatformChannelSpecifics = new AndroidNotificationDetails(
'channel id', 'channel name', 'channel description',
importance: Importance.Max, priority: Priority.High);
var iOSPlatformChannelSpecifics = new IOSNotificationDetails();
var platformChannelSpecifics = new NotificationDetails(
androidPlatformChannelSpecifics, iOSPlatformChannelSpecifics);
await flutterLocalNotificationsPlugin.show(
0,
name,
message['data']['body'],
platformChannelSpecifics,
payload: message['data']['body'],
);
}
class NotificationHandler {
final FirebaseMessaging fcm = FirebaseMessaging();
StreamSubscription iosSubscription;
String deviceToken = "";
Future<void> initialize() async {
flutterLocalNotificationsPlugin = new FlutterLocalNotificationsPlugin();
var initializationSettingsAndroid =
new AndroidInitializationSettings('@mipmap/ic_launcher');
var initializationSettingsIOS = new IOSInitializationSettings(onDidReceiveLocalNotification: onDidReceiveLocalNotification);
var initializationSettings = new InitializationSettings(initializationSettingsAndroid, initializationSettingsIOS);
flutterLocalNotificationsPlugin.initialize(initializationSettings, onSelectNotification: onClickNotification);
fcm.configure(
onMessage: (Map<String, dynamic> message) async {
if(message.containsKey('data')) {
print(message);
_showNotification(message);
}
},
onBackgroundMessage: Platform.isIOS
? null
: backgroundMessageHandler,
onLaunch: (Map<String, dynamic> message) async {
if(message.containsKey('data')) {
print(message);
_showNotification(message);
}
},
onResume: (Map<String, dynamic> message) async {
if(message.containsKey('data')) {
print(message);
_showNotification(message);
}
},
);
_updateDeviceToken();
}
.
.
.
Of course, the initialize above is called early on in the application lifecycle.
class NotificationHandler {
static final flutterLocalNotificationsPlugin = FlutterLocalNotificationsPlugin(); // make it a static field of the class
// ...
}
Future _showNotification(message) async {
// ...
await NotificationHandler.flutterLocalNotificationsPlugin.show( // access it
// ...
);
}
希望这对你有用。
这个插件比我更好地解释了这一切,但碰巧背景是一个完全不同的隔离/上下文,因此如果他们使用旧的(Flutter 12 之前(API,它无法访问任何插件。 https://pub.dev/packages/android_alarm_manager#flutter-android-embedding-v1
嵌入 v1 要求您注册要从后台访问的任何插件。这样做可以使其flutter_local_notifications正常工作。
不幸的是,FCM文档严重缺乏。