Full SDK Integration

This page describes on how to do a straightforward Incode Mobile Onboarding implementation.

Prerequisites

It is first required to Setup Environment install the SDK into the project follow the details outlined in the Setup Environment

The minimum supported Android version is 21, which supports 99.6% of devices.

The minimum supported iOS version is iOS 13, which supports 99.6% of devices.

High level

The below diagram illustrates from a high level what happens during a straightforward integration.
A full example of the straightforwrd integration can be see at this github repo. Alternatively, view the recipe.

Step 1: Initialize the Incode SDK

Fast way SDK Initialize

Before the Incode SDK can be utilized, it must be initialized. Check the following code on how to initialize the Incode SDK on each platform.

public static void IncodeSDKInitialize(Application app) {
  try {

    new IncodeWelcome.Builder(app, Constants.API_URL, Constants.API_KEY)
      .setLoggingEnabled(true)
      .build();

    incodeSDKInitialized = true;
  } catch (Exception exception) {
    incodeSDKInitialized = false;
  }
}

private void setIncodeCommonConfig() {
  CommonConfig commonConfig = new com.incode.welcome_sdk.CommonConfig.Builder()
    .setShowExitConfirmation(true)
    .setShowCloseButton(true)
    .build();

  IncodeWelcome.getInstance().setCommonConfig(commonConfig);
}
func incodeSDKInitialize(api_url: String, api_key: String) {
  IncdOnboardingManager.shared.initIncdOnboarding(url: api_url,
                                                  apiKey: api_key,
                                                  loggingEnabled: true,
                                                  testMode: testMode) { (success, error) in
        print("IncdOnboarding SDK initialization, success: \(success == true), error: \(error?.description ?? "nil")")
        self.dispatchGroup.leave()
    }
}

func setIncodeCommonConfig() {
  IncdOnboardingManager.shared.allowUserToCancel = true
  IncdOnboardingManager.shared.idBackShownAsFrontCheck = true
  IncdOnboardingManager.shared.idFrontShownAsBackCheck = true
}
const IncodeSDKInitialize = (API_URL, API_KEY) => {
    IncodeSdk.initialize({
      testMode: false,
      apiConfig: {
        url: API_URL,
        key: API_KEY,
      },
      waitForTutorials: true,
    })
      .then((_) => {
        startStraightforwardOnboarding();
      })
      .catch((e) => {
        console.error('Incode SDK failed init', e);
      });
  };
void IncodeSDKInitialize(API_URL,API_KEY) {
	IncodeOnboardingSdk.init(
      apiKey: API_KEY,
      apiUrl: API_URL,
      onError: (String error) {
        print('Incode SDK init failed: $error');
      },
      onSuccess: () {
        onStartOnboarding();
      },
    );
}
// NOTE:
//
// To Initialize the SDK in Xamarin, you need to add the next code into the Native project
// that Xamarin creates.

// ANDROID
// **************************************************

using System;

using Android.App;
using Android.Content.PM;
using Android.Runtime;
using Android.OS;
using Incode.Onboarding;

namespace IncodeOnboardingApplication.Droid
{
    [Activity(Label = "IncodeOnboardingApplication", Icon = "@mipmap/icon", Theme = "@style/MainTheme", MainLauncher = true, ConfigurationChanges = ConfigChanges.ScreenSize | ConfigChanges.Orientation | ConfigChanges.UiMode | ConfigChanges.ScreenLayout | ConfigChanges.SmallestScreenSize)]
    public class MainActivity : global::Xamarin.Forms.Platform.Android.FormsAppCompatActivity
    {
        protected override void OnCreate(Bundle savedInstanceState)
        {
            base.OnCreate(savedInstanceState);

            Xamarin.Essentials.Platform.Init(this, savedInstanceState);
            global::Xamarin.Forms.Forms.Init(this, savedInstanceState);

            // TODO - 1 - Set environment configuration
            var API_URL = "";
            var API_KEY = "";

            IncodeOnboarding.Current.Init(
                url: API_URL,
                apiKey: API_KEY,
                isLoggingEnabled: true,
                isTestMode: false
            );

            LoadApplication(new App());
        }
        public override void OnRequestPermissionsResult(int requestCode, string[] permissions, [GeneratedEnum] Android.Content.PM.Permission[] grantResults)
        {
            Xamarin.Essentials.Platform.OnRequestPermissionsResult(requestCode, permissions, grantResults);

            base.OnRequestPermissionsResult(requestCode, permissions, grantResults);
        }
    }
}

// iOS
// **************************************************
using System;
using System.Collections.Generic;
using System.Linq;

