mirror of https://github.com/oxen-io/session-ios
refactor to plug in callkit
parent
e2de82a11d
commit
0ef7bdc9ce
@ -0,0 +1,132 @@
|
|||||||
|
import Foundation
|
||||||
|
import WebRTC
|
||||||
|
import SessionMessagingKit
|
||||||
|
|
||||||
|
public final class SessionCall: NSObject {
|
||||||
|
// MARK: Metadata Properties
|
||||||
|
let uuid: UUID
|
||||||
|
let sessionID: String
|
||||||
|
let mode: Mode
|
||||||
|
let webRTCSession: WebRTCSession
|
||||||
|
var contactName: String {
|
||||||
|
let contact = Storage.shared.getContact(with: self.sessionID)
|
||||||
|
return contact?.displayName(for: Contact.Context.regular) ?? self.sessionID
|
||||||
|
}
|
||||||
|
var profilePicture: UIImage {
|
||||||
|
if let result = OWSProfileManager.shared().profileAvatar(forRecipientId: sessionID) {
|
||||||
|
return result
|
||||||
|
} else {
|
||||||
|
return Identicon.generatePlaceholderIcon(seed: sessionID, text: contactName, size: 300)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// MARK: Mode
|
||||||
|
enum Mode {
|
||||||
|
case offer
|
||||||
|
case answer(sdp: RTCSessionDescription)
|
||||||
|
}
|
||||||
|
|
||||||
|
// MARK: Call State Properties
|
||||||
|
var connectingDate: Date? {
|
||||||
|
didSet {
|
||||||
|
stateDidChange?()
|
||||||
|
hasStartedConnectingDidChange?()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var connectedDate: Date? {
|
||||||
|
didSet {
|
||||||
|
stateDidChange?()
|
||||||
|
hasConnectedDidChange?()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var endDate: Date? {
|
||||||
|
didSet {
|
||||||
|
stateDidChange?()
|
||||||
|
hasEndedDidChange?()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Not yet implemented
|
||||||
|
var isOnHold = false {
|
||||||
|
didSet {
|
||||||
|
stateDidChange?()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// MARK: State Change Callbacks
|
||||||
|
var stateDidChange: (() -> Void)?
|
||||||
|
var hasStartedConnectingDidChange: (() -> Void)?
|
||||||
|
var hasConnectedDidChange: (() -> Void)?
|
||||||
|
var hasEndedDidChange: (() -> Void)?
|
||||||
|
|
||||||
|
// MARK: Derived Properties
|
||||||
|
var hasStartedConnecting: Bool {
|
||||||
|
get { return connectingDate != nil }
|
||||||
|
set { connectingDate = newValue ? Date() : nil }
|
||||||
|
}
|
||||||
|
|
||||||
|
var hasConnected: Bool {
|
||||||
|
get { return connectedDate != nil }
|
||||||
|
set { connectedDate = newValue ? Date() : nil }
|
||||||
|
}
|
||||||
|
|
||||||
|
var hasEnded: Bool {
|
||||||
|
get { return endDate != nil }
|
||||||
|
set { endDate = newValue ? Date() : nil }
|
||||||
|
}
|
||||||
|
|
||||||
|
var duration: TimeInterval {
|
||||||
|
guard let connectedDate = connectedDate else {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
return Date().timeIntervalSince(connectedDate)
|
||||||
|
}
|
||||||
|
|
||||||
|
// MARK: Initialization
|
||||||
|
init(for sessionID: String, uuid: String, mode: Mode) {
|
||||||
|
self.sessionID = sessionID
|
||||||
|
self.uuid = UUID(uuidString: uuid)!
|
||||||
|
self.mode = mode
|
||||||
|
self.webRTCSession = WebRTCSession.current ?? WebRTCSession(for: sessionID, with: uuid)
|
||||||
|
super.init()
|
||||||
|
reportIncomingCallIfNeeded()
|
||||||
|
}
|
||||||
|
|
||||||
|
func reportIncomingCallIfNeeded() {
|
||||||
|
guard case .offer = mode else { return }
|
||||||
|
AppEnvironment.shared.callManager.reportIncomingCall(self, callerName: contactName) { error in
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// MARK: Actions
|
||||||
|
func startSessionCall(completion: (() -> Void)?) {
|
||||||
|
guard case .offer = mode else { return }
|
||||||
|
Storage.write { transaction in
|
||||||
|
self.webRTCSession.sendPreOffer(to: self.sessionID, using: transaction).done {
|
||||||
|
self.webRTCSession.sendOffer(to: self.sessionID, using: transaction).done {
|
||||||
|
self.hasStartedConnecting = true
|
||||||
|
}.retainUntilComplete()
|
||||||
|
}.retainUntilComplete()
|
||||||
|
}
|
||||||
|
completion?()
|
||||||
|
}
|
||||||
|
|
||||||
|
func answerSessionCall(completion: (() -> Void)?) {
|
||||||
|
guard case let .answer(sdp) = mode else { return }
|
||||||
|
hasStartedConnecting = true
|
||||||
|
webRTCSession.handleRemoteSDP(sdp, from: sessionID) // This sends an answer message internally
|
||||||
|
completion?()
|
||||||
|
}
|
||||||
|
|
||||||
|
func endSessionCall() {
|
||||||
|
guard !hasEnded else { return }
|
||||||
|
Storage.write { transaction in
|
||||||
|
self.webRTCSession.endCall(with: self.sessionID, using: transaction)
|
||||||
|
}
|
||||||
|
hasEnded = true
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,96 @@
|
|||||||
|
import CallKit
|
||||||
|
import SessionMessagingKit
|
||||||
|
|
||||||
|
public final class SessionCallManager: NSObject, CXProviderDelegate {
|
||||||
|
private let provider: CXProvider
|
||||||
|
var currentCall: SessionCall?
|
||||||
|
|
||||||
|
private static var _sharedProvider: CXProvider?
|
||||||
|
class 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
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class func buildProviderConfiguration(useSystemCallLog: Bool) -> CXProviderConfiguration {
|
||||||
|
let localizedName = NSLocalizedString("APPLICATION_NAME", comment: "Name of application")
|
||||||
|
let providerConfiguration = CXProviderConfiguration(localizedName: localizedName)
|
||||||
|
providerConfiguration.supportsVideo = true
|
||||||
|
providerConfiguration.maximumCallsPerCallGroup = 1
|
||||||
|
providerConfiguration.supportedHandleTypes = [.generic]
|
||||||
|
let iconMaskImage = #imageLiteral(resourceName: "SessionGreen32")
|
||||||
|
providerConfiguration.iconTemplateImageData = iconMaskImage.pngData()
|
||||||
|
providerConfiguration.includesCallsInRecents = useSystemCallLog
|
||||||
|
|
||||||
|
return providerConfiguration
|
||||||
|
}
|
||||||
|
|
||||||
|
init(useSystemCallLog: Bool = false) {
|
||||||
|
AssertIsOnMainThread()
|
||||||
|
self.provider = type(of: self).sharedProvider(useSystemCallLog: useSystemCallLog)
|
||||||
|
|
||||||
|
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)
|
||||||
|
}
|
||||||
|
|
||||||
|
public func providerDidReset(_ provider: CXProvider) {
|
||||||
|
AssertIsOnMainThread()
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public func reportOutgoingCall(_ call: SessionCall, completion: @escaping (Error?) -> Void) {
|
||||||
|
AssertIsOnMainThread()
|
||||||
|
self.provider.reportOutgoingCall(with: call.uuid, startedConnectingAt: call.connectingDate)
|
||||||
|
self.currentCall = call
|
||||||
|
call.hasConnectedDidChange = {
|
||||||
|
self.provider.reportOutgoingCall(with: call.uuid, connectedAt: call.connectedDate)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public func reportIncomingCall(_ call: SessionCall, callerName: String, completion: @escaping (Error?) -> Void) {
|
||||||
|
AssertIsOnMainThread()
|
||||||
|
|
||||||
|
// Construct a CXCallUpdate describing the incoming call, including the caller.
|
||||||
|
let update = CXCallUpdate()
|
||||||
|
update.localizedCallerName = callerName
|
||||||
|
update.remoteHandle = CXHandle(type: .generic, value: call.uuid.uuidString)
|
||||||
|
update.hasVideo = true
|
||||||
|
|
||||||
|
disableUnsupportedFeatures(callUpdate: update)
|
||||||
|
|
||||||
|
// Report the incoming call to the system
|
||||||
|
self.provider.reportNewIncomingCall(with: call.uuid, update: update) { error in
|
||||||
|
guard error == nil else {
|
||||||
|
completion(error)
|
||||||
|
Logger.error("failed to report new incoming call, error: \(error!)")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
self.currentCall = call
|
||||||
|
completion(nil)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 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
|
||||||
|
}
|
||||||
|
}
|
@ -1,11 +0,0 @@
|
|||||||
import CallKit
|
|
||||||
|
|
||||||
extension WebRTCSession: CXProviderDelegate {
|
|
||||||
public func providerDidReset(_ provider: CXProvider) {
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
public func reportIncomingCall() {
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
Loading…
Reference in New Issue