Bug fixes in notification and support iOS Communication Notification!

This commit is contained in:
LittleSheep 2024-07-21 23:43:18 +08:00
parent dbd05dbb79
commit 62edab0131
3 changed files with 53 additions and 40 deletions

View File

@ -8,55 +8,65 @@
import UserNotifications
import Intents
enum ParseNotificationPayloadError: Error {
case noMetadata(String)
case noAvatarUrl(String)
}
class NotificationService: UNNotificationServiceExtension {
var contentHandler: ((UNNotificationContent) -> Void)?
var bestAttemptContent: UNMutableNotificationContent?
func didReceive(_ request: UNNotificationRequest, withContentHandler contentHandler: @escaping (UNNotificationContent) -> Void) async {
override func didReceive(_ request: UNNotificationRequest, withContentHandler contentHandler: @escaping (UNNotificationContent) -> Void) {
self.contentHandler = contentHandler
bestAttemptContent = (request.content.mutableCopy() as? UNMutableNotificationContent)
if let bestAttemptContent = bestAttemptContent {
do {
switch bestAttemptContent.categoryIdentifier {
case "messaging.message", "messaging.callStart":
let metadata = bestAttemptContent.userInfo["metadata"] as! [AnyHashable : Any]
let handle = INPersonHandle(value: String(metadata["user_id"] as! Int), type: .unknown)
guard let metadata = bestAttemptContent.userInfo["metadata"] as? [AnyHashable : Any] else {
throw ParseNotificationPayloadError.noMetadata("The notification has no metadata.")
}
let userId = metadata["user_id"] as! Int
let userName = metadata["user_name"] as? String
let userNick = metadata["user_nick"] as? String
guard let avatarUrl = bestAttemptContent.userInfo["avatar"] as? String else {
throw ParseNotificationPayloadError.noMetadata("The notification has no avatar url.")
}
let handle = INPersonHandle(value: String(userId), type: .unknown)
let avatar = INImage(
url: URL(string: bestAttemptContent.userInfo["avatar"] as! String)!
url: URL(string: avatarUrl)!
)!
let sender = INPerson(personHandle: handle,
nameComponents: nil,
displayName: metadata["user_nick"] as? String,
displayName: userNick,
image: avatar,
contactIdentifier: nil,
customIdentifier: nil)
customIdentifier: userName)
let intent = INSendMessageIntent(recipients: nil,
outgoingMessageType: .outgoingMessageText,
content: bestAttemptContent.body,
speakableGroupName: nil,
conversationIdentifier: String(metadata["channel_id"] as! Int),
serviceName: nil,
serviceName: "PostPigeon",
sender: sender,
attachments: nil)
let interaction = INInteraction(intent: intent, response: nil)
interaction.direction = .incoming
do {
try await interaction.donate()
let updatedContent = try request.content.updating(from: intent)
contentHandler(updatedContent)
} catch {
return
}
interaction.donate(completion: nil)
break
default:
contentHandler(bestAttemptContent)
break
}
} catch {
contentHandler(bestAttemptContent)
}
}
}

View File

@ -9,7 +9,7 @@ class Notification {
String? avatar;
String? picture;
int? senderId;
int recipientId;
int accountId;
Notification({
required this.id,
@ -22,7 +22,7 @@ class Notification {
required this.avatar,
required this.picture,
required this.senderId,
required this.recipientId,
required this.accountId,
});
factory Notification.fromJson(Map<String, dynamic> json) => Notification(
@ -40,7 +40,7 @@ class Notification {
avatar: json['avatar'],
picture: json['picture'],
senderId: json['sender_id'],
recipientId: json['recipient_id'],
accountId: json['account_id'],
);
Map<String, dynamic> toJson() => {
@ -54,6 +54,6 @@ class Notification {
'avatar': avatar,
'picture': picture,
'sender_id': senderId,
'recipient_id': recipientId,
'account_id': accountId,
};
}

View File

@ -127,6 +127,8 @@ class WebSocketProvider extends GetxController {
if (deviceUuid == null || deviceUuid.isEmpty) {
log("Unable to active push notifications, couldn't get device uuid");
} else {
log('Device UUID is $deviceUuid');
}
if (PlatformInfo.isIOS || PlatformInfo.isMacOS) {
@ -136,6 +138,7 @@ class WebSocketProvider extends GetxController {
provider = 'firebase';
token = await FirebaseMessaging.instance.getToken();
}
log('Device Push Token is $token');
final client = auth.configureClient('auth');