using Foundation;
using Incode.Onboarding;
using UIKit;

namespace IncodeOnboardingApplication.iOS
{
    // The UIApplicationDelegate for the application. This class is responsible for launching the 
    // User Interface of the application, as well as listening (and optionally responding) to 
    // application events from iOS.
    [Register("AppDelegate")]
    public partial class AppDelegate : global::Xamarin.Forms.Platform.iOS.FormsApplicationDelegate
    {
        //
        // This method is invoked when the application has loaded and is ready to run. In this 
        // method you should instantiate the window, load the UI into it and then make the window
        // visible.
        //
        // You have 17 seconds to return from this method, or iOS will terminate your application.
        //
        public override bool FinishedLaunching(UIApplication app, NSDictionary options)
        {
            global::Xamarin.Forms.Forms.Init();

            // TODO - 1 - Set environment configuration
            var API_URL = "";
            var API_KEY = "";

            IncodeOnboarding.Current.Init(
                url: API_URL,
                apiKey:API_KEY,
                isLoggingEnabled: true,
                isTestMode: false
            );

            LoadApplication(new App());

            return base.FinishedLaunching(app, options);
        }
    }
}
function incodeSDKInitialize() {
             cordova.exec(function(param) {
             console.log("Success: "+param);
             // start the incode session here
             startSession();
             }, function(err) {
              console.log("Error: "+ err);
               callbackError('Nothing to echo.' +err);
           }, "Cplugin", "initializeSDK", [apiKey,apiUrl,loggingEnabled,testMode]); //apikey,apiurl,loggingenabled, last testmode should be false for production


   }
incodeSDKInitialize(): void {
    cordova.exec((param: any) => {
      console.log("Success: ", param);
      this.startSession();
    }, (err: any) => {
      console.log("Error:::: ", err);
    }, "Cplugin", "initializeSDK", [Constants.API_KEY, Constants.API_URL, "true", "false", "true"]);
  }

SDK Initialize with session created at backend

If you are creating the session in the backend, the API_URL needs to end with /0/ in order to let our bakcned knows how to manage the request using only the token of the session. You need to initialize the SDK in this way:

public static void IncodeSDKInitialize(Application app) {
  try {

    new IncodeWelcome.Builder(app, Constants.API_URL+"/0/")
      .setLoggingEnabled(true)
      .build();

    incodeSDKInitialized = true;
  } catch (Exception exception) {
    incodeSDKInitialized = false;
  }
}

private void setIncodeCommonConfig() {
  CommonConfig commonConfig = new com.incode.welcome_sdk.CommonConfig.Builder()
    .setShowExitConfirmation(true)
    .setShowCloseButton(true)
    .build();

  IncodeWelcome.getInstance().setCommonConfig(commonConfig);
}
func incodeSDKInitialize(api_url: String, api_key: String) {
  IncdOnboardingManager.shared.initIncdOnboarding(url: api_url+"/0/",
                                                  loggingEnabled: true,
                                                  testMode: testMode) { (success, error) in
        print("IncdOnboarding SDK initialization, success: \(success == true), error: \(error?.description ?? "nil")")
        self.dispatchGroup.leave()
    }
}

func setIncodeCommonConfig() {
  IncdOnboardingManager.shared.allowUserToCancel = true
  IncdOnboardingManager.shared.idBackShownAsFrontCheck = true
  IncdOnboardingManager.shared.idFrontShownAsBackCheck = true
}
const IncodeSDKInitialize = (API_URL, API_KEY) => {
    IncodeSdk.initialize({
      testMode: false,
      apiConfig: {
        url: API_URL+"/0/",
      },
      waitForTutorials: true,
    })
      .then((_) => {
        startStraightforwardOnboarding();
      })
      .catch((e) => {
        console.error('Incode SDK failed init', e);
      });
  };
void IncodeSDKInitialize(API_URL,API_KEY) {
	IncodeOnboardingSdk.init(
      apiUrl: API_URL+"/0/",
      onError: (String error) {
        print('Incode SDK init failed: $error');
      },
      onSuccess: () {
        onStartOnboarding();
      },
    );
}
// Not possible to do it in Xamarin for now
function incodeSDKInitialize() {
             cordova.exec(function(param) {
                console.log("SDK initializeSDKWithToken Success: "+param);
                startSession();
             }, function(err) {
                    console.log("Error initializeSDK: "+ err);
           }, "Cplugin", "initializeSDK", ["",apiUrlfortoken+"/0/",loggingEnabled,testMode]); //apikey,apiurl,loggingenabled, testmode should be false for production
   }
