session-ios/Session/Meta/AppDelegate.swift

171 lines
8.7 KiB
Swift
Raw Normal View History

import PromiseKit
import WebRTC
2021-10-06 08:00:12 +02:00
import SessionUIKit
import UIKit
2021-11-29 02:10:33 +01:00
import SessionMessagingKit
extension AppDelegate {
// MARK: Call handling
@objc func handleAppActivatedWithOngoingCallIfNeeded() {
guard let call = AppEnvironment.shared.callManager.currentCall else { return }
2022-02-15 03:59:01 +01:00
guard MiniCallView.current == nil else { return }
if let callVC = CurrentAppContext().frontmostViewController() as? CallVC, callVC.call == call { return }
2022-02-15 03:59:01 +01:00
guard let presentingVC = CurrentAppContext().frontmostViewController() else { preconditionFailure() } // FIXME: Handle more gracefully
let callVC = CallVC(for: call)
if let conversationVC = presentingVC as? ConversationVC, let contactThread = conversationVC.thread as? TSContactThread, contactThread.contactSessionID() == call.sessionID {
callVC.conversationVC = conversationVC
conversationVC.inputAccessoryView?.isHidden = true
conversationVC.inputAccessoryView?.alpha = 0
}
presentingVC.present(callVC, animated: true, completion: nil)
2021-11-18 03:19:47 +01:00
}
2021-11-19 00:04:00 +01:00
private func dismissAllCallUI() {
2021-11-18 03:19:47 +01:00
if let currentBanner = IncomingCallBanner.current { currentBanner.dismiss() }
if let callVC = CurrentAppContext().frontmostViewController() as? CallVC { callVC.handleEndCallMessage() }
if let miniCallView = MiniCallView.current { miniCallView.dismiss() }
}
2021-11-19 00:04:00 +01:00
private func showCallUIForCall(_ call: SessionCall) {
DispatchQueue.main.async {
call.reportIncomingCallIfNeeded{ error in
if let error = error {
SNLog("[Calls] Failed to report incoming call to CallKit due to error: \(error)")
} else {
if CurrentAppContext().isMainAppAndActive {
2022-02-15 03:59:01 +01:00
guard let presentingVC = CurrentAppContext().frontmostViewController() else { preconditionFailure() } // FIXME: Handle more gracefully
if let conversationVC = presentingVC as? ConversationVC, let contactThread = conversationVC.thread as? TSContactThread, contactThread.contactSessionID() == call.sessionID {
let callVC = CallVC(for: call)
callVC.conversationVC = conversationVC
conversationVC.inputAccessoryView?.isHidden = true
conversationVC.inputAccessoryView?.alpha = 0
presentingVC.present(callVC, animated: true, completion: nil)
}
}
2021-11-19 00:04:00 +01:00
}
}
}
}
2021-11-29 06:32:02 +01:00
private func insertCallInfoMessage(for message: CallMessage, using transaction: YapDatabaseReadWriteTransaction) -> TSInfoMessage {
let thread = TSContactThread.getOrCreateThread(withContactSessionID: message.sender!, transaction: transaction)
let infoMessage = TSInfoMessage.from(message, associatedWith: thread)
infoMessage.save(with: transaction)
return infoMessage
}
2021-11-30 05:51:15 +01:00
private func showMissedCallTipsIfNeeded(caller: String) {
2021-11-30 03:46:53 +01:00
let userDefaults = UserDefaults.standard
guard !userDefaults[.hasSeenCallMissedTips] else { return }
2021-11-30 05:51:15 +01:00
guard let presentingVC = CurrentAppContext().frontmostViewController() else { preconditionFailure() }
let callMissedTipsModal = CallMissedTipsModal(caller: caller)
presentingVC.present(callMissedTipsModal, animated: true, completion: nil)
2021-11-30 03:46:53 +01:00
userDefaults[.hasSeenCallMissedTips] = true
}
@objc func setUpCallHandling() {
2021-11-03 05:31:50 +01:00
// Pre offer messages
MessageReceiver.handleNewCallOfferMessageIfNeeded = { (message, transaction) in
2021-11-03 05:31:50 +01:00
guard CurrentAppContext().isMainApp else { return }
guard let timestamp = message.sentTimestamp, TimestampUtils.isWithinOneMinute(timestamp: timestamp) else {
// Add missed call message for call offer messages from more than one minute
let infoMessage = self.insertCallInfoMessage(for: message, using: transaction)
infoMessage.updateCallInfoMessage(.missed, using: transaction)
return
}
2021-11-26 06:57:57 +01:00
guard SSKPreferences.areCallsEnabled else {
2021-11-29 06:32:02 +01:00
let infoMessage = self.insertCallInfoMessage(for: message, using: transaction)
infoMessage.updateCallInfoMessage(.missed, using: transaction)
2021-11-30 05:51:15 +01:00
let contactName = Storage.shared.getContact(with: message.sender!, using: transaction)?.displayName(for: Contact.Context.regular) ?? message.sender!
DispatchQueue.main.async {
self.showMissedCallTipsIfNeeded(caller: contactName)
}
2021-11-26 06:57:57 +01:00
return
}
2021-11-11 01:09:52 +01:00
let callManager = AppEnvironment.shared.callManager
2021-11-19 00:04:00 +01:00
// Ignore pre offer message after the same call instance has been generated
if let currentCall = callManager.currentCall, currentCall.uuid == message.uuid! { return }
2021-11-26 06:57:57 +01:00
guard callManager.currentCall == nil else {
callManager.handleIncomingCallOfferInBusyState(offerMessage: message, using: transaction)
2021-11-11 01:09:52 +01:00
return
}
2021-11-29 06:32:02 +01:00
let infoMessage = self.insertCallInfoMessage(for: message, using: transaction)
// Handle UI
if let caller = message.sender, let uuid = message.uuid {
let call = SessionCall(for: caller, uuid: uuid, mode: .answer)
2021-11-29 02:10:33 +01:00
call.callMessageID = infoMessage.uniqueId
2021-11-19 00:04:00 +01:00
self.showCallUIForCall(call)
2021-11-09 06:05:23 +01:00
}
2021-11-03 05:31:50 +01:00
}
2021-08-18 02:33:33 +02:00
// Offer messages
MessageReceiver.handleOfferCallMessage = { message in
DispatchQueue.main.async {
guard let call = AppEnvironment.shared.callManager.currentCall, message.uuid! == call.uuid else { return }
2021-08-17 08:02:20 +02:00
let sdp = RTCSessionDescription(type: .offer, sdp: message.sdps![0])
2021-11-03 05:31:50 +01:00
call.didReceiveRemoteSDP(sdp: sdp)
}
}
2021-08-18 05:07:15 +02:00
// Answer messages
MessageReceiver.handleAnswerCallMessage = { message in
DispatchQueue.main.async {
2021-11-12 03:33:19 +01:00
guard let call = AppEnvironment.shared.callManager.currentCall, message.uuid! == call.uuid else { return }
2021-11-18 03:19:47 +01:00
if message.sender! == getUserHexEncodedPublicKey() {
guard !call.hasStartedConnecting else { return }
2021-11-18 03:19:47 +01:00
self.dismissAllCallUI()
AppEnvironment.shared.callManager.reportCurrentCallEnded(reason: .answeredElsewhere)
} else {
2022-03-24 05:05:00 +01:00
call.invalidateTimeoutTimer()
2021-11-18 03:19:47 +01:00
call.hasStartedConnecting = true
let sdp = RTCSessionDescription(type: .answer, sdp: message.sdps![0])
call.didReceiveRemoteSDP(sdp: sdp)
guard let callVC = CurrentAppContext().frontmostViewController() as? CallVC else { return }
callVC.handleAnswerMessage(message)
}
2021-08-18 05:07:15 +02:00
}
}
2021-08-18 02:33:33 +02:00
// End call messages
MessageReceiver.handleEndCallMessage = { message in
DispatchQueue.main.async {
2021-11-18 03:19:47 +01:00
guard let call = AppEnvironment.shared.callManager.currentCall, message.uuid! == call.uuid else { return }
self.dismissAllCallUI()
if message.sender! == getUserHexEncodedPublicKey() {
AppEnvironment.shared.callManager.reportCurrentCallEnded(reason: .declinedElsewhere)
} else {
AppEnvironment.shared.callManager.reportCurrentCallEnded(reason: .remoteEnded)
}
2021-08-18 02:33:33 +02:00
}
}
}
// MARK: Configuration message
@objc(syncConfigurationIfNeeded)
func syncConfigurationIfNeeded() {
2021-02-26 05:56:41 +01:00
guard Storage.shared.getUser()?.name != nil else { return }
let userDefaults = UserDefaults.standard
let lastSync = userDefaults[.lastConfigurationSync] ?? .distantPast
guard Date().timeIntervalSince(lastSync) > 7 * 24 * 60 * 60 else { return } // Sync every 2 days
MessageSender.syncConfiguration(forceSyncNow: false)
.done {
// Only update the 'lastConfigurationSync' timestamp if we have done the first sync (Don't want
// a new device config sync to override config syncs from other devices)
if userDefaults[.hasSyncedInitialConfiguration] {
userDefaults[.lastConfigurationSync] = Date()
}
}
.retainUntilComplete()
}
// MARK: Closed group poller
@objc func startClosedGroupPoller() {
guard OWSIdentityManager.shared().identityKeyPair() != nil else { return }
ClosedGroupPoller.shared.start()
}
@objc func stopClosedGroupPoller() {
ClosedGroupPoller.shared.stop()
}
}