use CallKit for all cases

pull/560/head
ryanzhao 3 years ago
parent 86aced218a
commit f019fe7733

@ -128,6 +128,11 @@ public final class SessionCall: NSObject, WebRTCSessionDelegate {
WebRTCSession.current = self.webRTCSession WebRTCSession.current = self.webRTCSession
super.init() super.init()
self.webRTCSession.delegate = self self.webRTCSession.delegate = self
if AppEnvironment.shared.callManager.currentCall == nil {
AppEnvironment.shared.callManager.currentCall = self
} else {
SNLog("[Calls] A call is ongoing.")
}
} }
func reportIncomingCallIfNeeded(completion: @escaping (Error?) -> Void) { func reportIncomingCallIfNeeded(completion: @escaping (Error?) -> Void) {
@ -147,7 +152,7 @@ public final class SessionCall: NSObject, WebRTCSessionDelegate {
} }
// MARK: Actions // MARK: Actions
func startSessionCall(completion: (() -> Void)?) { func startSessionCall() {
guard case .offer = mode else { return } guard case .offer = mode else { return }
var promise: Promise<Void>! var promise: Promise<Void>!
Storage.write(with: { transaction in Storage.write(with: { transaction in
@ -161,10 +166,9 @@ public final class SessionCall: NSObject, WebRTCSessionDelegate {
} }
} }
}) })
completion?()
} }
func answerSessionCall(completion: (() -> Void)?) { func answerSessionCall() {
guard case .answer = mode else { return } guard case .answer = mode else { return }
hasStartedConnecting = true hasStartedConnecting = true
if let sdp = remoteSDP { if let sdp = remoteSDP {
@ -172,7 +176,6 @@ public final class SessionCall: NSObject, WebRTCSessionDelegate {
} else { } else {
isWaitingForRemoteSDP = true isWaitingForRemoteSDP = true
} }
completion?()
} }
func endSessionCall() { func endSessionCall() {

@ -2,7 +2,7 @@ import CallKit
import SessionUtilitiesKit import SessionUtilitiesKit
extension SessionCallManager { extension SessionCallManager {
public func startCall(_ call: SessionCall, completion: (() -> Void)?) { public func startCall(_ call: SessionCall, completion: ((Error?) -> Void)?) {
guard case .offer = call.mode else { return } guard case .offer = call.mode else { return }
let handle = CXHandle(type: .generic, value: call.sessionID) let handle = CXHandle(type: .generic, value: call.sessionID)
let startCallAction = CXStartCallAction(call: call.uuid, handle: handle) let startCallAction = CXStartCallAction(call: call.uuid, handle: handle)
@ -13,17 +13,23 @@ extension SessionCallManager {
transaction.addAction(startCallAction) transaction.addAction(startCallAction)
reportOutgoingCall(call) reportOutgoingCall(call)
requestTransaction(transaction) requestTransaction(transaction, completion: completion)
completion?() }
public func answerCall(_ call: SessionCall, completion: ((Error?) -> Void)?) {
let answerCallAction = CXAnswerCallAction(call: call.uuid)
let transaction = CXTransaction()
transaction.addAction(answerCallAction)
requestTransaction(transaction, completion: completion)
} }
public func endCall(_ call: SessionCall, completion: (() -> Void)?) { public func endCall(_ call: SessionCall, completion: ((Error?) -> Void)?) {
let endCallAction = CXEndCallAction(call: call.uuid) let endCallAction = CXEndCallAction(call: call.uuid)
let transaction = CXTransaction() let transaction = CXTransaction()
transaction.addAction(endCallAction) transaction.addAction(endCallAction)
requestTransaction(transaction) requestTransaction(transaction, completion: completion)
completion?()
} }
// Not currently in use // Not currently in use
@ -35,13 +41,14 @@ extension SessionCallManager {
requestTransaction(transaction) requestTransaction(transaction)
} }
private func requestTransaction(_ transaction: CXTransaction) { private func requestTransaction(_ transaction: CXTransaction, completion: ((Error?) -> Void)? = nil) {
callController.request(transaction) { error in callController.request(transaction) { error in
if let error = error { if let error = error {
SNLog("Error requesting transaction: \(error)") SNLog("Error requesting transaction: \(error)")
} else { } else {
SNLog("Requested transaction successfully") SNLog("Requested transaction successfully")
} }
completion?(error)
} }
} }
} }

@ -9,18 +9,22 @@ extension SessionCallManager: CXProviderDelegate {
public func provider(_ provider: CXProvider, perform action: CXStartCallAction) { public func provider(_ provider: CXProvider, perform action: CXStartCallAction) {
AssertIsOnMainThread() AssertIsOnMainThread()
guard let call = self.currentCall else { return action.fail() } guard let call = self.currentCall else { return action.fail() }
call.startSessionCall(completion: nil) call.startSessionCall()
action.fulfill() action.fulfill()
} }
public func provider(_ provider: CXProvider, perform action: CXAnswerCallAction) { public func provider(_ provider: CXProvider, perform action: CXAnswerCallAction) {
AssertIsOnMainThread() AssertIsOnMainThread()
guard let _ = self.currentCall else { return action.fail() } guard let call = self.currentCall else { return action.fail() }
let userDefaults = UserDefaults.standard if let _ = CurrentAppContext().frontmostViewController() as? CallVC {
if userDefaults[.hasSeenCallIPExposureWarning] { call.answerSessionCall()
showCallVC()
} else { } else {
showCallModal() let userDefaults = UserDefaults.standard
if userDefaults[.hasSeenCallIPExposureWarning] {
showCallVC()
} else {
showCallModal()
}
} }
action.fulfill() action.fulfill()
} }

@ -4,7 +4,19 @@ import SessionMessagingKit
public final class SessionCallManager: NSObject { public final class SessionCallManager: NSObject {
let provider: CXProvider let provider: CXProvider
let callController = CXCallController() let callController = CXCallController()
var currentCall: SessionCall? var currentCall: SessionCall? = 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? private static var _sharedProvider: CXProvider?
class func sharedProvider(useSystemCallLog: Bool) -> CXProvider { class func sharedProvider(useSystemCallLog: Bool) -> CXProvider {
@ -46,12 +58,13 @@ public final class SessionCallManager: NSObject {
public func reportOutgoingCall(_ call: SessionCall) { public func reportOutgoingCall(_ call: SessionCall) {
AssertIsOnMainThread() AssertIsOnMainThread()
self.currentCall = call call.stateDidChange = {
call.hasStartedConnectingDidChange = { if call.hasStartedConnecting {
self.provider.reportOutgoingCall(with: call.uuid, startedConnectingAt: call.connectingDate) self.provider.reportOutgoingCall(with: call.uuid, startedConnectingAt: call.connectingDate)
} }
call.hasConnectedDidChange = { if call.hasConnected {
self.provider.reportOutgoingCall(with: call.uuid, connectedAt: call.connectedDate) self.provider.reportOutgoingCall(with: call.uuid, connectedAt: call.connectedDate)
}
} }
} }
@ -69,11 +82,11 @@ public final class SessionCallManager: NSObject {
// Report the incoming call to the system // Report the incoming call to the system
self.provider.reportNewIncomingCall(with: call.uuid, update: update) { error in self.provider.reportNewIncomingCall(with: call.uuid, update: update) { error in
guard error == nil else { guard error == nil else {
self.currentCall = nil
completion(error) completion(error)
Logger.error("failed to report new incoming call, error: \(error!)") Logger.error("failed to report new incoming call, error: \(error!)")
return return
} }
self.currentCall = call
completion(nil) completion(nil)
} }
} }

@ -167,6 +167,15 @@ final class CallVC : UIViewController, VideoPreviewDelegate {
} }
} }
} }
self.call.hasStartedConnectingDidChange = {
DispatchQueue.main.async {
self.callInfoLabel.text = "Connecting..."
self.answerButton.alpha = 0
UIView.animate(withDuration: 0.5, delay: 0, usingSpringWithDamping: 1, initialSpringVelocity: 1, options: .curveEaseIn, animations: {
self.answerButton.isHidden = true
}, completion: nil)
}
}
self.call.hasConnectedDidChange = { self.call.hasConnectedDidChange = {
DispatchQueue.main.async { DispatchQueue.main.async {
self.callInfoLabel.text = "Connected" self.callInfoLabel.text = "Connected"
@ -180,8 +189,10 @@ final class CallVC : UIViewController, VideoPreviewDelegate {
} }
} }
self.call.hasEndedDidChange = { self.call.hasEndedDidChange = {
self.conversationVC?.showInputAccessoryView() DispatchQueue.main.async {
self.presentingViewController?.dismiss(animated: true, completion: nil) self.conversationVC?.showInputAccessoryView()
self.presentingViewController?.dismiss(animated: true, completion: nil)
}
} }
} }
@ -194,9 +205,16 @@ final class CallVC : UIViewController, VideoPreviewDelegate {
if shouldRestartCamera { cameraManager.prepare() } if shouldRestartCamera { cameraManager.prepare() }
touch(call.videoCapturer) touch(call.videoCapturer)
titleLabel.text = self.call.contactName titleLabel.text = self.call.contactName
AppEnvironment.shared.callManager.startCall(call) { AppEnvironment.shared.callManager.startCall(call) { error in
self.callInfoLabel.text = "Ringing..." DispatchQueue.main.async {
self.answerButton.isHidden = true if let _ = error {
self.callInfoLabel.text = "Can't start a call."
self.endCall()
} else {
self.callInfoLabel.text = "Ringing..."
self.answerButton.isHidden = true
}
}
} }
if shouldAnswer { answerCall() } if shouldAnswer { answerCall() }
} }
@ -305,12 +323,13 @@ final class CallVC : UIViewController, VideoPreviewDelegate {
@objc private func answerCall() { @objc private func answerCall() {
let userDefaults = UserDefaults.standard let userDefaults = UserDefaults.standard
if userDefaults[.hasSeenCallIPExposureWarning] { if userDefaults[.hasSeenCallIPExposureWarning] {
self.call.answerSessionCall{ AppEnvironment.shared.callManager.answerCall(call) { error in
self.callInfoLabel.text = "Connecting..." DispatchQueue.main.async {
self.answerButton.alpha = 0 if let _ = error {
UIView.animate(withDuration: 0.5, delay: 0, usingSpringWithDamping: 1, initialSpringVelocity: 1, options: .curveEaseIn, animations: { self.callInfoLabel.text = "Can't answer the call."
self.answerButton.isHidden = true self.endCall()
}, completion: nil) }
}
} }
} else { } else {
userDefaults[.hasSeenCallIPExposureWarning] = true userDefaults[.hasSeenCallIPExposureWarning] = true
@ -319,7 +338,12 @@ final class CallVC : UIViewController, VideoPreviewDelegate {
} }
@objc private func endCall() { @objc private func endCall() {
AppEnvironment.shared.callManager.endCall(call, completion: nil) AppEnvironment.shared.callManager.endCall(call) { error in
if let _ = error {
self.call.endSessionCall()
AppEnvironment.shared.callManager.reportCurrentCallEnded(reason: nil)
}
}
} }
@objc private func minimize() { @objc private func minimize() {

@ -155,7 +155,11 @@ final class IncomingCallBanner: UIView, UIGestureRecognizerDelegate {
} }
@objc private func endCall() { @objc private func endCall() {
AppEnvironment.shared.callManager.endCall(call) { AppEnvironment.shared.callManager.endCall(call) { error in
if let _ = error {
self.call.endSessionCall()
AppEnvironment.shared.callManager.reportCurrentCallEnded(reason: nil)
}
self.dismiss() self.dismiss()
} }
} }

@ -31,6 +31,7 @@ extension ConversationVC : InputViewDelegate, MessageCellDelegate, ContextMenuAc
let userDefaults = UserDefaults.standard let userDefaults = UserDefaults.standard
if userDefaults[.hasSeenCallIPExposureWarning] { if userDefaults[.hasSeenCallIPExposureWarning] {
guard let contactSessionID = (thread as? TSContactThread)?.contactSessionID() else { return } guard let contactSessionID = (thread as? TSContactThread)?.contactSessionID() else { return }
guard AppEnvironment.shared.callManager.currentCall == nil else { return }
let call = SessionCall(for: contactSessionID, uuid: UUID().uuidString, mode: .offer) let call = SessionCall(for: contactSessionID, uuid: UUID().uuidString, mode: .offer)
let callVC = CallVC(for: call) let callVC = CallVC(for: call)
callVC.conversationVC = self callVC.conversationVC = self

@ -17,7 +17,6 @@ extension AppDelegate {
conversationVC.inputAccessoryView?.isHidden = true conversationVC.inputAccessoryView?.isHidden = true
conversationVC.inputAccessoryView?.alpha = 0 conversationVC.inputAccessoryView?.alpha = 0
presentingVC.present(callVC, animated: true, completion: nil) presentingVC.present(callVC, animated: true, completion: nil)
return
} }
} }
call.reportIncomingCallIfNeeded{ error in call.reportIncomingCallIfNeeded{ error in

Loading…
Cancel
Save