incodeSDKInitialize(): void {
    cordova.exec((param: any) => {
      console.log("Success: ", param);
      this.startSession();
    }, (err: any) => {
      console.log("Error:::: ", err);
    }, "Cplugin", "initializeSDK", ["", Constants.API_URL+"/0/", "true", "false", "true"]);
  }

Step 2: Create the session

Next, a session needs to be created in order for the data to be captured. This requires a Configuration ID, which is also known as a flow ID. To configure a flow, check out the getting started guide.

Fast way session creation

private SessionConfig getSimpleSession() {
        SessionConfig sessionConfig = new SessionConfig.Builder()
                .setRegionIsoCode("ALL")
                .setConfigurationId(Constants.CONFIGURATION_ID)
                .build();

        return sessionConfig;
    }
func createSessionConfiguration() -> IncdOnboardingSessionConfiguration {
  return IncdOnboardingSessionConfiguration(regionCode: "ALL",
                                            configurationId: "PASTE__HERE_CONFIGURATION_ID")
}
function getSimpleSession() {
    let sessionConfig: {
      region: 'ALL',
      configurationId: CONFIGURATION_ID,
    };

    return sessionConfig;
  }
OnboardingSessionConfiguration getSimpleSession() {
  // Common configs
  IncodeOnboardingSdk.showCloseButton(allowUserToCancel: true);
  
  // Session config
  OnboardingSessionConfiguration sessionConfig = OnboardingSessionConfiguration(configurationId: CONFIGURATION_ID);
  
	return sessionConfig;
}
var config = new OnboardingConfiguration()
                .AddRegion("ALL")
                .AddConfigurationId(CONFIGURATION_ID)
                .AllowUserToCancel(true);
function startSession(){
              cordova.exec(function(data) {
                       // Start the Incode moudles or sections from here
                       startOnboardingSection();                       
                       }, function(err) {
                        console.log(" startSession Error: "+ err);
                         // handle the error by showing some UI or alert.
                         callbackError('Nothing to echo.' +err);
                     }, "Cplugin", "setupOnboardingSession", [configId,null,null,null]);
      }
                     
startSession(): void {
    const configId = ''; 
    cordova.exec((data: { interviewId: any; token: any; }) => {
      console.log("OnSession created Success: ", data.interviewId , " Token:" , data.token);
      this.startOnboardingSection(); 
    }, (err: string) => {
      console.log(" startSession Error: ", err);
      this.callbackError(err);
    }, "Cplugin", "setupOnboardingSession", [Constants.CONFIGURATION_ID,null,null,null]);
  }

Setup a session created at backend

If you are creating the session in the backend, you need to set the session token in order to let the SDK save all the information captured during the onboarding to that session.

You need to create a way to share that token from your backend to your frontend, usually this is done by an additional API that you need to create. Once the frontend has the token (SESSION_TOKEN_FROM_BACKEND), you just need to pass it to the SDK. You can do that in this way:

private SessionConfig getSimpleSession() {
        SessionConfig sessionConfig = new SessionConfig.Builder()
                .setRegionIsoCode("ALL")
                .setExternalToken(SESSION_TOKEN_FROM_BACKEND)
                .build();

        return sessionConfig;
    }
func createSessionConfiguration() -> IncdOnboardingSessionConfiguration {
  return IncdOnboardingSessionConfiguration(regionCode: "ALL",
                                            token: SESSION_TOKEN_FROM_BACKEND)
}
function getSimpleSession() {
    let sessionConfig: {
      region: 'ALL',
      token: SESSION_TOKEN_FROM_BACKEND,
    };

    return sessionConfig;
  }
OnboardingSessionConfiguration getSimpleSession() {
  // Common configs
  IncodeOnboardingSdk.showCloseButton(allowUserToCancel: true);
  
  // Session config
  OnboardingSessionConfiguration sessionConfig = OnboardingSessionConfiguration(token: SESSION_TOKEN_FROM_BACKEND);
  
	return sessionConfig;
}
// Not possible to do it in Xamarin for now

function startSession(){

        cordova.exec(function(data) {
         // Start the Incode moudles or sections from here
         startOnboardingSection(); //userdefined
         }, function(err) {
          console.log("Error startSession: "+ err);
       }, "Cplugin", "setupOnboardingSession", [null,null,null,externalToken]);
}
                     
startSession(): void {
    const configId = ''; 
    cordova.exec((data: { interviewId: any; token: any; }) => {
      console.log("OnSession created Success: ", data.interviewId , " Token:" , data.token);
      this.startOnboardingSection(); 
    }, (err: string) => {
      console.log(" startSession Error: ", err);
      this.callbackError(err);
    }, "Cplugin", "setupOnboardingSession", [ null,null,null,Constants.TOKEN]);  //TOKEN CREATED IN BACKEND HERE
  }

