Skip to content

Receiving an incoming call 3.x

Marco Brescianini edited this page Nov 5, 2024 · 2 revisions

This guide will show you how to handle incoming calls and how to show the Kaleyra Video call UI in your app for the Kaleyra Video iOS SDK 3.0 version. If you are looking for the 2.x guide please take a look here

Table of contents

Overview

From now on we are assuming you read the previous guides regarding the call client initialization and its lifecycle. We are also assuming the Kaleyra Video SDK has started connecting.

The steps required to handle an incoming call are the following:

  1. Register as an observer on the call client
  2. Implement the IncomingCallObserver protocol callClient:didReceiveIncomingCall: method
  3. Create or retrieve the call window
  4. Provide a HandleIncomingCallIntent to the call window

Incoming call observer

The first thing you must do in order to be notified when an incoming call is received, whether it is received from VoIP notification or from the WebSocket connection, is to register as an IncomingCallObserver on the CallClient object.

Swift example
class MyViewController: UIViewController, IncomingCallObserver {
    
    override func viewDidLoad() {
        super.viewDidLoad()
    
        //When the view loads we register as an incoming call observer, in order to receive notifications about incoming calls received and client state changes.
        BandyerSDK.instance.callClient.addIncomingCall(observer: self, queue: .main)
    }
    
    // MARK: - IncomingCallObserver
    
    func callClient(_ client: CallClient, didReceiveIncomingCall call: Call) {
        //When the call client notify an incoming call is received we handle it showing the user interface for it, more on this later...
        handleIncoming(call)
    }
}
Objective-c example
@interface MyViewController <BDKIncomingCallObserver>
@end

@implementation MyViewController

- (void)viewDidLoad
{
    [super viewDidLoad];

    //When the view loads we register as an incoming call observer, in order to receive notifications about incoming calls received and client state changes.
    [BandyerSDK.instance.callClient addIncomingCallObserver:self queue:dispatch_get_main_queue()];
}

#pragma mark - BDKIncomingCallObserver

- (void)callClient:(id <BDKCallClient>)client didReceiveIncomingCall:(id <BDKCall>)call
{
    //When the call client notify an incoming call is received we handle it showing the user interface for it, more on this later...
    [self handleIncomingCall:call];
}

@end

The HandleIncomingCallIntent

When making an outgoing call you are required to create an intent that will contain all the information the SDK needs to place the outgoing call. The same is true when you need to handle an incoming call. The Kaleyra Video iOS SDK provides a HandleIncomingCallIntent you must instantiate and give to the Call window before the UI can be presented. The difference between a StartOutgoingCallIntent and a IncomingCallHandlingIntent is that the latter requires you to provide it a call object.

The CallWindow

Once you created the intent for the call, it's time to create the window where the call user interface will be presented. Starting from version 1.2.0 we introduced a subclass of UIWindow that will present the call UI in front of any content your app is displaying. This window will take care of creating and present a call view controller for you. In your code you need to do three things: create the call window (or retrieve it if you have created one already), provide the call window a configuration object, tell it to handle the previously created intent. As you might recall from the Making an outgoing call guide, the steps you need to make are the same that are needed to present the call user interface for an outgoing call. The code listing below are the same presented in the Making an outgoing call guide, with some small changes made for handling the incoming call received.

Swift example
class MyViewController: UIViewController, IncomingCallObserver {

    //The call window instance must be created only once and kept alive keeping a reference to it
    lazy var callWindow: CallWindow = {
        var window: CallWindow
        if CallWindow.instance != nil {
            window = CallWindow.instance!
        } else {
            window = CallWindow()
        }

        window.callDelegate = self
        return window
    }()
    
    override func viewDidLoad() {
        super.viewDidLoad()
    
        //When the view loads we register as an incoming call observer, in order to receive notifications about incoming calls received and client state changes.
        BandyerSDK.instance.callClient.addIncomingCall(observer: self, queue: .main)
    }
    
    func callClient(_ client: CallClient, didReceiveIncomingCall call: Call) {
    //When the call client notify an incoming call is received we handle it showing the user interface for it
        handleIncoming(call)
    }
    
