-
Notifications
You must be signed in to change notification settings - Fork 0
CallKit 2.x
The Kaleyra Video iOS SDK has built-in support for CallKit framework. The SDK leverages the power of CallKit framework allowing your app to seamlessly integrate with the iOS call system. The SDK will take care of interfacing with CallKit framework without you to know how it works or that it exists at all. Well, sort of... there are a few caveats that must be taken into account.
- Requirements
- Application Setup
- SDK Configuration
- Opt-out CallKit
- Sample apps
- Where to go from here
- What's next
Apple's CallKit framework enables VoIP apps to integrate their calling services with other call-related apps on the system. CallKit provides the calling interface, and your app handles the back-end communication with Kaleyra Video VoIP service through the Kaleyra Video iOS SDK. For incoming and outgoing calls, CallKit displays the same interfaces as the Phone app, giving your app a more native look and feel. CallKit also responds appropriately to system-level behaviors such as Do Not Disturb. Enabling CallKit support in the Kaleyra Video iOS SDK allows your app to take advantage of the following features (only a few are listed below) provided by the CallKit framework:
- Calls won't be interrupted when the user puts the app in background
- Calls won't be interrupted when the user locks her / his screen
- Your app can receive incoming calls when it is in suspended state (requires VoIP notifications)
- Your app can receive incoming calls when it is in background state (requires VoIP notifications)
- Calls won't be interrupted when a regular phone call is received
- Calls won't be interrupted when a VoIP calls is received by another app installed on the user's device
- Seamless integration with the Phone app
- The device ringtone is played when an incoming call is received
- Do not disturb is enforced by the system
Beware, when Kaleyra Video SDK CallKit support is disabled all of the features above won't be available to your app. Thus, if your app is put in the background by the user while a call is in progress, the call will end.
- CallKit framework requires you to specify some background application modes in your app configuration.
- Applications downloaded from the Chinese AppStore MUST disable CallKit framework.
- VoIP push notifications are required in order to fully take advantage of CallKit features.
The following steps will guide you through the configuration of your app to enable CallKit support in the Kaleyra Video SDK.
In order for CallKit to work properly the app background modes must be updated to let the system know that your App supports VOIP communication. This trivial step must be done once and then you can almost forget about it, but it is extremely important otherwise CallKit is not going to work.
The easiest way to enable background modes is adding those capabilities through Xcode. Otherwise you can add those capabilities in your App Info.plist file by yourself.
The required background modes needed to make CallKit work are: audio
and voip
.
You can enable them in Xcode's Capabilities panel of your App. Make sure "Background modes" switch is on and both "Audio, AirPlay and Picture in Picture" and "Voice over IP" checkboxes are enabled.
If you plan to release your app in the Chinese AppStore, please follow along this chapter. If you know you are not going to ship your application on the Chinese AppStore you can safely skip this chapter altogether.
Starting from May 2018, any app using the CallKit framework cannot be sold on the Chinese AppStore anymore. Chinese government has explicitly forbidden Apple from selling apps using CallKit on the Chinese AppStore. If you plan to release your app in the Chinese AppStore you must disable the Kaleyra Video iOS SDK CallKit functionality before submitting your app for review. Even if you disable the CallKit functionality in your app, you must prove to app reviewers your app does not uses CallKit when it is downloaded from the Chinese AppStore (CallKit framework is linked to your app binary anyway because the Kaleyra Video SDK links it strongly).
In order to pass the app review process and comply with the Chinese laws, you have three options:
This is the most preferred way to disable the CallKit framework only for Chinese audience. It can be easily accomplished by checking the device locale when setting up the Kaleyra Video iOS SDK and disabling the SDK CallKit functionality when the locale country region is China. The following snippets of code show you how to do it:
import UIKit
import Bandyer
@UIApplicationMain
class AppDelegate: UIResponder, UIApplicationDelegate {
var window: UIWindow?
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {
let config = BDKConfig()
if let regionCode = Locale.current.regionCode, regionCode.lowercased() == "cn" {
config.isCallKitEnabled = false
} else {
config.isCallKitEnabled = true
}
BandyerSDK.instance().initialize(withApplicationId: "mAppId_31f6da9f24c1c7s01bd2c66a", config: config)
return true
}
}
#import "AppDelegate.h"
#import <Bandyer/Bandyer.h>
@interface AppDelegate ()
@end
@implementation AppDelegate
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
BDKConfig *config = [BDKConfig new];
if ([[NSLocale.currentLocale.countryCode lowercaseString] isEqualToString:@"cn"])
{
config.callKitEnabled = NO;
} else
{
config.callKitEnabled = YES;
}
[BandyerSDK.instance initializeWithApplicationId:@"APP ID HERE" config:config];
return YES;
}
@end
This solution has some drawbacks. First, let's pretend a Chinese user living in UK is using your app, if the user has set her/his device country region setting to "China mainland", CallKit functionality is going to be disabled for the user even if she/he should not be affected by the restriction enforced in China country. Second, you are going to face an App review rejection anyway, unless you inform the App reviewers telling them how your app disables CallKit when it is running on devices in China.
Even if it sounds dumb, one way to overcome this issue is two create two separate apps, one for global audience and one for the Chinese AppStore. The former will take advantage of CallKit features whereas the latter will not. The drawback of this approach is that you are now required to make two separate Apps, and well you know the drill... two AppStore connect pages, two binaries, #ifdef preprocessor flags scattered throughout your code base.
In order to disable CallKit for all users you must initialise the Kaleyra Video iOS SDK with CallKit disabled. This is the most restrictive solution because it disables CallKit for any user using your app, and because your app cannot take advantage of all the great features of CallKit. Take a look at the Opt-out CallKit section of this guide for an example of how to disable the SDK CallKit integration. Beware, you are going to face an App review rejection even if you disabled CallKit in your app unless you explicitly tell the app reviewers how CallKit functionality is disabled in your app when the app is running on devices in China.
In case your app build has been rejected by the AppStore reviewers because of CallKit, don't panic, it is likely you are going to have the possibility to reply to the reviewers explaining how you have disabled CallKit in your app for the Chinese audience. For further information take a look at this link from the Apple forums
Although CallKit support is enabled by default in the Kaleyra Video iOS SDK, you might need to customize the system call UI, to do that you should provide your custom data to the SDK Configuration object. We will show you how to do it in the code section below.
Let's pretend, we are configuring the Kaleyra Video iOS SDK in the application's UIApplicationDelegate
application:didFinishLaunchingWithOptions:
method. The following snippet of code will help you customize the system call UI:
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {
let config = Config()
config.environment = .production
//On iOS 10 and above this statement is not needed, the default configuration object
//enables CallKit by default, it is here for completeness sake
config.isCallKitEnabled = true
//The following statement is going to change the name of the app that is going to be shown by the system call UI.
//If you don't set this value during the configuration, the SDK will look for to the value of the
//CFBundleDisplayName key (or the CFBundleName, if the former is not available) found in your App Info.plist
config.nativeUILocalizedName = "My wonderful app"
//The following statement is going to change the ringtone used by the system call UI when an incoming call
//is received. You should provide the name of the sound resource in the app bundle that is going to be used as
//ringtone. If you don't set this value, the SDK will use the default system ringtone.
config.nativeUIRingToneSound = @"MyRingtoneSound";
//The following statements are going to change the app icon shown in the system call UI. When the user answers
//a call from the lock screen or when the app is not in foreground and a call is in progress, the system
//presents the system call UI to the end user. One of the buttons gives the user the ability to get back into your
//app. The following statements allows you to change that icon.
//Beware, the configuration object property expects the image as an NSData object. You must provide a side
//length 40 points square png image.
//It is highly recommended to set this property, otherwise a "question mark" icon placeholder is used instead.
let callKitIcon = UIImage(named: "callkit-icon")
config.nativeUITemplateIconImageData = callKitIcon?.pngData()
//The following statement is going to tell CallKit the app supports email address as handles for contacts
//For a complete list of handle types see https://developer.apple.com/documentation/callkit/cxhandle/handletype
let handleTypes: Set<NSNumber> = [NSNumber(value: CXHandle.HandleType.emailAddress.rawValue)]
config.supportedHandleTypes = handleTypes
BandyerSDK.instance().initialize(withApplicationId: "YOUR_APP_ID", config: config)
return true
}
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
BDKConfig *config = [BDKConfig new];
config.environment = BDKEnvironment.production;
//On iOS 10 and above this statement is not needed, the default configuration object
//enables CallKit by default, it is here for completeness sake
config.callKitEnabled = YES;
//The following statement is going to change the name of the app that is going to be shown by the system call UI.
//If you don't set this value during the configuration, the SDK will look for to the value of the
//CFBundleDisplayName key (or the CFBundleName, if the former is not available) found in your App Info.plist
config.nativeUILocalizedName = @"My wonderful app";
//The following statement is going to change the ringtone used by the system call UI when an incoming call
//is received. You should provide the name of the sound resource in the app bundle that is going to be used as
//ringtone. If you don't set this value, the SDK will use the default system ringtone.
config.nativeUIRingToneSound = @"MyRingtoneSound";
//The following statements are going to change the app icon shown in the system call UI. When the user answers
//a call from the lock screen or when the app is not in foreground and a call is in progress, the system
//presents the system call UI to the end user. One of the buttons gives the user the ability to get back into your
//app. The following statements allows you to change that icon.
//Beware, the configuration object property expects the image as an NSData object. You must provide a side
//length 40 points square png image.
//It is highly recommended to set this property, otherwise a "question mark" icon placeholder is used instead.
UIImage *callKitIconImage = [UIImage imageNamed:@"callkit_icon"];
config.nativeUITemplateIconImageData = UIImagePNGRepresentation(callKitIconImage);
//The following statement is going to tell CallKit the app supports email address as handles for contacts
//For a complete list of handle types see https://developer.apple.com/documentation/callkit/cxhandletype?language=objc
config.supportedHandleTypes = [NSSet setWithObject:@(CXHandleTypeEmailAddress)];
[BandyerSDK.instance initializeWithApplicationId:APP_ID config:config];
return YES;
}
You must provide a PKPushRegistryDelegate
protocol implementation to the Kaleyra Video config object when CallKit is enabled. You must provide this object regardless of whether you plan to support VoIP notifications or not in your app. If you don't plan to support VoIP notifications in your app, you can provide an object conforming to the PKPushRegistryDelegate
protocol with an empty implementation of the protocol's required method.
For further information take a look at our VoIP notifications guide where we will explain you how VoIP notifications can be integrated in your app.
Beware, if you don't provide a
PKPushRegistryDelegate
protocol implementation the Kaleyra Video iOS SDK will raise an exception.
Here you'll find some snippets of code showing you how the PKPushRegistryDelegate
protocol implementation must be provided to the Kaleyra Video SDK:
import Foundation
import UIKit
import PushKit
import CallKit
import Bandyer
@UIApplicationMain
class AppDelegate: UIResponder, UIApplicationDelegate {
var window: UIWindow?
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {
let config = Config()
config.isCallKitEnabled = true
config.pushRegistryDelegate = self
BandyerSDK.instance().initialize(withApplicationId: "PUT YOUR APP ID HERE", config: config)
return true
}
}
extension AppDelegate: PKPushRegistryDelegate {
func pushRegistry(_ registry: PKPushRegistry, didUpdate pushCredentials: PKPushCredentials, for type: PKPushType) {
guard let token = pushCredentials.tokenAsString else { return }
debugPrint("Push credentials updated \(token), you should send them to your backend system")
}
}
#import "AppDelegate.h"
#import <Bandyer/Bandyer.h>
#import <PushKit/PushKit.h>
#import <CallKit/CallKit.h>
@interface AppDelegate () <PKPushRegistryDelegate>
@end
@implementation AppDelegate
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
BDKConfig *config = [BDKConfig new];
config.callKitEnabled = YES;
config.pushRegistryDelegate = self;
// Here you should put any other configuration options for the Kaleyra Video SDK
[BandyerSDK.instance initializeWithApplicationId:@"PUT YOUR APP ID HERE" config:config];
return YES;
}
- (void)pushRegistry:(PKPushRegistry *)registry didUpdatePushCredentials:(PKPushCredentials *)pushCredentials forType:(PKPushType)type
{
NSLog(@"Updated push credentials %@", pushCredentials.token);
}
In order to show a localized application name in the system UI, you should provide a value to the "nativeUILocalizedName" property of the Kaleyra Video iOS SDK config object. For example if you provide "My Wonderful App" to that property like in the following code:
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {
[...]
config.nativeUILocalizedName = "My Wonderful App"
[...]
}
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
[...]
config.nativeUILocalizedName = @"My Wonderful App";
[...]
}
When an incoming call or an outgoing call is in progress the system UI will be presented like this:
When a call is in progress and the system UI is presented, the user has the ability to return to your application touching an a button on the right egde of the screen identified by the name of your App. By default the system doesn't show an icon for that button, and it looks very weird as you can see in the image below
If you want to show an icon for that button, you must tell the Kaleyra Video iOS SDK which image must be shown, like so:
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {
[...]
let callKitIcon = UIImage(named: "callkit-icon")
config.nativeUITemplateIconImageData = callKitIcon?.pngData()
[...]
}
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
[...]
UIImage *callKitIconImage = [UIImage imageNamed:@"callkit-icon"];
config.nativeUITemplateIconImageData = UIImagePNGRepresentation(callKitIconImage);
[...]
}
Beware you must provide the raw png data of a 40 points square image. The alpha channel of the image will be used to create a white mask for the native system UI. For more info head over to https://developer.apple.com/documentation/callkit/cxproviderconfiguration/2274376-icontemplateimagedata.
This is the final result when an icon is provided:
When an incoming call is received CallKit will play the default system ringtone based on user preferences on the device. If you want to play a different ringtone anytime an incoming call is received you can do so providing the name of the file to reproduce found in your Application Main Bundle.
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {
[...]
config.nativeUIRingToneSound = "MyRingtoneSound"
[...]
}
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
[...]
config.nativeUIRingToneSound = @"MyRingtoneSound";
[...]
}
Kaleyra Video SDKs are designed to work with the bare minimum information about a user, this means that user contact information like first name, last name, email and so on, are not available in the SDK. We refer to a user in the Kaleyra Video platform through hers/his "user alias", which is an alphanumeric string unique within a company (you can think of it as a slug). This approach has the advantage that we don't store any user information on our back-end, nor we send user's information on the network, but it has one drawback, whenever the end user has to be presented with contact information about another user, the only information when can show her/him is an "user alias". This holds also true for CallKit. Whenever the system call UI is presented to the end-user, we must provide it an handle object that has the purpose (in CallKit land) of identifying a contact in the end-user's address book.
The following image will show how the system looks like if you don't provide an handle provider to the SDK
If you look at the name of the caller in the image above, you'll see that instead of the caller name a weird id appears.
CallKit needs CXHandle objects to identify a contact inside the end-user address book. When an incoming or outgoing call is going to be processed by CallKit, the Kaleyra Video SDK must provide those CXHandle objects to it. In order to create CXHandle objects with meaningful user information, the SDK must be provided an object that is able to create CXHandle objects from "user aliases". The UserDetailsProvider protocol serves this purpose. For more information take a look at our guide. This is the final result when you implement a custom UserDetailsProvider:
As you might have noticed, the weird id that was appearing as the caller name is now replaced by a meaningful contact name.
When a call is being performed the native system call UI will show a video button that is enabled only when the call has video support, as you can see in the image below.
If the user taps the button a SiriKit Intent is generated and forwarded to your App delegate. In order to enable the video in the call you must forward the received Siri intent to the Kaleyra Video SDK CallViewController
.
Beware, SiriKit has changed its public api on iOS 13.0. In iOS 10.0 till iOS 13.0 you will receive an NSUserActivity containing an interaction object that carries a INStartVideoCallIntent object whereas starting from iOS 13.0 the interaction object will carry a INStartCallIntent. If your app has a minimum deployment target lower than iOS 13.0 it is likely you are going receive an INStartVideoCallIntent even on iOS 13.0 anyway.
The following code will show you how to handle those intents.
import UIKit
import Bandyer
@UIApplicationMain
class AppDelegate: UIResponder, UIApplicationDelegate {
var window: UIWindow?
var callWindow: CallWindow?
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {
[...]
return true
}
}
extension AppDelegate {
func application(_ application: UIApplication, continue userActivity: NSUserActivity, restorationHandler: @escaping ([UIUserActivityRestoring]?) -> Void) -> Bool {
//When System call ui is shown to the user, it will show a "video" button if the call supports it.
//The code below will handle the siri intent received from the system and it will hand it to the call view controller
//if the controller is presented
if #available(iOS 13.0, *) {
guard let callController = callWindow?.rootViewController as? CallViewController else {
return false
}
if let startCallIntent = userActivity.interaction?.intent as? INStartCallIntent {
callController.handle(startCallIntent: callIntent)
return true
}
if let videoCallIntent = userActivity.interaction?.intent as? INStartVideoCallIntent {
callController.handle(startVideoCallIntent: videoCallIntent)
return true
}
return false
} else {
guard let videoCallIntent = userActivity.interaction?.intent as? INStartVideoCallIntent else {
return false
}
guard let callController = callWindow?.rootViewController as? CallViewController else {
return false
}
callController.handle(startVideoCallIntent: videoCallIntent)
return true
}
}
}
#import <Intents/Intents.h>
#import <Bandyer/Bandyer.h>
#import "AppDelegate.h"
@interface AppDelegate ()
@property (nonatomic, strong) BDKCallWindow *callWindow;
@end
@implementation AppDelegate
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
[...]
return YES;
}
- (BOOL)application:(UIApplication *)application continueUserActivity:(NSUserActivity *)userActivity restorationHandler:(void (^)(NSArray<id <UIUserActivityRestoring>> *__nullable restorableObjects))restorationHandler
{
//When System call ui is shown to the user, it will show a "video" button if the call supports it.
//The code below will handle the siri intent received from the system and it will hand it to the call view controller
//if the controller is presented
if (@available(iOS 13.0, *))
{
if ([userActivity.interaction.intent isKindOfClass:INStartCallIntent.class])
{
[self.callWindow handleINStartCallIntent:(INStartCallIntent *) userActivity.interaction.intent];
return YES;
} else if ([userActivity.interaction.intent isKindOfClass:INStartVideoCallIntent.class])
{
[self.callWindow handleINStartVideoCallIntent:(INStartVideoCallIntent *) userActivity.interaction.intent];
return YES;
}
} else
{
if ([userActivity.interaction.intent isKindOfClass:INStartVideoCallIntent.class])
{
[self.callWindow handleINStartVideoCallIntent:(INStartVideoCallIntent *) userActivity.interaction.intent];
return YES;
}
}
return NO;
}
@end
If you choose to opt-out CallKit altoghether, you can do so telling the Kaleyra Video SDK to not use CallKit during the SDK initialization. The "callKitEnabled" property on Config class serves this purpose.
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
config.callKitEnabled = NO;
}
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {
config.isCallKitEnabled = false
}
We created two sample apps, one in objective-c and one in swift to show you how to integrate the Kaleyra Video SDK in your app with CallKit support enabled.
Now that you have configured your app to handle CallKit you should take a look at our VoIP notifications guide which will show you how to integrate VoIP nofications enabling your app to receive calls even when the app is suspended or in background.
Looking for other platforms? Take a look at Android, Flutter, ReactNative, Ionic / Cordova. Anything unclear or inaccurate? Please let us know by submitting an Issue or write us here.