Step 3: Configure the User Experience

The code snippet below creates an experience which the user scans their ID, the ID is processed, the Selfie is scanned, and finally face match is performed. This is the default onboarding experience and can be customized as needed.

private FlowConfig getStraightforwardFlowConfig() {
    FlowConfig flowConfig = null;
    try {
        flowConfig = new FlowConfig.Builder()
                .addID(new IdScan.Builder()
                        .setIdType(IdScan.IdType.ID)
                        .setShowIdTutorials(true)
                        .setWaitForTutorials(true)
                        .setEnableFrontShownAsBackCheck(true)
                        .setEnableBackShownAsFrontCheck(true)
                        .setIdCategory(IdCategory.FIRST)
                        .build())
                .addProcessId(new ProcessId.Builder()
                        .setIdCategory(IdCategory.FIRST)
                        .build()
                )
                .addSelfieScan(new SelfieScan.Builder()
                        .setShowTutorials(true)
                        .setWaitForTutorials(true)
                        .setMaskCheckEnabled(true)
                        .setLensesCheckEnabled(true)
                        .build()
                )
                .addFaceMatch(new FaceMatch.Builder()
                        .setShowUserExists(false)
                        .build()
                )
                .build();
    } catch (ModuleConfigurationException e) {
        e.printStackTrace();
    } catch (Exception e) {
        e.printStackTrace();
    }

    return flowConfig;
}
func createFlowConfiguration() -> IncdOnboardingFlowConfiguration {
  let flowConfig = IncdOnboardingFlowConfiguration(waitForTutorials: true)
  flowConfig.addIdScan()
  flowConfig.addProcessId()
  flowConfig.addSelfieScan()
  flowConfig.addFaceMatch()
  return flowConfig
}
function getStraightforwardFlowConfig(){
    let flowConfig = [
      { module: 'IdScanFront', showTutorial: true, idType: 'id' },
      { module: 'IdScanBack', showTutorial: true, idType: 'id' },
      { module: 'ProcessId'},
      { module: 'SelfieScan', showTutorial: true, lensesCheck: false },
      { module: 'FaceMatch' },
    ];

    return flowConfig;
}
OnboardingFlowConfiguration getStraightforwardFlowConfig(){
 	OnboardingFlowConfiguration flowConfig = OnboardingFlowConfiguration();
  
  flowConfig.addIdScan(scanStep: ScanStepType.front, idType: IdType.id, showTutorials: true);
  flowConfig.addIdScan(scanStep: ScanStepType.back, idType: IdType.id, showTutorials: true);
  flowConfig.addProcessId();
  flowConfig.addSelfieScan(showTutorials: true);
  flowConfig.addFaceMatch();
  
  return flowConfig;
}
cofig.AddIdScan(new IdScanParams(showTutorials: true), result => ProcessIdScanCallback(result));
config.AddSelfieScan(new SelfieScanParams(showTutorials: true, lensesCheck: true, maskCheckEnabled:true, brightnessThreshold: null), result => ProcessSelfieScanCallback(result));
config.AddFaceMatch(new FaceMatchParams(), result => ProcessFaceMatchCallback(result));

function startOnboardingSection(){

              cordova.exec(function(winParam) {
                                    console.log("startOnboardingSection Success: ");
                                    finishOnboarding();//user defined
                                    }, function(err) {
                                     console.log("Error: "+ err);
                                     // handle the error by showing some UI or alert.
                                      callbackError('Nothing to echo.' +err);
                                  }, "Cplugin", "startOnboardingSection", [ 
                                  {"module":"addId","showTutorials":"true","waitForTutorials":"true",
                                    "idCategory":"FIRST","enableBackShownAsFrontCheck":"true",
                                    "enableFrontShownAsBackCheck":"true","autocaptureUXMode":"HOLDSTILL"},
                                  {"module":"addSelfieScan","showTutorials":"true","waitForTutorials":"true",
                                    "maskCheckEnabled":"true","lensesCheckEnabled":"true"},
                                  {"module":"addFaceMatch"}]);
}
                                  