    func handleIncoming(_ call: Call) {
        guard let filePath = Bundle.main.path(forResource: "SAMPLE VIDEO URL", ofType: "mp4") else {
            fatalError("The fake file for the file capturer could not be found")
        }

        let config = CallViewControllerConfigurationBuilder()
            .withFakeCapturerFileURL(URL(fileURLWithPath: filePath))
            .withCallInfoTitleFormatter(ContactInfoFormatter())
            .withFeedbackEnabled()
            .build()
        
        callWindow.setConfiguration(config)
        
        //We create the incoming call handling intent
        let intent = HandleIncomingCallIntent(call: call)
        
        //Finally, we present the call view controller on the window
        callWindow.presentCallViewController(for: intent) { error in
            guard let error = error else { return }
        
            debugPrint("An error occurred while presenting the call interface \(error)")    
        } 
    }
}

extension MyViewController: CallWindowDelegate {

    func callWindowDidFinish(_ window: CallWindow) {
        window.isHidden = true
    }

    func callWindow(_ window: CallWindow, openChatWith intent: OpenChatIntent) {
        print("Open chat with intent \(intent)")
    }
}
Objective-c example
@interface MyViewController <BDKIncomingCallObserver>
@end

@implementation MyViewController

//The call window instance must be created only once and kept alive keeping a reference to it
- (BDKCallWindow *)callWindow
{
    if (!_callWindow)
    {
        if (BDKCallWindow.instance)
        {
            _callWindow = BDKCallWindow.instance;
        } else
        {
             //This will automatically save the new instance inside BDKCallWindow.instance.
            _callWindow = [[BDKCallWindow alloc] init];
        }
        _callWindow.callDelegate = self;
    }

    return _callWindow; 
}

- (void)viewDidLoad
{
    [super viewDidLoad];

    //When the view loads we register as an incoming call observer, in order to receive notifications about incoming calls received and client state changes.
    [BandyerSDK.instance.callClient addIncomingCallObserver:self queue:dispatch_get_main_queue()];
}

#pragma mark - BDKIncomingCallObserver 

- (void)callClient:(id <BDKCallClient>)client didReceiveIncomingCall:(id <BDKCall>)call
{
    //When the call client notify an incoming call is received we handle it showing the user interface for it
    [self handleIncomingCall:call];
}

- (void)handleIncomingCall:(id<BDKCall>)call
{
    //First we create the view controller configuration object
    BDKCallViewControllerConfiguration *config = BDKCallViewControllerConfigurationBuilder
        .create()
        .withFakeCapturerFileURL([NSURL fileURLWithPath:[[NSBundle mainBundle] pathForResource:@"SAMPLE VIDEO URL" ofType:@"mp4"]])
        .withFeedbackEnabled()
        .withCallInfoTitleFormatter([ContactInfoFormatter new])
        .build();
    
    //Then we set the configuration object on the call window   
    [self.callWindow setConfiguration:config];
    
    //We create the incoming call handling intent
    BDKHandleIncomingCallIntent* intent = [[BDKHandleIncomingCallIntent alloc] initWithCall:call];

    //Finally, we present the call view controller on the window
    [self.callWindow presentCallViewControllerFor:intent completion:^(NSError * error) {
        if (error != nil) 
        {
            NSLog(@"An error occurred while presenting the call interface %@", error);
        }
    }];
}

#pragma mark - Call window delegate

- (void)callWindowDidFinish:(BDKCallWindow *)window
{
    self.callWindow.hidden = YES;
}

- (void)callWindow:(BDKCallWindow *)window openChatWith:(BDKOpenChatIntent *)intent
{
    NSLog(@"Open chat with intent %@", intent);
}

@end

That's it! You don't have to do anything fancy, just present the call UI and you're good to go.

CallKit

As a side note, if you have enabled CallKit support in the SDK, when an incoming call is received the native system UI will be always displayed even if your application is in foreground and active.

VoIP notifications

If your app supports VoIP notifications, when an incoming call is received from notifications the call client will call the IncomingCallObserver protocol callClient:didReceiveIncomingCall: method as if it where an incoming call received when your app is in foreground.

Where to go from here

If you followed our guides regarding the call client you should have now a complete integration with our call system. If you haven't already, we suggest you to take a look at our sample apps (objective-c swift) to see how to handle incoming calls in a real world app. You should be now ready to move to more advanced topics, like enabling CallKit support in your app, or start integrating the VoIP notifications.

What's next

Clone this wiki locally