Integration Guide
Incode Onboarding Flutter SDK
Incode Onboarding provides effortless onboarding where security matters.
Incode Onboarding is part of Incode Omnichannel Biometric Identity Platform, that is powered by Incode's world class Face Recognition, Liveness detection and ID Validation models.
In this repo you can find an example onboarding app that uses Incode Onboarding Flutter SDK to enable remote account opening.
SDK Setup
Requirements
- Flutter version >=1.20.0
Installation
onboarding_flutter_wrapper:
git:
url: [email protected]:Incode-Technologies-Example-Repos/IncdOnboardingFlutter.git
ref: master
If you need a variant that supports streaming the camera feed during Selfie and/or ID scan, please use the following option instead:
onboarding_flutter_wrapper:
git:
url: [email protected]:Incode-Technologies-Example-Repos/IncdOnboardingFlutter.git
ref: master-vc
To use a version older than the latest one, specify the release branch in the ref
field, e.g., release/4.2.0
.
Additional setup for iOS
After installation, it's necessary to do the linking for the iOS, after running the command above.
- Change your
Podfile
insideios
folder so it requires deployment target 13 or higher.
-platform :ios, '11.0'
+platform :ios, '13.0'
- Run
pod install
within theios
folder:
pod install
- Adapt
Info.plist
by adding mandatory permission related entries depending on the modules you need:
- For camera modules like IdScan, SelfieScan, DocumentScan or VideoSelfie the
NSCameraUsageDesscription
is mandatory. Geolocation
module requiresNSLocationWhenInUseUsageDescription
VideoSelfie
module and its voice consent step requiresNSMicrophoneUsageDescription
Additional setup for Android
- Modify
app/build.gradle
so that you enablemultiDexEnabled
and set the minimum API level to 21:
defaultConfig {
…
multiDexEnabled true
minSdkVersion 21
}
- Modify your
build.gradle
so it contains Artifactory username and password, provided by Incode:
allprojects {
repositories {
...
+ maven { url "https://jitpack.io" }
+ maven {
+ url "https://repo.incode.com/artifactory/libs-incode-welcome"
+ credentials {
+ username = "ARTIFACTORY_USERNAME"
+ password = "ARTIFACTORY_PASSWORD"
+ }
+ }
...
}
}
Additionaly, if you're explicitly setting kotlin-gradle-plugin
version make sure kotlin version is set to 1.9.0
:
buildscript {
+ ext.kotlin_version = '1.9.0'
///
dependencies {
...
classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version"
}
}
- Optionally, modify your
app/build.gradle
to add dependencies depending on the features you want to use:
Other optional dependencies can be added for optimized Face Login features:
- Add 'com.incode.sdk:model-mask-detection:2.0.0' to detect face mask during face login.
- Add 'com.incode.sdk:model-liveness-detection:2.0.0' to use local liveness by specifying .hybrid or .local
FaceAuthMode
for face login. - Add 'com.incode.sdk:model-face-recognition:2.0.0' to use local face recognition by specifying .local
FaceAuthMode
for face login.
Updating to latest version
Run flutter pub upgrade
or flutter packages upgrade
, and in case iOS SDK version was updated run pod install --repo-update
and pod update IncdOnboarding
inside your ios folder.
Usage example
- Initialize the SDK
IncodeOnboardingSdk.init(
apiKey: 'YOUR_API_KEY',
apiUrl: 'YOUR_API_URL',
testMode: false,
onError: (String error) {
IncodeSdkInitError? e = error.toIncodeSdkInitError();
switch (e) {
case IncodeSdkInitError.simulatorDetected:
print('Incode init failed, simulator detected: $IncodeSdkInitError.simulatorDetected');
break;
case IncodeSdkInitError.testModeEnabled:
print('Incode init failed, test mode enabled: $IncodeSdkInitError.testModeEnabled');
break;
default:
print('Incode init failed: $error');
break;
}
},
onSuccess: () {
// Update UI, safe to start Onboarding
print('Incode initialize successfully!');
},
);
apiUrl
and apiKey
will be provided to you by Incode.
If you're running the app on a simulator, please set the testMode
parameter to true
.
In case initialization isn't successful, onError
callback will be triggered and the error
String
will contain more information. Possible values are listed in the IncodeSdkInitError
enum: simulatorDetected
, testModeEnabled
, and unknown
.
- Configure Onboarding session
You should create an instance ofOnboardingSessionConfiguration
:
OnboardingSessionConfiguration sessionConfig = OnboardingSessionConfiguration();
Optionally, you can provide these parameters to the OnboardingSessionConfiguration
object constructor:
- region:
ALL
by default - onboardingValidationModules: list of
OnboardingValidationModule
items. This list determines which modules are used for verification and calculation of the onboarding score. If you pass null as validationModuleList, the default values will be used: id, faceRecognition and liveness. - customFields: custom fields which are sent to the server.
- externalId: User identifier outside of Incode Omni database.
- interviewId: Unique identifier of an existion session
- token: Token of an existing session
- configurationId: Flow configurationId found on Incode dashboard.
Specifying interviewId
or token
will return an existing session that will be resumed.
Specifying externalId
will return an existing session in case a session with the same externalId
was already started, otherwise new session will be created.
- Configure Onboarding Flow
You should create an instance of OnboardingFlowConfiguration
:
OnboardingFlowConfiguration flowConfig = OnboardingFlowConfiguration()
Depending on your needs you should specify the steps you want to include, ie.:
flowConfig.addIdScan();
flowConfig.addSelfieScan();
flowConfig.addFaceMatch();
The steps will be executed in the order you added them to the flowConfig.
- Start the onboarding
IncodeOnboardingSdk.startOnboarding(
sessionConfig: sessionConfig,
flowConfig: flowConfig,
onSuccess: () {
print('Incode Onboarding completed!');
},
onError: (String error) {
print('Incode onboarding error: $error');
IncodeSdkFlowError? e = error.toIncodeSdkFlowError();
// Handle the error accordingly
switch (e) {
case IncodeSdkFlowError.rootDetected:
print('Incode SDK onError rooted device detected: $IncodeSdkFlowError.rootDetected');
break;
case IncodeSdkFlowError.hookDetected:
print('Incode SDK onError hooking frameworks detected: $IncodeSdkFlowError.hookDetected');
break;
case IncodeSdkFlowError.simulatorDetected:
print('Incode SDK onError detected simulator: $IncodeSdkFlowError.simulatorDetected');
break;
case IncodeSdkFlowError.permissionsDenied:
print('Incode SDK onError user denied permissions: $IncodeSdkFlowError.permissionsDenied');
break;
case IncodeSdkFlowError.badEnvDetected:
print('Incode SDK onError bad environment detected: $IncodeSdkFlowError.badEnvDetected');
break;
case IncodeSdkFlowError.virtualEnvDetected:
print('Incode SDK onError virtual environment detected: $IncodeSdkFlowError.virtualEnvDetected');
break;
case IncodeSdkFlowError.unknown:
print('Incode SDK onError unknown error: $IncodeSdkFlowError.unknown');
break;
default:
print('Incode SDK onError called: $error');
break;
}
},
onSelfieScanCompleted: (SelfieScanResult result) {
print('Selfie completed result: $result');
},
onIdFrontCompleted: (IdScanResult result) {
print('onIdFrontCompleted result: $result');
},
onIdBackCompleted: (IdScanResult result) {
print('onIdBackCompleted result: $result');
},
onIdProcessed: (String ocrData) {
print('onIdProcessed result: $ocrData');
},
);
Once all the steps are completed by the user, the onSuccess
method will be called.
In case some error occurred that stopped the flow from completing, the onError
method will be called and the error
String
will contain more information. Possible values are listed in IncodeSdkFlowError
enum.
If user cancels the flow, the onUserCancelled
method is triggered.
To listen for the results of the steps in the flow as soon as they're completed, you can add optional callback methods, ie. onSelfieScanCompleted
that was added in the above example.
Optionally, if you want to store ID and Selfie capture session recordings, specify recordSessionConfig
parameter to the startOnboarding
method. Set OnboardingRecordSessionConfiguration.forcePermission
to true if you wish to force the user to accept the recording permissions, otherwise the onboarding session will be aborted.
OnboardingRecordSessionConfiguration recordSessionConfig = OnboardingRecordSessionConfiguration(recordSession: true, forcePermission: false);
IncodeOnboardingSdk.startOnboarding(
sessionConfig: sessionConfig,
flowConfig: flowConfig,
recordSessionConfig: recordSessionConfig,
onSuccess: () {
print('Incode Onboarding completed!');
},
onError: (String error) {
print('Incode onboarding error: $error');
},
onSelfieScanCompleted: (SelfieScanResult result) {
print('Selfie completed result: $result');
},
onIdFrontCompleted: (IdScanResult result) {
print('onIdFrontCompleted result: $result');
},
onIdBackCompleted: (IdScanResult result) {
print('onIdBackCompleted result: $result');
},
onIdProcessed: (String ocrData) {
print('onIdProcessed result: $ocrData');
},
);
Modules
The modules supported by the SDK are listed here, and elaborated in more detail throughout the rest of this document.
Phone
- Ask the user to enter a phone number.Name
- Ask the user to enter a name.Email
- Ask the user to enter an email.IdScan
- Ask the user to capture ID or Passport.ProcessId
- Process the ID in case Id Scan was separated to front and back captures.SelfieScan
- Ask the user to capture a selfie.FaceMatch
- Perform a face match between captured ID and SelfieGeolocation
- Get the information about the users current location.GovernmentValidation
- Perform government validation of the IDVideoSelfie
- Records the device's screen while the user needs to do a selfie, show his ID, answer a couple of questions and verbally confirms that he accept the terms and conditions. The recorded video is then uploaded and stored for later usage.UserScore
- Shows the info and scoring for all the steps performedSignature
- Ask the user to draw a signatureDocumentScan
- Ask the user to capture a documentCaptcha
- Ask the user to complete CAPTCHACURP
- Validate user's curp from the IDOCREdit
- Review OCR dataEKYB
- Enables a KYB validation step with the business information and an optional source for this information including: business name, addresses, city, state, postal code and bank account number.EKYC
- Enables a KYC validation step with the user's information and an optional source for this information including: data obtained from the id, proof of address, or a manual capture.Approve
- Based on a score approves the current onboarding session and adds the user to the omni database.MLConsent
- Ask the user for Machine Learning consentUserConsent
- Ask the user for User consentCombinedConsent
- Ask the user for Data sharing consentQRScan
- Ask the user for QR scanAntifraud
- Gives ability to compare current interview with existing interviews and customers, detecting anomalies that could be signs of fraud.GlobalWatchlist
- This module checks customer identities against sources of Sanctions, Politically Exposed Persons (PEPs), & Watchlists.CustomWatchlist
- This module checks if the user is present inside Incode's watchlist. The watchlist is configured in Incode's dashboard.Aes
- This module enables an advanced electronic signature to ensure legally binding and compliant document signing with enhanced security and authentication measures.
Modules configuration
Below is the list of all available modules, with additional configuration parameters.
- Phone
defaultRegionPrefix
(optional): int?. Default region prefix for phone input. If set, will override the current default prefix based on the user's device region selection.
- Name
- no additional parameters
- Email
- no additional parameters
- IdScan
showTutorials
(optional, defaults totrue
): bool?idType
(optional):IdType.id
orIdType.passport
. If omitted, the app will display a chooser to the user.idCategory
(optional):IdCategory.primary
orIdCategory.secondary
scanStep
(otpional): If you wish to separate front and back ID scan, specifyScanStepType.front
orScanStepType.back
.ScanStepType.both
is the default and will capture both front and back of the ID, and process the ID afterwards. If you specifyScanStepType.front
orScanStepType.back
you will have to addProcessId
module after the captures.showRetakeScreen
(optional, defaults totrue
): bool? - Specifyfalse
to ommit the photo review screen for manual captures.showAutoCaptureRetakeScreen
(optional, defaults tofalse
): bool? - Specifytrue
to show photo review screen for auto captures.enableFrontShownAsBackCheck
(optional, defaults tofalse
): bool? - Specifytrue
to show realtime feedback to the user during back ID capture that he should capture the other side of the IDenableBackShownAsFrontCheck
(optional, defaults tofalse
): bool? - Specifytrue
to show realtime feedback to the user during front ID capture that he should capture the other side of the IDenableRotationOnRetakeScreen
(optional, defaults totrue
): bool? - Specifyfalse
to disable capture image rotation when user enters photo review screen.autocaptureUxMode
(optional, defaults toIdScanAutocaptureUXMode.holdStill
) - SpecifyIdScanAutocaptureUXMode.countdown
to introduce 3-2-1 timer before ID is auto capturedautoCaptureBestFrameTimeout
(optional, defaults to 25) - Timer that starts when ID is detected for the first time, defaults to 25 seconds. Manual capture mode will be activated If ID isn't autocaptured sucessfully before time runs out.autoCaptureNoIdTimeout
(optional, defaults to 60) - Timer that starts when ID scan autocapture starts, defaults to 60 seconds. Manual capture mode will be activated if no ID is detected before time runs out.streamFrames
(optional, defaults to false): boolean. Enables/disables streaming of the camera feed. It requires SDK-vc
variant.
- ProcessId (used only if
ScanStepType.front
orScanStepType.back
were specified asscanStep
forIdScan
module)idCategory
(optional):IdCategory.primary
orIdCategory.secondary
enableIdSummaryScreen
(optional, defaults totrue
): bool? - Specifyfalse
to disable ID capture summary screen- Note: Please make sure to call this module only after the both
ScanStepType.front
andScanStepType.back
IdScans are completed.
- SelfieScan
showTutorials
(optional, defaults totrue
): bool?lensesCheck
(optional, defaults totrue
): bool?. Performs check if user is wearing glasses during Selfie Scan.cameraFacing
(optional, defaults toIdScanCameraFacing.front
): Specifies which camera will be used during Selfie capture. To use back camera instead, setIdScanCameraFacing.back
.faceMaskCheck
(optional, defaults tofalse
): bool?. This checks if a person has a face mask on.brightnessThreshold
(optional, defaults to50
): int?. Adjust minimum requirements for a well lit face during capture.streamFrames
(optional, defaults to false): boolean. Enables/disables streaming of the camera feed. It requires SDK-vc
variant.
- FaceMatch
idCategory
(optional, defaults toprimary
):IdCategory.primary
orIdCategory.secondary
- Note: has to be specified after IDScan and SelfieScan modules.
- Geolocation
- no additional parameters
- GovernmentValidation
isBackgroundExecuted
- (optional, defaults tofalse
): bool?. Specify true to hide the module UI during its execution.
- VideoSelfie:
showTutorials
(optional, defaults totrue
): bool?. Show tutorial for video selfie.selfieScanMode
(optional, defaults toselfieMatch
):SelfieScanMode.selfieMatch
orSelfieScanMode.faceMatch
; Specify if you would like to do selfie comparison, or comparison with the photo from ID.selfieLivenessCheck
(optional, defaults tofalse
): bool?. Check for user liveness during video selfie.showIdScan
(optional, defaults totrue
): bool?. Ask for ID scan during video selfie.showDocumentScan
(optional, defaults totrue
): bool?. Ask for Proof of Address during video selfieshowVoiceConsent
(optional, defaults totrue
): bool?. Ask for Voice consent during video selfievoiceConsentQuestionsCount
(optional, defaults to3
): int?. Choose number of questions for video selfie voice consent steps.idScanCameraFacing
(optional, defaults toback
):IdScanCameraFacing.front
orIdScanCameraFacing.back
; Specify if you would like to use front or back camera for ID scan
- UserScore
mode
(optional, defaults toaccurate
):UserScoreFetchMode.accurate
orUserScoreFetchMode.fast
. Ifaccurate
the results will be fetched from server, which may exhibit some latency, but will rely on server-side processing of information, which may be more reliable than on-device processing. Iffast
, then results based on on-device processing will be shown.
- Approve
forceApproval
(optional, defaults tofalse
): bool? - iftrue
the user will be force-approved
- Signature
- no additional parameters
- DocumentScan
showTutorials
(optional, defaults totrue
): bool?. Show tutorial for document scan.showDocumentProviderOptions
(optional, defaults tofalse
).documentType
(optional, defaults toDocumentType.addressStatement
).
- Captcha
- no additional parameters
- CURP
- no additional parameters
- OCREdit
isEditable
(optional): defaults tofalse
: bool?
- EKYB
checkBusinessName
(optional): defaults totrue
: bool?checkAddress
(optional): defaults totrue
: bool?checkTaxId
(optional): defaults totrue
: bool?
- EKYC
checkName
(optional): defaults totrue
: bool?checkEmail
(optional): defaults totrue
: bool?checkAddress
(optional): defaults totrue
: bool?checkPhone
(optional): defaults totrue
: bool?checkSsn
(optional): defaults totrue
: bool?checkDob
(optional): defaults totrue
: bool?checkNationality
(optional): defaults totrue
: bool?
- MLConsent
type
: MLConsentType.gdpr or MLConsentType.us
- UserConsent
title
: String, title for user consentcontent
: String, content for user consent
-
CombinedConsentconsentId
: String, id for combined consent
- QRScan
showTutorials
(optional, defaults totrue
): bool?. Show tutorial for QR scan.
- Antifraud
- no additional parameters
- GlobalWatchlist
- no additional parameters
- CustomWatchlist
- no additional parameters
- Aes
uploadDocument
(optional, defaults tofalse
): bool?.downloadDocument
(optional, defaults tofalse
): bool?.
Modules interdependencies
ProcessId
module expectsIdScan
bothScanStepType.front
orScanStepType.back
to have executed.FaceMatch
module expectsIdScan
andSelfieScan
to have executed, in order to perform the match. In other words,IdScan
andSelfieScan
must precedeFaceMatch
VideoSelfie
module expectsIdScan
infaceMatch
mode, orSelfieScan
inselfieMatch
mode.UserScore
module should succeed all other modules (must be at the end)Approve
module should succeed all other modules (must be at the end)- The
UserScore
andApprove
modules do not depend on each other, so their order can be arbitrary. Aes
module expectsPhone
,IdScan
andSelfieScan
to precede it.
SDK results
Phone
Specify onAddPhoneNumberCompleted
to the startOnboarding
method, to receive PhoneNumberResult
:
phone
: String, the phone number user entered
Name
Specify onAddFullNameCompleted
to the startOnboarding
method, to receive AddFullNameResult
:
name
: String, the name user entered
Email
Specify onAddEmailCompleted
to the startOnboarding
method, to receive AddEmailResult
:
email
: String, the email user entered
IdScan
- Specify
onIdFrontCompleted
,onIdBackCompleted
to thestartOnboarding
method, to receiveIdScanResult
result:
image
: Captured ID imagebase64Image
: String?. Captured front ID in base64 formatcroppedFace
: Cropped face from captured ID imagechosenIdType
: User chosen type on ID selection screen-id
orpassport
classifiedIdType
: type of the captured IDidCategory
: IdCategory. Category of the scanned IDfailReason
: String?. Reason why ID scan failedissueName
: String?. Detailed type of the IDissueYear
: int?, year when the ID was issuedcountryCode
: String?, Country code where the ID was issuedscanStatus
: If status has a value other thanIdValidationStatus.ok
you can consider the ID scan and/or validation did not come through successfully. Other status messages are:errorClassification
,noFacesFound
,errorCropQuality
,errorGlare
,errorSharpness
,errorTypeMismatch
,userCancelled
,unknown
,errorAddress
,errorPassportClassification
.
- Specify
onIdProcessed
to thestartOnboarding
method, to receiveString
OCR result:
ocrData
: String?. Raw JSON containing full OCR data ie.exteriorNumber
,interiorNumber
,typeOfId
,documentFrontSubtype
SelfieScan
Specify onSelfieScanCompleted
to the startOnboarding
method, to receive SelfieScanResult
:
image
: Uint8List?. Captured Selfie imagespoofAttempt
: bool.false
means that person trying to do selfie is a real person.true
means it is a spoof attempt, meaning that person is trying to spoof the system by pretending to be someone else using a physical paper, digital photo or other methods.nil
means that unexpected error happened so it couldn't be determined if the selfie scan was a spoof attempt or not.base64Images
: SelfieScanBase64Images. Contains image, in different formats, taken during Selfie ScanselfieBase64
: String?. Captured Selfie base64 imageselfieEncryptedBase64
: String?. Captured Selfie encrypted base64 image
FaceMatch
Specify onFaceMatchCompleted
to the startOnboarding
method, to receive FaceMatchResult
:
faceMatched
: bool.true
means person's selfie matched successfully with the front ID.false
it means that person's selfie isn't matched with the front ID image.null
it means that front ID image wasn't uploaded at all, so the face match service didn't have data to compare with selfieidCategory
: IdCategory. Category of the ID that was used for face match.existingUser
: bool. Indicates whether the user is new or existing one.existingInterviewId
: String?. If user is existing user this field is populated with existing interview id.
Geolocation
Specify onGeolocationCompleted
to the startOnboarding
method, to receive GeoLocationResult
:
city
: String?colony
: String?postalCode
: String?state
: String?street
: String?
GovernmentValidation
Specify onGovernmentValidationCompleted
to the startOnboarding
method, to receive GovernmentValidationResult
:
success
: bool,true
if the government validation was performed successfully,false
otherwise.
VideoSelfie
Specify onVideoSelfieCompleted
to the startOnboarding
method, to receive VideoSelfieResult
:
success
: bool,true
if the video selfie was performed successfully,false
otherwise.
Antifraud
Specify onAntifraudCompleted
to the startOnboarding
method, to receive AntifraudResult
:
success
: bool,true
if the antifraud was passed successfully,false
otherwise.
UserScore
Specify onUserScoreFetched
to the startOnboarding
method, to receive UserScoreResult
.
Example UserScoreResult
for completion of the module:
{
ovarall: {
value: '0.0/100',
status: 'ok'
},
faceRecognition: {
value: '0.0/100',
status: 'warn',
},
liveness: {
value: '95.2/100',
status: 'manual',
},
idValidation: {
value: '79.0/100',
status: 'fail',
},
}
The field status
can have one of the following values: warning
, unknown
, manual
, fail
and ok
.
Approve
Specify onApproveCompleted
to the startOnboarding
method, to receive ApprovalResult
.
success
: bool.true
if the approval was successful,false
otherwise.uuid
: String?. Customer Id of newly created customer if approval was successful,null
otherwise.customerToken
: String?. Customer token for newly created customer if approval was successful,null
otherwise.
Signature
Specify onSignatureCollected
to the startOnboarding
method, to receive SignatureResult
.
signature
: Uint8List?. Collected signature image.
Document
Specify onDocumentScanCompleted
to the startOnboarding
method, to receive DocumentScanResult
.
image
: Uint8List?. Document scan image.documentType
: DocumentType. Type of scanned document.address
: Map<String, dynamic>?. Address fetched from the document. Will be available only forDocumentType.addressStatement
ocrData
: Raw JSON containing full OCR data
Captcha
Specify onCaptchaCompleted
to the startOnboarding
method, to receive CaptchaResult
.
captcha
: String?. Entered captcha.
CURP
Specify onCurpValidationCompleted
to the startOnboarding
method, to receive CurpValidationResult
.
curp
: String?. User's CURP.valid
: bool?. Tells if user's CURP is valid. Null means there is no result (user decided to skip).data
: Map<String, dynamic>?. User's CURP data.
OCREdit
Specify onOCREditCompleted
to the startOnboarding
method, to receive OCREditResult
.
success
: bool.true
if OCREdit completed successfully,false
otherwise.
EKYB
Specify onEKYBCompleted
to the startOnboarding
method, to receive EKYBResult
.
success
: bool.true
if eKYB scan completed successfully,false
otherwise.
EKYC
Specify onEKYCCompleted
to the startOnboarding
method, to receive EKYCResult
.
success
: bool.true
if eKYC scan completed successfully,false
otherwise.
MLConsent
Specify onMLConsentCompleted
to the startOnboarding
method, to receive MLConsentResult
.
success
: bool.true
if the user has given the machine learning consent,false
otherwise.
UserConsent
Specify onUserConsentCompleted
to the startOnboarding
method, to receive UserConsentResult
.
success
: bool.true
if the user has given consent,false
otherwise.
CombinedConsent
Specify onCombinedConsentCompleted
to the startOnboarding
method, to receive CombinedConsentResult
.
success
: bool.true
if the user has given the data sharing consent,false
otherwise.
QRScan
Specify onQRScanCompleted
to the startOnboarding
method, to receive QRScanResult
.
success
: bool.true
if QR scan completed successfully,false
otherwise.
GlobalWatchlist
Specify onGlobalWatchlistCompleted
to the startOnboarding
method, to receive GlobalWatchlistResult
.
success
: bool.true
if GlobalWatchlist completed successfully,false
otherwise.
CustomWatchlist
Specify onCustomWatchlistCompleted
to the startOnboarding
method, to receive CustomWatchlistResult
.
success
: bool.true
if CustomWatchlist completed successfully,false
otherwise.
Aes
Specify onAesCompleted
to the startOnboarding
method, to receive AesResult
.
success
: bool.true
if Aes completed successfully,false
otherwise.error
: AesError.
Advanced Usage
If you would like to use SDK in a way that the default flow builder doesn't provide,
you can use SDK APIs for advanced usage where you'll be able to fully customize the experience of the flow,
ie. by calling individual SDK modules, or grouping SDK modules in sections, and returning control to your host application in between.
Setup an onboarding session
Before calling any other Onboarding SDK components it is necessary to setup an onboarding session.
OnboardingSessionConfiguration sessionConfiguration =
OnboardingSessionConfiguration();
IncodeOnboardingSdk.setupOnboardingSession(
sessionConfig: sessionConfiguration,
onError: (String error) {
print('Incode onboarding session error: $error');
},
onSuccess: (OnboardingSessionResult result) {
print('Incode Onboarding session created! $result');
},
);
Session configuration can be configured in the same way as explained in the Usage Example
section Configure Onboarding session
.
Configure section flow
Once the new onboarding session is created (See previous section), you can separate Onboarding SDK flow into multiple sections based on your needs.
IncodeOnboardingFlowConfiguration flowConfig = IncodeOnboardingFlowConfiguration();
flowConfig.addIdScan();
Start onboarding section
Once the IncodeOnboardingFlowConfiguration
is created call the following method:
IncodeOnboardingSdk.startNewOnboardingSection(
flowConfig: flowConfig,
flowTag: 'idSection',
onError: (String error) {
print('Incode onboarding session error: $error');
},
onIdFrontCompleted: (IdScanResult result) {
print('onIdFrontCompleted result: $result');
},
onIdBackCompleted: (IdScanResult result) {
print('onIdBackCompleted result: $result');
},
onIdProcessed: (String ocrData) {
print('onIdProcessed result: $ocrData');
},
onOnboardingSectionCompleted: (String flowTag) {
print('section completed');
},
);
Start flow
Starts the flow based on the OnboardingSessionConfiguration
provided - specify the configurationId
, and optionally the interviewId
if you want to resume a certain onboarding session and/or moduleId
to start from a specific step within the flow.
OnboardingSessionConfiguration sessionConfig =
OnboardingSessionConfiguration(
configurationId: "YOUR_CONFIGURATION_ID",
interviewId: "YOUR_INTERVIEW_ID"); // optional
IncodeOnboardingSdk.startFlow(
sessionConfig: sessionConfig,
moduleId: 'YOUR_MODULE_ID', // optional, ie. "PHONE"
onError: (String error) {
print('Incode startFlow error: $error');
},
onSuccess: () {
print('Incode startFlow completed!');
},
onUserCancelled: () {
print('User cancelled');
},
);
Start flow from deep link
Starts the flow based on the deeplink URL. This method will read configurationId
, interviewId
and a step from which it should start.
IncodeOnboardingSdk.startFlowFromDeepLink(
url: 'YOUR_DEEPLINK_URL',
onError: (String error) {
print('Incode startFlowFromDeepLink error: $error');
},
onSuccess: () {
print('Incode startFlowFromDeepLink completed!');
},
onUserCancelled: () {
print('User cancelled');
},
);
Finish onboarding session
Make sure to call finishFlow()
at the end of the flow (when you are sure user has finished all onboarding modules and you won't be reusing same interviewId again).
IncodeOnboardingSdk.finishFlow();
Face Login
Prerequisites for a successful Face Login is that user has an approved account with an enrolled face.
Face Login 1:1
1:1 Face Login performs a 1:1 face comparison, and returns a successful match if the two faces match.
For 1:1 Face login you need to have at hand his customerUUID
. If the user was approved during onboarding on mobile device, you should have received customerUUID
as a result of Approve step during onboarding.
IncodeOnboardingSdk.startFaceLogin(
faceLogin: FaceLogin(customerUUID: "yourCustomerUUID"),
onSuccess: (FaceLoginResult result) {
print(result);
},
onError: (String error) {
print(error);
},
);
FaceLoginResult
will contain:
- image: selfie image
- spoofAttempt: boolean that indicates if user tried to spoof the system
- base64Images: base64 representations of the selfie image
- faceMatched: boolean that indicates if the faces matched
- customerUUID: unique user identifier if the user successfully authenticated
- interviewId: sessionId in which the user got approved
- interviewToken: session token in which the user got approved
- token: token that can be used for further API calls
- transactionId: unique identifier of face login attempt
- hasLenses: indicator if login attempt failed due to person wearing lenses
- hasFaceMask: indicator if login attempt failed due to person wearing a face mask.
Face Login 1:N
1:N Face Login performs a 1:N database face lookup, and returns a successful match if face is found in the database.
IncodeOnboardingSdk.startFaceLogin(
faceLogin: FaceLogin(),
onSuccess: (FaceLoginResult result) {
print(result);
},
onError: (String error) {
print(error);
},
);
FaceLoginResult
will contain:
- image: selfie image
- spoofAttempt: boolean that indicates if user tried to spoof the system
- base64Images: base64 representations of the selfie image
- faceMatched: boolean that indicates if the faces matched
- customerUUID: unique user identifier if the user successfully authenticated
- interviewId: sessionId in which the user got approved
- interviewToken: session token in which the user got approved
- token: token that can be used for further API calls
- transactionId: unique identifier of face login attempt
Face Login parametrization
Authorization modes
By default Face Login will perform server spoof and face recogniton check.
In order to optimize and speed up Face Login there are two other options that you can provide to FaceLogin
parameter faceAuthMode
: FaceAuthMode.hybrid
and FaceAuthMode.local
.
FaceAuthMode.hybrid
will perform spoof check locally on the device, and if it is successful it will perform server face recognition check.
FaceAuthMode.local
will perform both spoof check and face recognition check locally on the device, thus making it possible to authenticate users while being offline. Prerequisite for a successful offline face login is that the user was onboarded and approved on the same device.
To use FaceAuthMode.local
mode on Android please add 'com.incode.sdk:model-liveness-detection:2.0.0' and 'com.incode.sdk:model-face-recognition:2.0.0' depenendcies to your app/build.gradle
file.
To perform a one-time server authentication in case a user isn't found locally on the device during FaceAuthMode.local
execution, specify faceAuthModeFallback
to true. It works only in 1:1 Face Login mode.
To adjust spoof and recognition thresholds for FaceAuthMode.local
and FaceAuthMode.hybrid
modes, specify spoofThreshold
and recognitionThreshold
params to FaceLogin
object.
FaceAuthMode.kiosk
will perform only face recognition check on the server. It is currently only supported on Android platform.
To use FaceAuthMode.kiosk
mode on Android please add 'com.incode.sdk:kiosk-login:1.0.0' dependency to your app/build.gradle
file.
Face mask check
By default Face Login won't detect if people wear face masks.
To enable face mask check specify faceMaskCheck
parameter to true in FaceLogin
.
To use faceMaskCheck
in Android please add 'com.incode.sdk:model-mask-detection:2.0.0' dependency to your app/build.gradle
file.
Leneses check
By default Face Login will detect if people wear lenses.
To disable lenses check specify lensesCheck
parameter to false in FaceLogin
.
Log Authentication attempts
By default each authentication attempt is logged, and some statistics info like device used is being tracked.
Specify logAuthenticationEnabled
false if you want to disable this to get faster performing authentications.
Manipulating locally stored identities
To be able to authenticate multiple users using 1:N and FaceAuhtMode.local
mode you'll need to add these users to the local database. This section will describe which methods you can use to perform CRUD operations with locally stored identities.
Add Face
To add a single identity to the local dabtase, use addFace
method and provide a FaceInfo
object, that contains:
faceTemplate
: String -> biometric representation of a user's facecustomerUUID
: String -> unique customer identifer in Incode's databasetemplateId
: String -> unique identifier of a biometric representation of a user's face in Incode's database
FaceInfo faceInfo = FaceInfo(faceTemplate: template,
customerUUID: uuid,
templateId: templateId)
IncodeOnboardingSdk.addFace(
faceInfo: faceInfo,
onSuccess: (bool result) {
print(result);
},
onError: (String error) {
print(error);
});
Remove Face
To remove a single identity from the local database use removeFace
method and provide a customerUUID
:
IncodeOnboardingSdk.removeFace(
customerUUID: "your_customer_uuid",
onSuccess: (bool result) {
print(result);
},
onError: (String error) {
print(error);
});
Get faces
To fetch all currently stored identities use getFaces
method.
IncodeOnboardingSdk.getFaces(
onSuccess: (List<FaceInfo> faceInfos) => {
print("faceInfos: $faceInfos")},
onError: (String error) {
print(error);
});
Set multiple faces
To add multiple identities at once you can use setFaces
method and provide a list of FaceInfo
objects, but keep in mind it will firstly wipe out all currently stored identities.
FaceInfo faceInfo =
FaceInfo("your_face_template", "your_customer_uuid", "your_template");
FaceInfo faceInfo2 =
FaceInfo("your_face_template2", "your_customer_uuid2", "your_template2");
final faceInfos = <FaceInfo>[
FaceInfo("your_face_template", "your_customer_uuid", "your_template"),
FaceInfo("your_face_template2", "your_customer_uuid2", "your_template2")
];
IncodeOnboardingSdk.setFaces(
faces: faceInfos,
onSuccess: (bool result) => {print("result: $result")},
onError: (String error) {
print(error);
});
Clear face database
To clear local database use setFaces
method and provide empty FaceInfo
list of objects.
IncodeOnboardingSdk.setFaces(
faces: List.empty(),
onSuccess: (bool result) => {print("result: $result")},
onError: (String error) {
print(error);
});
Non-UI ID Processing API
Processes IDs programmatically without displaying a user interface
IncodeOnboardingSdk.idProcess(
idCategory: IdCategory.primary,
onSuccess: (IdProcessResult result) {
print(result);
},
onError: (String error) {
print(error);
},
);
IdProcessResult
will contain:
ocrData
: String?. Raw JSON containing full OCR data ie.exteriorNumber
,interiorNumber
,typeOfId
,documentFrontSubtype
Customization
To change theme and resources (text, images and videos) on Android platform please look at a guide here.
To change resources on iOS platform please look at a guide here.
To change theme on iOS platform specify json theme configuration and call IncodeOnboardingSdk.setTheme(theme: theme)
:
Map<String, dynamic> theme = {
"colors": {
"accent": "#00B2FD",
"primary": "#20263D",
"background": "#FFFFFF",
"secondaryBackground": "#E9E9EB",
"success": "#0CD5A2",
"error": "#FF5C6F",
"warning": "#F3AB3C",
"cancel": "#20263D"
},
"fonts": {
"buttonBig": {
"name": "CircularXXTT-Black",
"size": "20"
},
"buttonMedium": {
"name": "CircularXXTT-Black",
"size": "16"
},
"buttonSmall": {
"name": "CircularXXTT-Black",
"size": "12"
},
"title": {
"name": "CircularXXTT-Black",
"size": "25"
},
"hugeTitle": {
"name": "CircularXXTT-Black",
"size": "40"
},
"subtitle": {
"name": "CircularXXTT-Black",
"size": "18"
},
"boldedSubtitle": {
"name": "CircularXXTT-Bold",
"size": "18"
},
"smallSubtitle": {
"name": "CircularXXTT-Black",
"size": "14"
},
"info": {
"name": "CircularXXTT-Black",
"size": "16"
},
"body": {
"name": "CircularXXTT-Medium",
"size": "14"
},
"boldedBody": {
"name": "CircularXXTT-Bold",
"size": "14"
},
"textFieldBig": {
"name": "CircularXXTT-Black",
"size": "20"
},
"textFieldMedium": {
"name": "CircularXXTT-Black",
"size": "15"
}
},
"buttons": {
"primary": {
"states": {
"normal": {
"animateStateChange": true,
"alpha": 1,
"backgroundColor": "#00B2FD",
"borderColor": "",
"borderWidth": 0,
"cornerRadius": 32,
"shadowColor": "#000000",
"shadowOffset": [0,5],
"shadowOpacity": 0.15,
"shadowRadius": 9,
"textColor": "#FFFFFF",
"transform": {
"a": 1,
"b": 0,
"c": 0,
"d": 1,
"tx": 0,
"ty": 0
}
},
"highlighted": {
"animateStateChange": true,
"alpha": 1,
"backgroundColor": "#20263D",
"borderColor": "",
"borderWidth": 0,
"cornerRadius": 32,
"shadowColor": "#000000",
"shadowOffset": [0,5],
"shadowOpacity": 0.15,
"shadowRadius": 9,
"textColor": "#00B2FD",
"transform": {
"a": 1,
"b": 0,
"c": 0,
"d": 1,
"tx": 0,
"ty": 0
}
},
"disabled": {
"animateStateChange": true,
"alpha": 1,
"backgroundColor": "#E9E9EB",
"borderColor": "",
"borderWidth": 0,
"cornerRadius": 32,
"shadowColor": "#000000",
"shadowOffset": [0,5],
"shadowOpacity": 0,
"shadowRadius": 9,
"textColor": "#FFFFFF",
"transform": {
"a": 1,
"b": 0,
"c": 0,
"d": 1,
"tx": 0,
"ty": 0
}
}
},
"big": {
"height": 64,
"minWidth": 200,
"contentInsets": {
"top": 19,
"left": 32,
"bottom": 19,
"right": 32
},
"kerning": 0
},
"medium": {
"height": 46,
"minWidth": 0,
"contentInsets": {
"top": 12,
"left": 24,
"bottom": 12,
"right": 24
},
"kerning": 0
}
},
"secondary": {
"states": {
"normal": {
"animateStateChange": true,
"alpha": 1,
"backgroundColor": "#FFFFFF",
"borderColor": "#20263D",
"borderWidth": 1,
"cornerRadius": 32,
"shadowColor": "",
"shadowOffset": [0,0],
"shadowOpacity": 0,
"shadowRadius": 0,
"textColor": "#20263D",
"transform": {
"a": 1,
"b": 0,
"c": 0,
"d": 1,
"tx": 0,
"ty": 0
}
},
"highlighted": {
"animateStateChange": true,
"alpha": 1,
"backgroundColor": "#20263D",
"borderColor": "#20263D",
"borderWidth": 1,
"cornerRadius": 32,
"shadowColor": "",
"shadowOffset": [0,0],
"shadowOpacity": 0,
"shadowRadius": 0,
"textColor": "#00B2FD",
"transform": {
"a": 1,
"b": 0,
"c": 0,
"d": 1,
"tx": 0,
"ty": 0
}
},
"disabled": {
"animateStateChange": true,
"alpha": 1,
"backgroundColor": "#FFFFFF",
"borderColor": "#E9E9EB",
"borderWidth": 1,
"cornerRadius": 32,
"shadowColor": "",
"shadowOffset": [0,0],
"shadowOpacity": 0,
"shadowRadius": 0,
"textColor": "#E9E9EB",
"transform": {
"a": 1,
"b": 0,
"c": 0,
"d": 1,
"tx": 0,
"ty": 0
}
}
},
"big": {
"height": 64,
"minWidth": 200,
"contentInsets": {
"top": 19,
"left": 32,
"bottom": 19,
"right": 32
},
"kerning": 0
},
"medium": {
"height": 46,
"minWidth": 0,
"contentInsets": {
"top": 12,
"left": 24,
"bottom": 12,
"right": 24
},
"kerning": 0
}
},
"text": {
"states": {
"normal": {
"animateStateChange": true,
"alpha": 1,
"backgroundColor": "",
"borderColor": "",
"borderWidth": 0,
"cornerRadius": 0,
"shadowColor": "",
"shadowOffset": [0,0],
"shadowOpacity": 0,
"shadowRadius": 0,
"textColor": "#20263D",
"transform": {
"a": 1,
"b": 0,
"c": 0,
"d": 1,
"tx": 0,
"ty": 0
}
},
"highlighted": {
"animateStateChange": true,
"alpha": 1,
"backgroundColor": "",
"borderColor": "",
"borderWidth": 0,
"cornerRadius": 0,
"shadowColor": "",
"shadowOffset": [0,0],
"shadowOpacity": 0,
"shadowRadius": 0,
"textColor": "#00B2FD",
"transform": {
"a": 1,
"b": 0,
"c": 0,
"d": 1,
"tx": 0,
"ty": 0
}
},
"disabled": {
"animateStateChange": true,
"alpha": 1,
"backgroundColor": "",
"borderColor": "",
"borderWidth": 0,
"cornerRadius": 0,
"shadowColor": "",
"shadowOffset": [0,0],
"shadowOpacity": 0,
"shadowRadius": 0,
"textColor": "#E9E9EB",
"transform": {
"a": 1,
"b": 0,
"c": 0,
"d": 1,
"tx": 0,
"ty": 0
}
}
},
"big": {
"height": 40,
"minWidth": 200,
"contentInsets": {
"top": 8,
"left": 16,
"bottom": 8,
"right": 16
},
"kerning": 0
},
"medium": {
"height": 30,
"minWidth": 0,
"contentInsets": {
"top": 12,
"left": 24,
"bottom": 12,
"right": 24
},
"kerning": 0
}
},
"help": {
"states": {
"normal": {
"animateStateChange": true,
"alpha": 1,
"backgroundColor": "#00B2FD",
"borderColor": "",
"borderWidth": 0,
"cornerRadius": 32,
"shadowColor": "#000000",
"shadowOffset": [0,5],
"shadowOpacity": 0.15,
"shadowRadius": 9,
"textColor": "#FFFFFF",
"transform": {
"a": 1,
"b": 0,
"c": 0,
"d": 1,
"tx": 0,
"ty": 0
}
},
"highlighted": {
"animateStateChange": true,
"alpha": 1,
"backgroundColor": "#20263D",
"borderColor": "",
"borderWidth": 0,
"cornerRadius": 32,
"shadowColor": "#000000",
"shadowOffset": [0,5],
"shadowOpacity": 0.15,
"shadowRadius": 9,
"textColor": "#00B2FD",
"transform": {
"a": 1,
"b": 0,
"c": 0,
"d": 1,
"tx": 0,
"ty": 0
}
},
"disabled": {
"animateStateChange": true,
"alpha": 1,
"backgroundColor": "#E9E9EB",
"borderColor": "",
"borderWidth": 0,
"cornerRadius": 32,
"shadowColor": "#000000",
"shadowOffset": [0,5],
"shadowOpacity": 0,
"shadowRadius": 9,
"textColor": "#FFFFFF",
"transform": {
"a": 1,
"b": 0,
"c": 0,
"d": 1,
"tx": 0,
"ty": 0
}
}
},
"big": {
"height": 64,
"minWidth": 200,
"contentInsets": {
"top": 19,
"left": 32,
"bottom": 19,
"right": 32
},
"kerning": 0
},
"medium": {
"height": 46,
"minWidth": 0,
"contentInsets": {
"top": 12,
"left": 24,
"bottom": 12,
"right": 24
},
"kerning": 0
}
}
},
"labels": {
"title": {
"textColor": "#20263D",
"kerning": 0
},
"secondaryTitle": {
"textColor": "#FFFFFF",
"kerning": 0
},
"subtitle": {
"textColor": "#20263D",
"kerning": 0
},
"secondarySubtitle": {
"textColor": "#FFFFFF",
"kerning": 0
},
"smallSubtitle": {
"textColor": "#20263D",
"kerning": 0
},
"info": {
"textColor": "#636670",
"kerning": 0
},
"secondaryInfo": {
"textColor": "#FFFFFF",
"kerning": 0
},
"body": {
"textColor": "#20263D",
"kerning": 0
},
"secondaryBody": {
"textColor": "#FFFFFF",
"kerning": 0
},
"code": {
"textColor": "#20263D",
"kerning": 16
}
},
"customComponents": {
"cameraFeedback": {
"alpha": 0.8,
"backgroundColor": "#000000",
"cornerRadius": 20,
"textBackgroundColor": "",
"textColor": "#FFFFFF"
},
"idCaptureHelp": {
"commonIssueLayoutOrientation": "horizontal"
},
"idSideLabel": {
"alpha": 1,
"backgroundColor": "#FFFFFF",
"borderColor": "",
"borderWidth": 0,
"cornerRadius": 5
},
"separator": {
"alpha": 1.0,
"color": "#20263D",
"cornerRadius": 0,
"padding": 24,
"thickness": 1
},
"signature": {
"signatureColor": "#04BD19",
"canvasBorderColor": "#EC03FC"
},
"idAutocaptureCountdown": {
"backgroundColor": "#00B2FD",
"numberColor": "#FFFFFF"
}
},
};
IncodeOnboardingSdk.setTheme(theme: theme);
Using SDK without an API KEY
To use the SDK without an API KEY, please follow these steps:
- provide only a specific
apiUrl
only to theinit
method. - Afterwards, configure
OnboardingSessionConfiguration
with atoken
- Provide it either to the
startOnboarding
, or if you're using sections to thesetupOnboardingSession
and then start your sections.
Example code that showcases steps 1) and 2:
IncodeOnboardingSdk.init(
apiUrl: 'https://demo-api.incodesmile.com/0/',
onError: (String error) {
print('Incode SDK init failed: $error');
setState(() {
initSuccess = false;
});
},
onSuccess: () {
// Update UI, safe to start Onboarding
print('Incode initialize successfully!');
OnboardingSessionConfiguration sessionConfiguration = OnboardingSessionConfiguration(token: "YOUR_TOKEN");
// call `startOnboarding` or `setupOnboardingSession` and then start sections.
},
);
Other Public API methods
SDK Mode
You can choose between two modes: SdkMode.standard
and SdkMode.captureOnly
. SdkMode.standard
is the default, but if you would like to skip image upload and server validations for id and selfie scan you can specify captureOnly mode using method:
IncodeOnboardingSdk.setSdkMode(sdkMode: SdkMode.captureOnly);
Allowing user to cancel the flow
You can use showCloseButton
method to display an 'X' button on the top right of each module, so that user can cancel the flow at any point:
IncodeOnboardingSdk.showCloseButton(allowUserToCancel: true);
Tracking events
- When onboarding has been started, information about events used to track all the user steps in their flow can be obtained using
onEvents
callback.
Specify onEvents
to the startOnboarding
method, to receive OnEventsResult
:
event
: String - Unique identifier of the eventdata
: String? - JSON string with additional event details
Set Localization Language
To programatically set localization language in runtime, call IncodeSdk.setLocalizationLanguage
.
Parameters available for IncodeSDK.setLocalizationLanguage
method:
language
- language used for runtime localization. Supported values are currently: 'en', 'es' , 'pt'.
IncodeOnboardingSdk.setLocalizationLanguage('es');
Additionally, on Android platform a new dependency needs to be added in your app's build/gradle:
implementation 'com.incode.sdk:extensions:1.1.0'
Set string
To programatically set and update strings in runtime, call IncodeSdk.setString
.
Parameters available for IncodeSDK.setString
method:
strings
: Map<String, dynamic>. Map of string keys and its values.
Map<String, dynamic> strings = {
/// iOS labels
'incdOnboarding.idChooser.title': 'My Custom chooser title',
/// Android labels
'onboard_sdk_id_type_chooser_title': 'My Custom chooser title',
};
IncodeOnboardingSdk.setString(strings: strings);
Additionally, on Android platform a new dependency needs to be added in your app's build/gradle:
implementation 'com.incode.sdk:extensions:1.1.0'
For iOS please check documentation here:
https://developer.incode.com/docs/localization-guide#list-of-localizable-texts
For Android please check documentation here:
https://developer.incode.com/docs/user-guide-customization#32-dynamic-localization
Get Score API
You can use getUserScore
method to fetch current onboarding session user score at any point:
IncodeOnboardingSdk.getUserScore(
onSuccess: (UserScoreResult result) {
print("userScore: $result");
},
onError: (String error) {
print("userScore error: $error");
});
UserScoreResult
has extendedUserScoreJsonData
String field which contains full user data in a json format.
AddNom151Archive API
You can use addNOM151Archive
method to generate and fetch Nom151Archive:
IncodeOnboardingSdk.addNOM151Archive(
onSuccess: (AddNom151Result addNom151Result) {
String? signature = addNom151Result.signature;
String? archiveUrl = addNom151Result.archiveUrl;
}, onError: (String error) {
print('Incode addNOM151Archive error: $error');
});
AddNom151Result
contains two String fields signature
and archiveUrl
as an and result.
Get user OCR data API
You can use getUserOCRData
method to fetch user OCR data for a specific session:
IncodeOnboardingSdk.getUserOCRData(
token: "{SESSION_TOKEN}",
onSuccess: (GetUserOCRDataResult result) {
print(result); //
},
onError: (String error) {
print(error);
},
);
GetUserOCRDataResult
contains a String field ocrData
, which represents full OCR data in a JSON format.
Known issues
- Running the app on iOS simulator from VSCode isn't supported currently. Please run the app from Xcode for now if you want to test it on iOS Simulator.
NOTE: Don't forget to settestMode
totrue
before running the app.
Updated 29 days ago