startOnboardingSection(): void{
    console.log("startOnboardingSection");

    cordova.exec((winParam: any) => {
      console.log("startOnboardingSection Success: ");
      const myObj = JSON.stringify(winParam);
      console.log("myObj Success: ", myObj);
      const myObj1 = JSON.parse(myObj);
      console.log("data Success: ", myObj1.processIdData);

      this.finishOnboarding();
    }, (err: string) => {
      console.log("Error: ", err);
      this.callbackError(err);
    }, "Cplugin", "startOnboardingSection", [
      {"module":"addId","showTutorials":"false","waitForTutorials":"false","idCategory":"FIRST","enableBackShownAsFrontCheck":"true","enableFrontShownAsBackCheck":"true","autocaptureUXMode":"HOLDSTILL"},
      {"module":"addSelfieScan","showTutorials":"false","waitForTutorials":"false","maskCheckEnabled":"true"},
      {"module":"addFaceMatch"}]);
  }

Step 4: Implement the callbacks

During the user's journey to the flow, various callbacks are executed. Custom business logic can be added in these call backs, but showing a custom screen in between steps is not supported.

private IncodeWelcome.OnboardingListener getStraightforwardOnboardingListener() {
        IncodeWelcome.OnboardingListener onboardingListener = new IncodeWelcome.OnboardingListener() {
            @Override
            public void onOnboardingSessionCreated(String token, String interviewId, String region) {
                super.onOnboardingSessionCreated(token, interviewId, region);

                ((MainActivity) activity).interviewId = interviewId;
            }
           
            @Override
            public void onIdFrontCompleted(IdScanResult result) {
                super.onIdFrontCompleted(result);

                Timber.d("onIdFrontCompleted: %s", result);
            }

            @Override
            public void onIdBackCompleted(IdScanResult result) {
                super.onIdBackCompleted(result);

                Timber.d("onIdBackCompleted: %s", result);
            }

            @Override
            public void onIdProcessed(IdProcessResult result) {
                super.onIdProcessed(result);

                Timber.d("onIdProcessed: %s", result);
            }

            @Override
            public void onSelfieScanCompleted(SelfieScanResult result) {
                super.onSelfieScanCompleted(result);

                Timber.d("onSelfieScanCompleted: %s", result);
            }

            @Override
            public void onFaceMatchCompleted(FaceMatchResult result) {
                super.onFaceMatchCompleted(result);

                Timber.d("onFaceMatchCompleted: %s", result);

                executeFaceMatchValidationCheck(result);
            }
            
            @Override
            public void onSuccess() {
                super.onSuccess();

                Timber.d("onSuccess");

                fetchUserScores(interviewId);
            }

            @Override
            public void onError(Throwable error) {
                super.onError(error);

                Timber.d("onError: %s", error);
            }

            @Override
            public void onUserCancelled() {
                super.onUserCancelled();

                Timber.d("onUserCancelled");
            }
        };

        return onboardingListener;
    }
extension MainViewController: IncdOnboardingDelegate {
    
    func onOnboardingSessionCreated(_ result: OnboardingSessionResult) {
        print("onOnboardingSessionCreated, result: \(result)")
        if result.error != nil {
            let interviewId = result.interviewId
        }
    }
    
    func onIdFrontCompleted(_ result: IdScanResult) {
        print("front IdScanResult: \(result)")
    }
    
    func onIdBackCompleted(_ result: IdScanResult) {
        print("back IdScanResult: \(result)")
    }
    
    func onIdProcessed(_ result: IdProcessResult) {
        print("idProcessResult: \(result)")
    }
    
    func onSelfieScanCompleted(_ result: SelfieScanResult) {
        print("onSelfieScanCompleted, result: \(result)")
    }
    
    func onFaceMatchCompleted(_ result: FaceMatchResult) {
        print("onFaceMatchCompleted, result: \(result)")
    }
    
    func onSuccess() {
        print("onSuccess")
        
        fetchUsetScore()
    }
    
    func onError(_ error: IncdFlowError) {
        print("onError, result: \(error)")
    }
    
    func userCancelledSession() {
        print("userCancelledSession")
    }
}
const setupListeners = ({}) => {
  // returns a callback to unregister your listener, e.g. when your screen is getting unmounted
  IncodeSdk.onSessionCreated((session) => {
    let interviewID = session.interviewId
  });

  const complete = IncodeSdk.onStepCompleted;

  return [
    complete({
      module: 'IdScanFront',
      listener: (e)  => {
        console.log('IdScanFront result: ', e.result);
      },
    }),
    complete({
      module: 'IdScanBack',
      listener: (e) => {
        console.log('IdScanBack result: ', e.result);
      },
    }),
    complete({
      module: 'ProcessId',
      listener: (e) => {
        console.log('ProcessId result: ', e.result);
      },
    }),
    complete({
      module: 'SelfieScan',
      listener: (e) => {
        console.log('SelfieScan result:', e.result);
      },
    }),
    complete({
      module: 'FaceMatch',
      listener: (e) => {
        console.log('FaceMatch result: ', e.result);
      },
    }),
  ];
};

