// Copyright © 2022 Rangeproof Pty Ltd. All rights reserved. import UIKit import CallKit import GRDB import SessionMessagingKit public final class SessionCallManager: NSObject, CallManagerProtocol { let provider: CXProvider? let callController: CXCallController? public var currentCall: CurrentCallProtocol? = nil { willSet { if (newValue != nil) { DispatchQueue.main.async { UIApplication.shared.isIdleTimerDisabled = true } } else { DispatchQueue.main.async { UIApplication.shared.isIdleTimerDisabled = false } } } } private static var _sharedProvider: CXProvider? static func sharedProvider(useSystemCallLog: Bool) -> CXProvider { let configuration = buildProviderConfiguration(useSystemCallLog: useSystemCallLog) if let sharedProvider = self._sharedProvider { sharedProvider.configuration = configuration return sharedProvider } else { SwiftSingletons.register(self) let provider = CXProvider(configuration: configuration) _sharedProvider = provider return provider } } static func buildProviderConfiguration(useSystemCallLog: Bool) -> CXProviderConfiguration { let providerConfiguration = CXProviderConfiguration(localizedName: "Session") providerConfiguration.supportsVideo = true providerConfiguration.maximumCallGroups = 1 providerConfiguration.maximumCallsPerCallGroup = 1 providerConfiguration.supportedHandleTypes = [.generic] let iconMaskImage = #imageLiteral(resourceName: "SessionGreen32") providerConfiguration.iconTemplateImageData = iconMaskImage.pngData() providerConfiguration.includesCallsInRecents = useSystemCallLog return providerConfiguration } // MARK: - Initialization init(useSystemCallLog: Bool = false) { if Preferences.isCallKitSupported { self.provider = SessionCallManager.sharedProvider(useSystemCallLog: useSystemCallLog) self.callController = CXCallController() } else { self.provider = nil self.callController = nil } super.init() // We cannot assert singleton here, because this class gets rebuilt when the user changes relevant call settings self.provider?.setDelegate(self, queue: nil) } // MARK: - Report calls public static func reportFakeCall(info: String) { let callId = UUID() let provider = SessionCallManager.sharedProvider(useSystemCallLog: false) provider.reportNewIncomingCall( with: callId, update: CXCallUpdate() ) { _ in SNLog("[Calls] Reported fake incoming call to CallKit due to: \(info)") } provider.reportCall( with: callId, endedAt: nil, reason: .failed ) } public func reportOutgoingCall(_ call: SessionCall) { AssertIsOnMainThread() UserDefaults.sharedLokiProject?.set(true, forKey: "isCallOngoing") call.stateDidChange = { if call.hasStartedConnecting { self.provider?.reportOutgoingCall(with: call.callId, startedConnectingAt: call.connectingDate) } if call.hasConnected { self.provider?.reportOutgoingCall(with: call.callId, connectedAt: call.connectedDate) } } } public func reportIncomingCall(_ call: SessionCall, callerName: String, completion: @escaping (Error?) -> Void) { AssertIsOnMainThread() let provider = provider ?? Self.sharedProvider(useSystemCallLog: false) // Construct a CXCallUpdate describing the incoming call, including the caller. let update = CXCallUpdate() update.localizedCallerName = callerName update.remoteHandle = CXHandle(type: .generic, value: call.callId.uuidString) update.hasVideo = false disableUnsupportedFeatures(callUpdate: update) // Report the incoming call to the system provider.reportNewIncomingCall(with: call.callId, update: update) { error in guard error == nil else { self.reportCurrentCallEnded(reason: .failed) completion(error) return } UserDefaults.sharedLokiProject?.set(true, forKey: "isCallOngoing") completion(nil) } } public func reportCurrentCallEnded(reason: CXCallEndedReason?) { guard Thread.isMainThread else { DispatchQueue.main.async { self.reportCurrentCallEnded(reason: reason) } return } func handleCallEnded() { WebRTCSession.current = nil UserDefaults.sharedLokiProject?.set(false, forKey: "isCallOngoing") if CurrentAppContext().isInBackground() { // Suspend the database NotificationCenter.default.post(name: Database.suspendNotification, object: self) } } guard let call = currentCall else { handleCallEnded() return } if let reason = reason { self.provider?.reportCall(with: call.callId, endedAt: nil, reason: reason) switch (reason) { case .answeredElsewhere: call.updateCallMessage(mode: .answeredElsewhere) case .unanswered: call.updateCallMessage(mode: .unanswered) case .declinedElsewhere: call.updateCallMessage(mode: .local) default: call.updateCallMessage(mode: .remote) } } else { call.updateCallMessage(mode: .local) } call.webRTCSession.dropConnection() self.currentCall = nil handleCallEnded() } // MARK: - Util private func disableUnsupportedFeatures(callUpdate: CXCallUpdate) { // Call Holding is failing to restart audio when "swapping" calls on the CallKit screen // until user returns to in-app call screen. callUpdate.supportsHolding = false // Not yet supported callUpdate.supportsGrouping = false callUpdate.supportsUngrouping = false // Is there any reason to support this? callUpdate.supportsDTMF = false } // MARK: - UI public func showCallUIForCall(caller: String, uuid: String, mode: CallMode, interactionId: Int64?) { guard Thread.isMainThread else { DispatchQueue.main.async { self.showCallUIForCall(caller: caller, uuid: uuid, mode: mode, interactionId: interactionId) } return } guard let call: SessionCall = Storage.shared.read({ db in SessionCall(db, for: caller, uuid: uuid, mode: mode) }) else { return } call.callInteractionId = interactionId call.reportIncomingCallIfNeeded { error in if let error = error { SNLog("[Calls] Failed to report incoming call to CallKit due to error: \(error)") return } guard CurrentAppContext().isMainAppAndActive else { return } guard let presentingVC = CurrentAppContext().frontmostViewController() else { preconditionFailure() // FIXME: Handle more gracefully } if let conversationVC: ConversationVC = presentingVC as? ConversationVC, conversationVC.viewModel.threadData.threadId == 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) } else if !Preferences.isCallKitSupported { let incomingCallBanner = IncomingCallBanner(for: call) incomingCallBanner.show() } } } public func handleAnswerMessage(_ message: CallMessage) { guard Thread.isMainThread else { DispatchQueue.main.async { self.handleAnswerMessage(message) } return } (CurrentAppContext().frontmostViewController() as? CallVC)?.handleAnswerMessage(message) } public func dismissAllCallUI() { guard Thread.isMainThread else { DispatchQueue.main.async { self.dismissAllCallUI() } return } IncomingCallBanner.current?.dismiss() (CurrentAppContext().frontmostViewController() as? CallVC)?.handleEndCallMessage() MiniCallView.current?.dismiss() } }