paint-brush
How To Implement a Video Call App Using iOS CallKit and ZEGOCLOUDby@davidrelo
822 reads
822 reads

How To Implement a Video Call App Using iOS CallKit and ZEGOCLOUD

by David ReloJuly 13th, 2022
Read on Terminal Reader
Read this story w/o Javascript
tldt arrow

Too Long; Didn't Read

CallKit is a new framework introduced by Apple on iOS10 in 2016. It aims to improve the VoIP experience by allowing apps to integrate with the native phone UI. By adopting CallKit, your app will be able to use the native incoming call screen in both the locked and unlocked states. This article will explain how to use [CallKit] and [ZEGOCLOUD] to build a Video Call App with a native calling experience. The following article will also explain how the Incoming Calls function is used to implement.

Companies Mentioned

Mention Thumbnail
Mention Thumbnail
featured image - How To Implement a Video Call App Using iOS CallKit and ZEGOCLOUD
David Relo HackerNoon profile picture


Introduction

CallKit is a new framework introduced by Apple on iOS10 in 2016 to improve the VoIP experience.


This article will explain how to use CallKit and ZEGOCLOUD to build a Video Call App with a native calling experience.


CallKit Demo

What is ZEGOCLOUD?

ZEGOCLOUD is a global cloud communication service provider that was founded in June 2015. It provides high-quality audio/video communication and IM services for more than 4,000 clients around the world.

What is CallKit?

CallKit is a framework that aims to improve the VoIP experience by allowing apps to integrate with the native phone UI. By adopting CallKit, your app will be able to:


  • Use the native incoming call screen in both the locked and unlocked states.
  • Start calls from the native phone app’s Contacts, Favorites, and Recents screens.
  • Interplay with other calls in the system.

1

CXProvider

Your app will use CXProvider to report any out-of-band notifications to the system. These are usually external events, such as an incoming call.


When such an event occurs, CXProvider creates a CXCallUpdate to notify the system.

CXCallUpdate

CXCallUpdate is used to encapsulate new or changing call-related information. The properties included in this class are:

  1. Caller's name
  2. whether a call is video or audio-only.

CXAction

CXAction is an abstract class that represents the phone's action. For different action, CallKit will provide different CXAction implementations.


For example, outgoing calls are represented by CXStartCallAction, and CXAnswerCallAction is used to answer incoming calls. Action is identified by a unique UUID.


When the system wants to notify the app that it has received an event, it will inform it in the form of CXAction.


The app communicates with CXProvider through CXProviderDelegate, this protocol defines the life cycle event methods of CXProvider, and calls CXAction.

CXCallController

Apps use CXCallController to let the system know about user-initiated requests, such as "call" actions. The biggest difference between CXProvider and CXCallController is that the job of CXProvider is to notify the system, while CXCallController makes requests to the user on behalf of the system.


CXCallController uses transactions when making requests. Transactions are represented by CXTransaction, which contains one or more CXAction instances. CXCallCotroller sends the transaction to the system, and if everything is OK, the system will respond with the corresponding CXAction to the CXProvider.

Incoming Calls

The following will explain how to use CallKit to implement the Incoming Call function.

2

  1. When an incoming call comes in, the App will create a CXCallUpdate and send it to the system through CXProvider.
  2. The system will issue an incoming call to its service.
  3. When the user answers the call, the system will send a CXAnswerCallAction to the CXProvider.
  4. App can respond to this animation by implementing the corresponding CXProviderDelegate protocol method.

ProviderDelegate

Create a ProviderDelegate.swift file in your project and add the following code:


import AVFoundation
import CallKit

class ProviderDelegate: NSObject {
  // 1.
  fileprivate let callController = CXCallController()
  fileprivate let provider: CXProvider

  init(callManager: CallManager) {
    // 2.
    provider = CXProvider(configuration: type(of: self).providerConfiguration)

    super.init()
    // 3.
    provider.setDelegate(self, queue: nil)
  }

  // 4.
  static var providerConfiguration: CXProviderConfiguration {
    let providerConfiguration = CXProviderConfiguration(localizedName: "ZEGOCall")

    providerConfiguration.supportsVideo = true
    providerConfiguration.maximumCallsPerCallGroup = 1
    providerConfiguration.supportedHandleTypes = [.generic]
    if let iconMaskImage = UIImage(named: "IconMask") {
        providerConfiguration.iconTemplateImageData =  iconMaskImage.pngData()
   }
   roviderConfiguration.ringtoneSound = "Ringtone.caf"
    return providerConfiguration
  }
}
  1. ProviderDelegate needs to deal with CXProvider and CXCallController, so keep two references to both.
  2. Initialize CXProvider with a CXProviderConfiguration, which will be defined as a static property later. CXProviderConfiguration is used to define the behavior and capabilities of the call.
  3. To be able to respond to events from CXProvider, you need to set its delegate.
  4. In this app, CXProviderConfiguration supports video call, and phone number handling, and limits the number of call groups to 1. For more customization, please refer to the CallKit documentation.