useEffect(() => {
    const unsubscribers = setupListeners({});
    // return a function unregistering all your listeners
    return () => {
      unsubscribers.forEach((unsubscriber) => unsubscriber());
    };
  }, []);
void onSuccessCallback(){
}

void onErrorCallback(String error){
}

void onUserCancelledCallback(){
}

void onIdFrontCompletedCallback(IdScanResult result) {
  print("ID Front");
  print(result);
}

void onIdBackCompletedCallback(IdScanResult result) {
  print("ID Back");
  print(result);
}

void onIdProcessedCallback(String result) {
  print("ID Process");
  print(result);
}

void onSelfieScanCompletedCallback(SelfieScanResult result) {
  print("Selfie Scan");
  print(result);
}

void onFaceMatchCompletedCallback(FaceMatchResult result) {
  print("Face Match");
  print(result);
}
private void ProcessIdScanCallback(IdValidationResult result)
{
  if (result != null)
  {
    if (result.Result == ResultCode.Success)
    {
      Debug.WriteLine("######## Id Scan Completed!");
      Debug.WriteLine($"######## Result => {result}");

      // TODO - Manage module success completion
      // TODO - Save/store relevant information for future use in the app
      // TODO - Save current module executed succesfully
      // TODO - Apply specific business logic
      // TODO - Save flag to be used to control de general onboarding flow

      Debug.WriteLine($"######## Id Scan Front Status => {result.FrontIdStatus}.");
      Debug.WriteLine($"######## Id Scan Back Status => {result.BackIdStatus}.");

      if(result.FrontIdStatus == IdValidationResult.ValidationStatus.Ok)
      {
        Debug.WriteLine($"####### Front Id Image in Base64 => {result.FrontId}.");
        Debug.WriteLine($"####### Front Id Cropped Face Image in Base64 => {result.FrontIdCroppedFace}.");
      }

      if (result.BackIdStatus == IdValidationResult.ValidationStatus.Ok)
      {
        Debug.WriteLine($"####### Back Id Image in Base64 => {result.BackId}.");
      }

      if(result.OcrData != null)
      {
        Debug.WriteLine($"######## Id Scan OcrData => {JsonConvert.SerializeObject(result.OcrData)}.");
        Debug.WriteLine($"######## Id Scan ExtendedOcrData {result.ExtendedOcrData}.");
      }

      if (result.FrontIdStatus == IdValidationResult.ValidationStatus.Ok &&
          result.BackIdStatus == IdValidationResult.ValidationStatus.Ok)
      {
        Debug.WriteLine("######## Id Scan Successful!");
      }
      else
      {
        Debug.WriteLine($"######## Id Scan Unsuccessful! due to: FrontIdStatus ({result.FrontIdStatus}) & BackIdStatus({result.BackIdStatus})");
      }
    }
    else if (result.Result == ResultCode.Error)
    {
      Debug.WriteLine($"######## Id Scan failed with Error => {result.ErrorMessage}.");

      // TODO - Manage module error
      // TODO - Save current module executed with failure
      // TODO - Apply specific business logic
      // TODO - Save flag to be used to control de general onboarding flow
    }
  }
}

private void ProcessSelfieScanCallback(SelfieScanResult result)
{
  if (result != null)
  {
    if (result.Result == ResultCode.Success)
    {
      Debug.WriteLine("######## Selfie Scan Completed!");
      Debug.WriteLine($"######## Result => {result}");

      // TODO - Manage module success completion
      // TODO - Save/store relevant information for future use in the app
      // TODO - Save current module executed succesfully
      // TODO - Apply specific business logic
      // TODO - Save flag to be used to control de general onboarding flow

      Debug.WriteLine($"####### Selfie Scan ScanStatus => {result.ScanStatus}.");


      if (result.ScanStatus == SelfieScanResult.Status.Ok)
      {
        Debug.WriteLine("######## Selfie Scan Successful!");
        Debug.WriteLine($"####### Selfie Image in Base64 => {result.SelfieImage}.");
        Debug.WriteLine($"####### Selfie Scan FaceMatched => {(result.FaceMatched.HasValue ? result.FaceMatched.Value ? "Yes" : "No" : "Null")}.");
        Debug.WriteLine($"####### Selfie Scan SpoofAttempt => {(result.SpoofAttempt.HasValue ? result.SpoofAttempt.Value ? "Yes" : "No" : "Null")}.");
      }
      else
      {
        Debug.WriteLine($"######## Selfie Scan Unsuccessful! due to: => {result.ScanStatus}");
      }
    }
    else if (result.Result == ResultCode.Error)
    {
      Debug.WriteLine($"######## Selfie Scan failed with Error => {result.ErrorMessage}.");

      // TODO - Manage module error
      // TODO - Save current module executed with failure
      // TODO - Apply specific business logic
      // TODO - Save flag to be used to control de general onboarding flow
    }
  }
}

