185 lines
6.8 KiB
Swift
185 lines
6.8 KiB
Swift
//
|
|
// Copyright 2023 New Vector Ltd
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
//
|
|
|
|
import Foundation
|
|
import Intents
|
|
import UserNotifications
|
|
|
|
enum NotificationIconType {
|
|
case sender(mediaSource: MediaSourceProxy?)
|
|
case group(mediaSource: MediaSourceProxy?, groupName: String)
|
|
|
|
var mediaSource: MediaSourceProxy? {
|
|
switch self {
|
|
case .group(let mediaSource, _), .sender(let mediaSource):
|
|
return mediaSource
|
|
}
|
|
}
|
|
|
|
var isSenderIcon: Bool {
|
|
switch self {
|
|
case .sender:
|
|
return true
|
|
case .group:
|
|
return false
|
|
}
|
|
}
|
|
}
|
|
|
|
extension UNNotificationContent {
|
|
@objc var receiverID: String? {
|
|
userInfo[NotificationConstants.UserInfoKey.receiverIdentifier] as? String
|
|
}
|
|
|
|
@objc var notificationID: String? {
|
|
userInfo[NotificationConstants.UserInfoKey.notificationIdentifier] as? String
|
|
}
|
|
|
|
@objc var roomID: String? {
|
|
userInfo[NotificationConstants.UserInfoKey.roomIdentifier] as? String
|
|
}
|
|
|
|
@objc var eventID: String? {
|
|
userInfo[NotificationConstants.UserInfoKey.eventIdentifier] as? String
|
|
}
|
|
}
|
|
|
|
extension UNMutableNotificationContent {
|
|
override var receiverID: String? {
|
|
get {
|
|
userInfo[NotificationConstants.UserInfoKey.receiverIdentifier] as? String
|
|
}
|
|
set {
|
|
userInfo[NotificationConstants.UserInfoKey.receiverIdentifier] = newValue
|
|
}
|
|
}
|
|
|
|
override var roomID: String? {
|
|
get {
|
|
userInfo[NotificationConstants.UserInfoKey.roomIdentifier] as? String
|
|
}
|
|
set {
|
|
userInfo[NotificationConstants.UserInfoKey.roomIdentifier] = newValue
|
|
}
|
|
}
|
|
|
|
override var eventID: String? {
|
|
get {
|
|
userInfo[NotificationConstants.UserInfoKey.eventIdentifier] as? String
|
|
}
|
|
set {
|
|
userInfo[NotificationConstants.UserInfoKey.eventIdentifier] = newValue
|
|
}
|
|
}
|
|
|
|
override var notificationID: String? {
|
|
get {
|
|
userInfo[NotificationConstants.UserInfoKey.notificationIdentifier] as? String
|
|
}
|
|
set {
|
|
userInfo[NotificationConstants.UserInfoKey.notificationIdentifier] = newValue
|
|
}
|
|
}
|
|
|
|
func addMediaAttachment(using mediaProvider: MediaProviderProtocol?,
|
|
mediaSource: MediaSourceProxy) async -> UNMutableNotificationContent {
|
|
guard let mediaProvider else {
|
|
return self
|
|
}
|
|
switch await mediaProvider.loadFileFromSource(mediaSource) {
|
|
case .success(let file):
|
|
do {
|
|
let identifier = ProcessInfo.processInfo.globallyUniqueString
|
|
let newURL = try FileManager.default.copyFileToTemporaryDirectory(file: file.url, with: "\(identifier).\(file.url.pathExtension)")
|
|
let attachment = try UNNotificationAttachment(identifier: identifier,
|
|
url: newURL,
|
|
options: nil)
|
|
attachments.append(attachment)
|
|
} catch {
|
|
MXLog.error("Couldn't add media attachment:: \(error)")
|
|
return self
|
|
}
|
|
case .failure(let error):
|
|
MXLog.error("Couldn't load the file for media attachment: \(error)")
|
|
}
|
|
|
|
return self
|
|
}
|
|
|
|
func addSenderIcon(using mediaProvider: MediaProviderProtocol?,
|
|
senderID: String,
|
|
senderName: String,
|
|
iconType: NotificationIconType) async throws -> UNMutableNotificationContent {
|
|
var image = INImage(named: "")
|
|
if let mediaSource = iconType.mediaSource {
|
|
switch await mediaProvider?.loadImageDataFromSource(mediaSource) {
|
|
case .success(let data):
|
|
image = INImage(imageData: data)
|
|
case .failure(let error):
|
|
MXLog.error("Couldn't add sender icon: \(error)")
|
|
case .none:
|
|
break
|
|
}
|
|
}
|
|
|
|
let senderHandle = INPersonHandle(value: senderID, type: .unknown)
|
|
let sender = INPerson(personHandle: senderHandle,
|
|
nameComponents: nil,
|
|
displayName: senderName,
|
|
image: iconType.isSenderIcon ? image : nil,
|
|
contactIdentifier: nil,
|
|
customIdentifier: nil)
|
|
|
|
// These are required to show the group name as subtitle
|
|
var speakableGroupName: INSpeakableString?
|
|
var recipients: [INPerson]?
|
|
if case let .group(_, groupName) = iconType {
|
|
let meHandle = INPersonHandle(value: receiverID, type: .unknown)
|
|
let me = INPerson(personHandle: meHandle, nameComponents: nil, displayName: nil, image: nil, contactIdentifier: nil, customIdentifier: nil, isMe: true)
|
|
speakableGroupName = INSpeakableString(spokenPhrase: groupName)
|
|
recipients = [sender, me]
|
|
}
|
|
|
|
let intent = INSendMessageIntent(recipients: recipients,
|
|
outgoingMessageType: .outgoingMessageText,
|
|
content: nil,
|
|
speakableGroupName: speakableGroupName,
|
|
conversationIdentifier: roomID,
|
|
serviceName: nil,
|
|
sender: sender,
|
|
attachments: nil)
|
|
if speakableGroupName != nil {
|
|
intent.setImage(image, forParameterNamed: \.speakableGroupName)
|
|
}
|
|
|
|
// Use the intent to initialize the interaction.
|
|
let interaction = INInteraction(intent: intent, response: nil)
|
|
|
|
// Interaction direction is incoming because the user is
|
|
// receiving this message.
|
|
interaction.direction = .incoming
|
|
|
|
// Donate the interaction before updating notification content.
|
|
try await interaction.donate()
|
|
// Update notification content before displaying the
|
|
// communication notification.
|
|
let updatedContent = try updating(from: intent)
|
|
|
|
// swiftlint:disable:next force_cast
|
|
return updatedContent.mutableCopy() as! UNMutableNotificationContent
|
|
}
|
|
}
|