snikket-ios/Snikket/AppDelegate.swift

775 lines
39 KiB
Swift

//
// AppDelegate.swift
//
// Siskin IM
// Copyright (C) 2016 "Tigase, Inc." <office@tigase.com>
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. Look for COPYING file in the top folder.
// If not, see https://www.gnu.org/licenses/.
//
import UIKit
import UserNotifications
import TigaseSwift
import Shared
import WebRTC
import BackgroundTasks
@UIApplicationMain
class AppDelegate: UIResponder, UIApplicationDelegate {
fileprivate let backgroundRefreshTaskIdentifier = "org.tigase.messenger.mobile.refresh";
var window: UIWindow?
var xmppService:XmppService! {
return XmppService.instance;
}
var dbConnection:DBConnection! {
return DBConnection.main;
}
// Temporary Fix for Subscription Requests
static var subscriptionsRequest = [String:String]()
let notificationCenterDelegate = NotificationCenterDelegate();
func application(_ application: UIApplication, willFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {
if #available(iOS 13, *) {
BGTaskScheduler.shared.register(forTaskWithIdentifier: backgroundRefreshTaskIdentifier, using: nil) { (task) in
self.handleAppRefresh(task: task as! BGAppRefreshTask);
}
}
try! DBConnection.migrateToGroupIfNeeded();
ImageCache.convertToAttachments();
// RTCInitFieldTrialDictionary([:]);
RTCInitializeSSL();
//RTCSetupInternalTracer();
Log.initialize();
Settings.initialize();
if #available(iOS 13.0, *) {
switch Settings.appearance.string()! {
case "light":
for window in application.windows {
window.overrideUserInterfaceStyle = .light;
}
case "dark":
for window in application.windows {
window.overrideUserInterfaceStyle = .dark;
}
default:
break;
}
}
_ = JingleManager.instance;
setUIAppearances()
NotificationManager.instance.initialize(provider: MainNotificationManagerProvider());
xmppService.initialize();
Settings.setDefaultSettings()
AccountSettings.initialize();
UNUserNotificationCenter.current().requestAuthorization(options: [.alert, .badge, .sound]) { (granted, error) in
// sending notifications not granted!
}
UNUserNotificationCenter.current().delegate = self.notificationCenterDelegate;
let categories = [
UNNotificationCategory(identifier: "MESSAGE", actions: [], intentIdentifiers: [], hiddenPreviewsBodyPlaceholder: "New message", options: [.customDismissAction])
];
UNUserNotificationCenter.current().setNotificationCategories(Set(categories));
application.registerForRemoteNotifications();
CallManager.initializeCallManager();
NotificationCenter.default.addObserver(self, selector: #selector(AppDelegate.newMessage), name: DBChatHistoryStore.MESSAGE_NEW, object: nil);
NotificationCenter.default.addObserver(self, selector: #selector(AppDelegate.unreadMessagesCountChanged), name: DBChatStore.UNREAD_MESSAGES_COUNT_CHANGED, object: nil);
NotificationCenter.default.addObserver(self, selector: #selector(AppDelegate.serverCertificateError), name: XmppService.SERVER_CERTIFICATE_ERROR, object: nil);
NotificationCenter.default.addObserver(self, selector: #selector(AppDelegate.authenticationFailure), name: XmppService.AUTHENTICATION_FAILURE, object: nil);
NotificationCenter.default.addObserver(self, selector: #selector(AppDelegate.presenceAuthorizationRequest), name: XmppService.PRESENCE_AUTHORIZATION_REQUEST, object: nil);
NotificationCenter.default.addObserver(self, selector: #selector(AppDelegate.mucRoomInvitationReceived), name: XmppService.MUC_ROOM_INVITATION, object: nil);
NotificationCenter.default.addObserver(self, selector: #selector(AppDelegate.pushNotificationRegistrationFailed), name: Notification.Name("pushNotificationsRegistrationFailed"), object: nil);
NotificationCenter.default.addObserver(self, selector: #selector(AppDelegate.accountsChanged), name: AccountManager.ACCOUNT_CHANGED, object: nil);
updateApplicationIconBadgeNumber(completionHandler: nil);
if #available(iOS 13, *) {
} else {
application.setMinimumBackgroundFetchInterval(UIApplication.backgroundFetchIntervalMinimum);
}
(self.window?.rootViewController as? UISplitViewController)?.preferredDisplayMode = .allVisible;
if AccountManager.getAccounts().isEmpty {
self.window?.rootViewController = UIStoryboard(name: "Welcome", bundle: nil).instantiateViewController(withIdentifier: "WelcomeScreenNavigationController");
}
// let callConfig = CXProviderConfiguration(localizedName: "Tigase Messenger");
// self.callProvider = CXProvider(configuration: callConfig);
// DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + 5.0) {
// let uuid = UUID();
// let handle = CXHandle(type: CXHandle.HandleType.generic, value: "andrzej.wojcik@tigase.org");
//
// let startCallAction = CXStartCallAction(call: uuid, handle: handle);
// startCallAction.handle = handle;
//
// let transaction = CXTransaction(action: startCallAction);
// let callController = CXCallController();
// callController.request(transaction, completion: { (error) in
// CXErrorCodeRequestTransactionError.invalidAction
// print("call request:", error?.localizedDescription);
// })
// DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + 30.0, execute: {
// print("finished!", callController);
// })
// }
//
let stackViewAppearance = UIStackView.appearance(whenContainedInInstancesOf: [UINavigationBar.self])
stackViewAppearance.spacing = 5
return true
}
func setUIAppearances() {
if #available(iOS 15, *) {
let tabAppearance = UITabBarAppearance()
tabAppearance.configureWithOpaqueBackground()
UITabBar.appearance().scrollEdgeAppearance = tabAppearance
}
if #available(iOS 13, *) {
let appearance = UINavigationBarAppearance()
appearance.configureWithOpaqueBackground()
appearance.titleTextAttributes = [NSAttributedString.Key.foregroundColor: UIColor(named: "AddToContactsBar")!]
appearance.backgroundColor = UIColor(named: "chatslistBackground")
UINavigationBar.appearance().standardAppearance = appearance;
UINavigationBar.appearance().scrollEdgeAppearance = appearance
UINavigationBar.appearance().compactAppearance = appearance
let tabAppearance = UITabBarAppearance()
tabAppearance.configureWithOpaqueBackground()
UITabBar.appearance().standardAppearance = tabAppearance
} else {
UINavigationBar.appearance().titleTextAttributes = [NSAttributedString.Key.foregroundColor: UIColor.black]
}
}
@objc func accountsChanged(_ notification: Notification) {
DispatchQueue.main.async {
if let rootView = self.window?.rootViewController {
let normalMode: Bool = rootView is UISplitViewController;
let expNormalMode = !AccountManager.getAccounts().isEmpty;
if normalMode != expNormalMode {
if expNormalMode {
(UIApplication.shared.delegate as? AppDelegate)?.hideSetupGuide();
} else {
self.window?.rootViewController = UIStoryboard(name: "Welcome", bundle: nil).instantiateViewController(withIdentifier: "WelcomeScreenNavigationController");
}
}
}
}
}
func applicationWillResignActive(_ application: UIApplication) {
// Sent when the application is about to move from active to inactive state. This can occur for certain types of temporary interruptions (such as an incoming phone call or SMS message) or when the user quits the application and it begins the transition to the background state.
// Use this method to pause ongoing tasks, disable timers, and throttle down OpenGL ES frame rates. Games should use this method to pause the game.
}
private var backgroundTaskId = UIBackgroundTaskIdentifier.invalid;
func applicationDidEnterBackground(_ application: UIApplication) {
// Use this method to release shared resources, save user data, invalidate timers, and store enough application state information to restore your application to its current state in case it is terminated later.
// If your application supports background execution, this method is called instead of applicationWillTerminate: when the user quits.
xmppService.applicationState = .inactive;
initiateBackgroundTask();
}
func initiateBackgroundTask() {
guard xmppService.applicationState == .inactive else {
return;
}
let application = UIApplication.shared;
backgroundTaskId = application.beginBackgroundTask {
print("keep online on away background task expired", self.backgroundTaskId);
self.applicationKeepOnlineOnAwayFinished(application);
}
print("keep online task started", backgroundTaskId, Date());
}
func applicationKeepOnlineOnAwayFinished(_ application: UIApplication) {
let taskId = backgroundTaskId;
guard taskId != .invalid else {
return;
}
backgroundTaskId = .invalid;
print("keep online task expired at", taskId, NSDate());
self.xmppService.backgroundTaskFinished();
print("keep online calling end background task", taskId, NSDate());
if #available(iOS 13, *) {
scheduleAppRefresh();
}
print("keep online task ended", taskId, NSDate());
application.endBackgroundTask(taskId);
}
func applicationWillEnterForeground(_ application: UIApplication) {
CallManager.initializeCallManager();
// Called as part of the transition from the background to the inactive state; here you can undo many of the changes made on entering the background.
UNUserNotificationCenter.current().getDeliveredNotifications { (notifications) in
let toDiscard = notifications.filter({(notification) in
switch NotificationCategory.from(identifier: notification.request.content.categoryIdentifier) {
case .UNSENT_MESSAGES:
return true;
case .MESSAGE:
return notification.request.content.userInfo["sender"] as? String == nil;
default:
return false;
}
}).map({ (notiication) -> String in
return notiication.request.identifier;
});
guard !toDiscard.isEmpty else {
return;
}
UNUserNotificationCenter.current().removeDeliveredNotifications(withIdentifiers: toDiscard)
self.updateApplicationIconBadgeNumber(completionHandler: nil);
}
}
func applicationDidBecomeActive(_ application: UIApplication) {
if #available(iOS 13, *) {
BGTaskScheduler.shared.cancel(taskRequestWithIdentifier: backgroundRefreshTaskIdentifier);
}
// Restart any tasks that were paused (or not yet started) while the application was inactive. If the application was previously in the background, optionally refresh the user interface.
// TODO: XmppService::initialize() call in application:willFinishLaunchingWithOptions results in starting a connections while it may not always be desired if ie. app is relauched in the background due to crash
// Shouldn't it wait for reconnection till it becomes active? or background refresh task is called?
xmppService.applicationState = .active;
applicationKeepOnlineOnAwayFinished(application);
self.updateApplicationIconBadgeNumber(completionHandler: nil);
}
func applicationWillTerminate(_ application: UIApplication) {
// Called when the application is about to terminate. Save data if appropriate. See also applicationDidEnterBackground:.
RTCShutdownInternalTracer();
RTCCleanupSSL();
print(NSDate(), "application terminated!")
}
func application(_ app: UIApplication, open url: URL, options: [UIApplication.OpenURLOptionsKey : Any] = [:]) -> Bool {
guard let components = URLComponents(url: url, resolvingAgainstBaseURL: false) else {
return false;
}
print("got url to open:", components);
guard let xmppUri = XmppUri(url: url) else {
return false;
}
print("got xmpp url with jid:", xmppUri.jid, "action:", xmppUri.action as Any, "params:", xmppUri.dict as Any);
if let action = xmppUri.action {
self.open(xmppUri: xmppUri, action: action);
return true;
} else {
DispatchQueue.main.async {
let alert = UIAlertController(title: NSLocalizedString("Open URL", comment: "Alert title"), message: String.localizedStringWithFormat(NSLocalizedString("What do you want to do with %@?", comment: "Placeholder is xmpp: URI"), url as CVarArg), preferredStyle: .alert);
alert.addAction(UIAlertAction(title: NSLocalizedString("Open chat", comment: "Action: open a chat with a JID"), style: .default, handler: { (action) in
self.open(xmppUri: xmppUri, action: .message);
}))
alert.addAction(UIAlertAction(title: NSLocalizedString("Join group", comment: "Action: Join the supplied JID as a group"), style: .default, handler: { (action) in
self.open(xmppUri: xmppUri, action: .join);
}))
alert.addAction(UIAlertAction(title: NSLocalizedString("Add contact", comment: "Action: add the provided JID as a new contact"), style: .default, handler: { (action) in
self.open(xmppUri: xmppUri, action: .roster);
}))
alert.addAction(UIAlertAction(title: NSLocalizedString("Cancel", comment: ""), style: .cancel, handler: nil));
self.window?.rootViewController?.present(alert, animated: true, completion: nil);
}
return false;
}
}
fileprivate func open(xmppUri: XmppUri, action: XmppUri.Action, then: (()->Void)? = nil) {
switch action {
case .join:
let navController = UIStoryboard(name: "MIX", bundle: nil).instantiateViewController(withIdentifier: "ChannelJoinNavigationViewController") as! UINavigationController;
let joinController = navController.visibleViewController as! ChannelSelectToJoinViewController;
joinController.joinConversation = (xmppUri.jid.bareJid, xmppUri.dict?["password"]);
joinController.hidesBottomBarWhenPushed = true;
navController.modalPresentationStyle = .formSheet;
self.window?.rootViewController?.present(navController, animated: true, completion: nil);
case .message:
let alert = UIAlertController(title: NSLocalizedString("Start Chat", comment: "Alert title - starting a new chat"), message: NSLocalizedString("Select account to open chat from:", comment: "Shown above a choice list of the user's accounts"), preferredStyle: .alert);
let accounts = self.xmppService.getClients().map({ (client) -> BareJID in
return client.sessionObject.userBareJid!;
}).sorted { (a1, a2) -> Bool in
return a1.stringValue.compare(a2.stringValue) == .orderedAscending;
}
let openChatFn: (BareJID)->Void = { (account) in
let xmppClient = self.xmppService.getClient(forJid: account);
let messageModule:MessageModule? = xmppClient?.modulesManager.getModule(MessageModule.ID);
guard messageModule != nil else {
return;
}
_ = messageModule!.chatManager!.getChatOrCreate(with: xmppUri.jid.withoutResource, thread: nil);
guard let destination = self.window?.rootViewController?.storyboard?.instantiateViewController(withIdentifier: "ChatViewNavigationController") as? UINavigationController else {
return;
}
let chatController = destination.children[0] as! ChatViewController;
chatController.hidesBottomBarWhenPushed = true;
chatController.account = account;
chatController.jid = xmppUri.jid.bareJid;
self.window?.rootViewController?.showDetailViewController(destination, sender: self);
}
if accounts.count == 1 {
openChatFn(accounts.first!);
} else {
accounts.forEach({ account in
alert.addAction(UIAlertAction(title: account.stringValue, style: .default, handler: { (action) in
openChatFn(account);
}));
})
self.window?.rootViewController?.present(alert, animated: true, completion: nil);
}
case .roster:
if let dict = xmppUri.dict, let ibr = dict["ibr"], ibr == "y" {
guard !AccountManager.getAccounts().isEmpty else {
self.open(xmppUri: XmppUri(jid: JID(xmppUri.jid.domain), action: .register, dict: dict), action: .register, then: {
DispatchQueue.main.async {
self.open(xmppUri: xmppUri, action: action);
}
});
return;
}
}
guard let navigationController = self.window?.rootViewController?.storyboard?.instantiateViewController(withIdentifier: "RosterItemEditNavigationController") as? UINavigationController else {
return;
}
let itemEditController = navigationController.visibleViewController as? RosterItemEditViewController;
itemEditController?.hidesBottomBarWhenPushed = true;
navigationController.modalPresentationStyle = .formSheet;
self.window?.rootViewController?.present(navigationController, animated: true, completion: {
itemEditController?.account = nil;
itemEditController?.jid = xmppUri.jid;
itemEditController?.jidTextField.text = xmppUri.jid.stringValue;
itemEditController?.nameTextField.text = xmppUri.dict?["name"];
itemEditController?.preauth = xmppUri.dict?["preauth"];
});
case .register:
let alert = UIAlertController(title: NSLocalizedString("Create Account", comment: "Alert title"), message: xmppUri.jid.localPart == nil ? String.localizedStringWithFormat(NSLocalizedString("Do you wish to register a new account at %@?", comment: "Alert text. Placeholder is server domain."), xmppUri.jid.domain!) : String.localizedStringWithFormat(NSLocalizedString("Do you wish to register the account %@?", comment: "Alert text. Placeholder is account JID."), xmppUri.jid.stringValue), preferredStyle: .alert);
alert.addAction(UIAlertAction(title: NSLocalizedString("Yes", comment: ""), style: .default, handler: { action in
let registerAccountController = RegisterAccountController.instantiate(fromAppStoryboard: .Account);
registerAccountController.hidesBottomBarWhenPushed = true;
registerAccountController.account = xmppUri.jid.bareJid;
registerAccountController.preauth = xmppUri.dict?["preauth"];
registerAccountController.onAccountAdded = then;
self.window?.rootViewController?.showDetailViewController(UINavigationController(rootViewController: registerAccountController), sender: self);
}));
alert.addAction(UIAlertAction(title: NSLocalizedString("No", comment: ""), style: .cancel, handler: nil));
self.window?.rootViewController?.present(alert, animated: true, completion: nil);
}
}
@available(iOS 13, *)
func scheduleAppRefresh() {
let request = BGAppRefreshTaskRequest(identifier: backgroundRefreshTaskIdentifier);
request.earliestBeginDate = Date(timeIntervalSinceNow: 3500);
do {
try BGTaskScheduler.shared.submit(request);
} catch {
print("Could not schedule app refresh: \(error)")
}
}
var backgroundFetchInProgress = false;
@available(iOS 13, *)
func handleAppRefresh(task: BGAppRefreshTask) {
guard DispatchQueue.main.sync(execute: {
if self.backgroundFetchInProgress {
return false;
}
backgroundFetchInProgress = true;
return true;
}) else {
task.setTaskCompleted(success: true);
return;
}
self.scheduleAppRefresh();
let fetchStart = Date();
print("starting fetching", fetchStart);
xmppService.preformFetch(completionHandler: {(result) in
let fetchEnd = Date();
let time = fetchEnd.timeIntervalSince(fetchStart);
print(Date(), "fetched date in \(time) seconds with result = \(result)");
self.backgroundFetchInProgress = false;
task.setTaskCompleted(success: result != .failed);
});
task.expirationHandler = {
print("task expiration reached, start", Date());
DispatchQueue.main.sync {
self.backgroundFetchInProgress = false;
}
self.xmppService.performFetchExpired();
print("task expiration reached, end", Date());
}
}
static func isChatVisible(account acc: String?, with j: String?) -> Bool {
guard let account = acc, let jid = j else {
return false;
}
guard let baseChatController = AppDelegate.getChatController(visible: true) else {
return false;
}
print("comparing", baseChatController.account.stringValue, account, baseChatController.jid.stringValue, jid);
return (baseChatController.account == BareJID(account)) && (baseChatController.jid == BareJID(jid));
}
static func getChatController(visible: Bool) -> BaseChatViewController? {
var topController = UIApplication.shared.keyWindow?.rootViewController;
while (topController?.presentedViewController != nil) {
topController = topController?.presentedViewController;
}
guard let splitViewController = topController as? UISplitViewController else {
return nil;
}
guard let navigationController = navigationController(fromSplitViewController: splitViewController) else {
return nil;
}
if visible {
return navigationController.viewControllers.last as? BaseChatViewController;
} else {
for controller in navigationController.viewControllers.reversed() {
if let baseChatViewController = controller as? BaseChatViewController {
return baseChatViewController;
}
}
return nil;
}
// } else {
// return selectedTabController as? BaseChatViewController;
// }
}
private static func navigationController(fromSplitViewController splitViewController: UISplitViewController) -> UINavigationController? {
if splitViewController.isCollapsed {
return splitViewController.viewControllers.first(where: { $0 is UITabBarController }).map({ $0 as! UITabBarController })?.selectedViewController as? UINavigationController;
} else {
return splitViewController.viewControllers.first(where: { !($0 is UITabBarController) }) as? UINavigationController;
}
}
func application(_ application: UIApplication, performFetchWithCompletionHandler completionHandler: @escaping (UIBackgroundFetchResult) -> Void) {
if #available(iOS 13, *) {
completionHandler(.noData);
} else {
let fetchStart = Date();
print(Date(), "OLD: starting fetching data");
xmppService.preformFetch(completionHandler: {(result) in
completionHandler(result);
let fetchEnd = Date();
let time = fetchEnd.timeIntervalSince(fetchStart);
print(Date(), "OLD: fetched date in \(time) seconds with result = \(result)");
});
}
}
func application(_ application: UIApplication, didRegisterForRemoteNotificationsWithDeviceToken deviceToken: Data) {
let tokenString = deviceToken.reduce("", {$0 + String(format: "%02X", $1)});
print("Device Token:", tokenString)
print("Device Token:", deviceToken.map({ String(format: "%02x", $0 )}).joined());
PushEventHandler.instance.deviceId = tokenString;
// Settings.DeviceToken.setValue(tokenString);
}
func application(_ application: UIApplication, didFailToRegisterForRemoteNotificationsWithError error: Error) {
print("Failed to register:", error);
PushEventHandler.instance.deviceId = nil;
// Settings.DeviceToken.setValue(nil);
}
func application(_ application: UIApplication, didReceiveRemoteNotification userInfo: [AnyHashable : Any]) {
print("Push notification received: \(userInfo)");
}
func application(_ application: UIApplication, didReceiveRemoteNotification userInfo: [AnyHashable : Any], fetchCompletionHandler completionHandler: @escaping (UIBackgroundFetchResult) -> Void) {
print("Push notification received with fetch request: \(userInfo)");
//let fetchStart = Date();
if let account = JID(userInfo[AnyHashable("account")] as? String) {
let sender = JID(userInfo[AnyHashable("sender")] as? String);
let body = userInfo[AnyHashable("body")] as? String;
if let unreadMessages = userInfo[AnyHashable("unread-messages")] as? Int, unreadMessages == 0 && sender == nil && body == nil {
let state = self.xmppService.getClient(forJid: account.bareJid)?.state;
print("unread messages retrieved, client state =", state as Any);
if state != .connected {
dismissNewMessageNotifications(for: account) {
completionHandler(.newData);
}
return;
}
} else if body != nil {
NotificationManager.instance.notifyNewMessage(account: account.bareJid, sender: sender?.bareJid, type: .unknown, nickname: userInfo[AnyHashable("nickname")] as? String, body: body!);
} else {
if let encryped = userInfo["encrypted"] as? String, let ivStr = userInfo["iv"] as? String, let key = NotificationEncryptionKeys.key(for: account.bareJid), let data = Data(base64Encoded: encryped), let iv = Data(base64Encoded: ivStr) {
print("got encrypted push with known key");
let cipher = Cipher.AES_GCM();
var decoded = Data();
if cipher.decrypt(iv: iv, key: key, encoded: data, auth: nil, output: &decoded) {
print("got decrypted data:", String(data: decoded, encoding: .utf8) as Any);
if let payload = try? JSONDecoder().decode(Payload.self, from: decoded) {
print("decoded payload successfully!");
// we require `media` to be present (even empty) in incoming push for jingle session initiation,
// so we can assume that if `media` is `nil` then this is a push for call termination
if let sid = payload.sid, payload.media == nil {
guard CallManager.isAvailable, CallManager.instance?.currentCall?.state ?? .ringing == .ringing else {
return;
}
CallManager.instance?.endCall(on: account.bareJid, sid: sid, completionHandler: {
print("ended call");
completionHandler(.newData);
})
return;
}
}
}
}
}
}
completionHandler(.newData);
}
@objc func newMessage(_ notification: NSNotification) {
guard let message = notification.object as? ChatMessage else {
return;
}
guard message.state == .incoming_unread || message.state == .incoming_error_unread || message.encryption == .notForThisDevice else {
return;
}
NotificationManager.instance.notifyNewMessage(account: message.account, sender: message.jid, type: message.authorNickname != nil ? .groupchat : .chat, nickname: message.authorNickname, body: message.message);
}
func dismissNewMessageNotifications(for account: JID, completionHandler: (()-> Void)?) {
UNUserNotificationCenter.current().getDeliveredNotifications { (notifications) in
let toRemove = notifications.filter({ (notification) in
switch NotificationCategory.from(identifier: notification.request.content.categoryIdentifier) {
case .MESSAGE:
return (notification.request.content.userInfo["account"] as? String) == account.stringValue;
default:
return false;
}
}).map({ (notification) in notification.request.identifier });
UNUserNotificationCenter.current().removeDeliveredNotifications(withIdentifiers: toRemove);
self.updateApplicationIconBadgeNumber(completionHandler: completionHandler);
}
}
@objc func unreadMessagesCountChanged(_ notification: NSNotification) {
updateApplicationIconBadgeNumber(completionHandler: nil);
}
@objc func presenceAuthorizationRequest(_ notification: NSNotification) {
let sender = notification.userInfo?["sender"] as? BareJID;
let account = notification.userInfo?["account"] as? BareJID;
var senderName:String? = nil;
if let sessionObject = xmppService.getClient(forJid: account!)?.sessionObject {
senderName = RosterModule.getRosterStore(sessionObject).get(for: JID(sender!))?.name;
}
if senderName == nil {
senderName = sender!.stringValue;
}
let content = UNMutableNotificationContent();
content.body = "Received presence subscription request from " + senderName!;
content.userInfo = ["sender": sender!.stringValue as NSString, "account": account!.stringValue as NSString, "senderName": senderName! as NSString];
content.categoryIdentifier = "SUBSCRIPTION_REQUEST";
content.threadIdentifier = "account=" + account!.stringValue + "|sender=" + sender!.stringValue;
UNUserNotificationCenter.current().add(UNNotificationRequest(identifier: UUID().uuidString, content: content, trigger: nil));
}
@objc func pushNotificationRegistrationFailed(_ notification: NSNotification) {
let account = notification.userInfo?["account"] as? BareJID;
let errorCondition = (notification.userInfo?["errorCondition"] as? ErrorCondition) ?? ErrorCondition.internal_server_error;
let content = UNMutableNotificationContent();
switch errorCondition {
case .remote_server_timeout:
content.body = "It was not possible to contact push notification component.\nTry again later."
case .remote_server_not_found:
content.body = "It was not possible to contact push notification component."
case .service_unavailable:
content.body = "Push notifications not available";
default:
content.body = "It was not possible to contact push notification component: \(errorCondition.rawValue)";
}
content.threadIdentifier = "account=" + account!.stringValue;
content.categoryIdentifier = "ERROR";
content.userInfo = ["account": account!.stringValue];
UNUserNotificationCenter.current().add(UNNotificationRequest(identifier: UUID().uuidString, content: content, trigger: nil));
}
@objc func mucRoomInvitationReceived(_ notification: Notification) {
guard let e = notification.object as? MucModule.InvitationReceivedEvent, let account = e.sessionObject.userBareJid else {
return;
}
let content = UNMutableNotificationContent();
content.body = "Invitation to groupchat \(e.invitation.roomJid.stringValue)";
if let from = e.invitation.inviter, let name = RosterModule.getRosterStore(e.sessionObject).get(for: from) {
content.body = "\(content.body) from \(name)";
}
content.threadIdentifier = "mucRoomInvitation=" + account.stringValue + "|room=" + e.invitation.roomJid.stringValue;
content.categoryIdentifier = "MUC_ROOM_INVITATION";
content.userInfo = ["account": account.stringValue, "roomJid": e.invitation.roomJid.stringValue, "password": e.invitation.password as Any];
UNUserNotificationCenter.current().add(UNNotificationRequest(identifier: UUID().uuidString, content: content, trigger: nil), withCompletionHandler: nil);
}
func updateApplicationIconBadgeNumber(completionHandler: (()->Void)?) {
NotificationManager.instance.provider.countBadge(withThreadId: nil, completionHandler: { count in
DispatchQueue.main.async {
print("setting badge to", count);
UIApplication.shared.applicationIconBadgeNumber = count;
completionHandler?();
}
});
}
@objc func serverCertificateError(_ notification: NSNotification) {
guard let certInfo = notification.userInfo else {
return;
}
let account = BareJID(certInfo["account"] as! String);
let content = UNMutableNotificationContent();
content.body = "Connection to server \(account.domain) failed";
content.userInfo = certInfo;
content.categoryIdentifier = "ERROR";
content.threadIdentifier = "account=" + account.stringValue;
UNUserNotificationCenter.current().add(UNNotificationRequest(identifier: UUID().uuidString, content: content, trigger: nil));
}
@objc func authenticationFailure(_ notification: NSNotification) {
guard let info = notification.userInfo else {
return;
}
let account = BareJID(info["account"] as! String);
let type = info["auth-error-type"] as! String;
let content = UNMutableNotificationContent();
content.body = "Authentication for account \(account) failed: \(type)";
content.userInfo = info;
content.categoryIdentifier = "ERROR";
content.threadIdentifier = "account=" + account.stringValue;
UNUserNotificationCenter.current().add(UNNotificationRequest(identifier: UUID().uuidString, content: content, trigger: nil));
}
func notifyUnsentMessages(count: Int) {
let content = UNMutableNotificationContent();
content.body = "It was not possible to send \(count) messages. Open the app to retry";
content.categoryIdentifier = "UNSENT_MESSAGES";
content.threadIdentifier = "unsent-messages";
content.sound = .default;
UNUserNotificationCenter.current().add(UNNotificationRequest(identifier: UUID().uuidString, content: content, trigger: nil));
}
func hideSetupGuide() {
self.window?.rootViewController = UIStoryboard(name: "Main", bundle: nil).instantiateInitialViewController();
(self.window?.rootViewController as? UISplitViewController)?.preferredDisplayMode = .allVisible;
}
struct XmppUri {
let jid: JID;
let action: Action?;
let dict: [String: String]?;
init?(url: URL?) {
guard url != nil else {
return nil;
}
guard let components = URLComponents(url: url!, resolvingAgainstBaseURL: false) else {
return nil;
}
guard components.host == nil else {
return nil;
}
self.jid = JID(components.path);
if var pairs = components.query?.split(separator: ";").map({ (it: Substring) -> [Substring] in it.split(separator: "=") }) {
if let first = pairs.first, first.count == 1 {
action = Action(rawValue: String(first.first!));
pairs = Array(pairs.dropFirst());
} else {
action = nil;
}
var dict: [String: String] = [:];
for pair in pairs {
dict[String(pair[0])] = pair.count == 1 ? "" : String(pair[1]);
}
self.dict = dict;
} else {
self.action = nil;
self.dict = nil;
}
}
init(jid: JID, action: Action?, dict: [String:String]?) {
self.jid = jid;
self.action = action;
self.dict = dict;
}
func toURL() -> URL? {
var parts = URLComponents();
parts.scheme = "xmpp";
parts.path = jid.stringValue;
if action != nil {
parts.query = action!.rawValue + (dict?.map({ (k,v) -> String in ";\(k)=\(v)"}).joined() ?? "");
} else {
parts.query = dict?.map({ (k,v) -> String in ";\(k)=\(v)"}).joined();
}
return parts.url;
}
enum Action: String {
case message
case join
case roster
case register
}
}
}