private void ProcessFaceMatchCallback(FaceMatchResult result) {
  if (result != null){
    if (result.Result == ResultCode.Success){
      Debug.WriteLine("######## Face Match Completed!");
      Debug.WriteLine($"######## Result => {result}");

      // TODO - Manage module success completion
      // TODO - Save/store relevant information for future use in the app
      // TODO - Save current module executed succesfully
      // TODO - Apply specific business logic
      // TODO - Save flag to be used to control de general onboarding flow

      Debug.WriteLine($"######## Face Mached: {(result.FaceMatched.HasValue ? result.FaceMatched.Value ? "Yes" : "No" : "Null")}");

      if (result.FaceMatched == true)
      {
        if(result.ExistingUser == true)
        {
          Debug.WriteLine($"######## User found in this session: {result.ExistingInterviewId}");

          if (result.NameMatched == true)
          {
            // TODO - EVERYTHING OK - This is just a new user onboarding attempt
          }
          else
          {
            // TODO - WARNING - Manage fraud attempt
          }
        }
        else
        {
          // TODO - EVERYTHING OK - This is just a completly new user
        }
      }
      else
      {
        // TODO - WARNING - Manage fraud attempt
      }
    }
    else if (result.Result == ResultCode.Error)
    {
      Debug.WriteLine($"######## Face Match failed with Error => {result.ErrorMessage}.");

      // TODO - Manage module error
      // TODO - Save current module executed with failure
      // TODO - Apply specific business logic
      // TODO - Save flag to be used to control de general onboarding flow
    }
  }
}
// Not possible to do it in Cordova for now
// Not possible to do it in Cordova for now

Step 5: Execute the Onboarding

Once the SDK is initialized, flow is configured, session is created, and callbacks are implemented, then it is ready for the user to start the flow by calling the startOnboarding function.

private void startStraightforwardOnboarding(Activity activity) {
				setIncodeCommonConfig();
        SessionConfig sessionConfig = getSimpleSession();
        FlowConfig flowConfig = getStraightforwardFlowConfig();
        IncodeWelcome.OnboardingListener onboardingListener = getStraightforwardOnboardingListener();

        IncodeWelcome.getInstance().startOnboarding(activity, sessionConfig, flowConfig, onboardingListener);
    }
func startStraightforwardOnboarding() {
  setIncodeCommonConfig()
  let sessionConfig = createSessionConfiguration()
  let flowConfig = createFlowConfiguration()
  IncdOnboardingManager.shared.presentingViewController = self
  IncdOnboardingManager.shared.startOnboarding(sessionConfig: sessionConfig, flowConfig: flowConfig, delegate: self)
}
function startStraightforwardOnboarding(){
    let sessionConfig = getSimpleSession();
    let flowConfig = getStraightforwardFlowConfig();

    IncodeSdk.startOnboarding({
      flowConfig: flowConfig,
      sessionConfig: sessionConfig,
    })
      .then((result) => {
        fetchUserScores();
      })
      .catch((e) => {
        // TODO - Manage Onboarding error
      });
  }
void startStraightforwardOnboarding(){
  var sessionConfig = getSimpleSession();
  var flowConfig = getStraightforwardFlowConfig();
  
  IncodeOnboardingSdk.startOnboarding(
    sessionConfig: sessionConfig,
    flowConfig: flowConfig,
    onSuccess: onSuccessCallback,
    onError: onErrorCallback,
    onUserCancelled: onUserCancelledCallback,
    onIdFrontCompleted: onIdFrontCompletedCallback,
    onIdBackCompleted: onIdBackCompletedCallback,
    onIdProcessed: onIdProcessedCallback,
    onSelfieScanCompleted: onSelfieScanCompletedCallback,
    onFaceMatchCompleted: onFaceMatchCompletedCallback
  );		
}
IncodeOnboarding.Current.StartOnboarding(config);
document.getElementById("element").addEventListener("click", incodeSDKInitialize);
this.incodeSDKInitialize()

Step 6: Process Results

Last but not least, when the user completes the flow, fetch the results and implement the business logic as needed to JUST redirect the user to the screen you need.

They recommended place to apply the business logic to evaluate the onboarding status and gran access to the product you offer is in the backend, please review this explanation of fetch scores in a deep to achive that.