Just below the configuration, add the following helper method:

func reportIncomingCall(uuid: UUID, handle: String, hasVideo: Bool = false, completion: ((NSError?) -> Void)?) {
  // 1.
  let update = CXCallUpdate()
  update.remoteHandle = CXHandle(type: .phoneNumber, value: handle)
  update.hasVideo = hasVideo

  // 2.
  provider.reportNewIncomingCall(with: uuid, update: update) { error in
    if error == nil {
        print("calling")
    }
  }
}

This utility method allows the App to report an incoming call through the CXProvider API. The code is explained as follows:

  1. Prepare to report a call update event to the system, which contains all call-related metadata.
  2. Call the reportIcomingCall(with:update:completion:) method of CXProvider to notify the system of an incoming call.

CXProviderDelegate

The next step is to ensure protocol conformance. Still in ProviderDelegate.swift, declare a new extension to conform to CXProviderDelegate:

extension ProviderDelegate: CXProviderDelegate {
  func providerDidReset(_ provider: CXProvider) {
    print("Provider did reset")
}

CXProviderDelegate specifies only one required method, providerDidReset(_:). The provider invokes this method when reset, giving your app the opportunity to clean up any ongoing calls and revert to a clean state. In this implementation, you’ll terminate the ongoing audio session and dispose of any active calls.

Open AppDelegate.swift and start by adding a new property to the class:


lazy var providerDelegate: ProviderDelegate = ProviderDelegate(callManager: self.callManager)

The provider delegate is ready for use! Add the following method to AppDelegate.swift:


func onReceiveCallInvited(uuid: UUID, handle: String, hasVideo: Bool = false, completion: ((NSError?) -> Void)?) {
  providerDelegate.reportIncomingCall(uuid: uuid, handle: handle, hasVideo: hasVideo, completion: completion)
}

This method lets other classes access the provider delegate’s helper method.


The final piece of the puzzle is hooking up this call to the user interface. Open CallsViewController.swift, which is the controller for the main screen of the app. Find the empty implementation of unwindForNewCall(_:) and replace it with the following code:


@IBAction private func unwindForNewCall(_ segue: UIStoryboardSegue) {
  // 1.
  let newCallController = segue.source as! NewCallViewController
  guard let handle = newCallController.handle else { return }
  let videoEnabled = newCallController.videoEnabled

  // 2.
  let backgroundTaskIdentifier = UIApplication.shared.beginBackgroundTask(expirationHandler: nil)
  DispatchQueue.main.asyncAfter(wallDeadline: DispatchWallTime.now() + 1.5) {
    AppDelegate.shared.displayIncomingCall(uuid: UUID(), handle: handle, hasVideo: videoEnabled) { _ in
      UIApplication.shared.endBackgroundTask(backgroundTaskIdentifier)
    }
  }
}
  1. Extracts the properties of the call from NewCallViewController, which is the source of this unwind segue.
  2. The user can suspend the app before the action completes, so it should use a background task.

Ending the Call

Next, I will explain how to end a call. This App will support two ways to end a call:

  1. The native interface ends the call.
  2. End the call in the app.

3

Notice the difference between steps A and B. When the user ends the call from the in-call screen (A), the system automatically sends a CXEndCallAction to the provider.


However, if you want to end a call using Hotline (B), it’s your job to wrap the action into a transaction and request it from the system. Once the system processes the request, it will send the CXEndCallAction back to the provider.

CXProviderDelegate

However it supports ending calls, your app has to implement the necessary CXProviderDelegate method for it to work. Open ProviderDelegate.swift and add the following implementation to the class extension:


func provider(_ provider: CXProvider, perform action: CXEndCallAction) {
  action.fulfill(withDateEnded: Date())
}

Now add the following methods to the class:


func end(call: Call) {
  // 1.
  let endCallAction = CXEndCallAction(call: call.uuid)
  // 2.
  let transaction = CXTransaction(action: endCallAction)

  requestTransaction(transaction)
}

// 3.
private func requestTransaction(_ transaction: CXTransaction) {
  callController.request(transaction) { error in
    if let error = error {
      print("Error requesting transaction: \(error)")
    } else {
      print("Requested transaction successfully")
    }
  }
}
  1. Create a CXEndCallAction. Pass in the call’s UUID to the initializer so it can be identified later.
  2. Wrap the action into a transaction so you can send it to the system.
  3. Invoke request(_:completion:) from the call controller. The system will request that the provider perform this transaction, which will in turn invoke the delegate method you just implemented.

Did You Know? 👏

Like and Follow is the biggest encouragement to me

Follow me to learn more technical knowledge

Thank you for reading :)

Learn More

This is one of the live technical articles. Welcome to other articles:


Also published here.