private void fetchUserScores(String interviewId) {
  IncodeWelcome.getInstance().getUserScore(FAST, interviewId, new GetUserScoreListener() {
    @Override
    public void onUserScoreFetched(UserScoreResult result) {
      Timber.d("onUserScoreFetched: %s", result);

      validateResultWithBusinessLogic(result);
    }

    @Override
    public void onUserCancelled() {
      Timber.d("getUserScore onUserCancelled");
    }

    @Override
    public void onError(Throwable error) {
      Timber.d("getUserScore onError: %s", error);
    }
  });
}

private void validateResultWithBusinessLogic(UserScoreResult result) {
  // TODO - Apply business rules to check the scores of the session
}
func fetchUserScore() {
	IncdOnboardingManager.shared.getUserScore(userScoreFetchMode: UserScoreFetchMode.fast, interviewId: interviewId ,completion: {
  		userScore in
      if(userScore.error != nil) {
      		validateResultWithBusinessLogic()
       }
       else {
         // TODO - Manage fetch user score error
       }
    })
}
    
func validateResultWithBusinessLogic(userScore:UserScore) {
	// TODO - Apply business rules to check the scores of the session
 }
function fetchUserScores(interviewId) {
  IncodeSdk.getUserScore({ mode: 'fast' })
    .then((result) => {
    validateResultWithBusinessLogic(result);
  })
    .catch((e) => {
    // TODO - Manage getting User Score error
  });
}

function validateResultWithBusinessLogic(result){
  // TODO - Apply business rules to check the scores of the session
}
void fetchUserScores() {
  IncodeOnboardingSdk.getUserScore(
    onSuccess: (result) {
  	  validateResultWithBusinessLogic(result);
	  },
    onError: (error) {
			// TODO - Manage getting User Score error
    }
  );
}

void validateResultWithBusinessLogic(result){
}


//To fetch scores or results once onboarding modules are done.

function getUserScore(){
             cordova.exec(function(winParam) {
                 console.log("showUserScore Success: "+winParam);
                 myObj = JSON.stringify(winParam);
                 console.log("Score : "+myObj);
                 deleteLocalUserData();
                }, function(err) {
                  console.log("Error: "+ err);
                callbackError('Nothing to echo.' +err);
           }, "Cplugin", "getUserScore", ["fast"]);
        }



//To finish the session below code is mandatory after all sections or modules are completed successfully. This should be called only once when all the sections are done.

function finishOnboarding(){
       cordova.exec(function(winParam) {
           console.log("finishOnboarding Success: "+winParam);
           getUserScore();
          }, function(err) {
            console.log("Error: "+ err);
          callbackError('Nothing to echo.' +err);
     }, "Cplugin", "finishOnboarding", []);
  }




//Make sure to delete local cache data once all all the above steps are over.

function deleteLocalUserData(){
          cordova.exec(function(winParam) {
               console.log("deleteLocalUserData Success: "+winParam);
             }, function(err) {
               console.log("Error: "+ err);
        }, "Cplugin", "deleteUserLocalData", []);
     }
//To finish the session below code is mandatory after all sections or modules are completed successfully. This should be called only once when all the sections are done.

finishOnboarding(): void {
    console.log("finishOnboarding");
    cordova.exec((winParam: any) => {
      console.log("finishOnboarding Success: ", winParam);
      this.getUserScore();
    }, (err: string) => {
      console.log("Error: ", err);
      this.callbackError(err);
    }, "Cplugin", "finishOnboarding", []);
  }

  //To fetch scores or results once onboarding modules are done.
  getUserScore(): void {
    console.log("getUserScore");
    cordova.exec((winParam: any) => {
      console.log("getUserScore Success: ", winParam);
      this.deleteLocalUserData();
    }, (err: string) => {
      console.log("Error: ", err);
      this.callbackError(err);
    }, "Cplugin", "getUserScore", ["fast"]);
  }

//Make sure to delete local cache data once all all the above steps are over.

  deleteLocalUserData(): void {
    console.log("deleteUserLocalData");
    cordova.exec((winParam: any) => {
      console.log("deleteLocalUserData Success: ", winParam);
      this.navigateTo();
    }, (err: string) => {
      console.log("Error: ", err);
    }, "Cplugin", "deleteUserLocalData", []);
  }

Note: It's very important to understand that you can manage your App UI navigation or API calls in TODO section (validateResultWithBusinessLogic method), but by safety and best practices, the business rules must be applied in your backend, for example user's onboarding approve, change status, etc.

Working example project

Once you setup your environment to integrate Incode and once you reaf the explanation about the general steps to integrate the SDK into the different availble flavors, we have a working examples on each platform that can be use as a playground to test the Incode modules available