After running evernym-sdk:configure
command all required modules and assets will set up with default values.
Ensure that you completed build configuration for target platforms.
You should be able to run the application at this point or proceed to modify provided default configuration.
For more convenience, we grouped all configuration options by files representing either a corresponding application screen or piece of functionality.
For example home.js
contains options for Home
screen.
Content:
- Configuration
- Application
- Environment
- Receiving Message
- Color theme
- Font
- End User License Agreement
- Start up
- Lock
- Home
- Connections
- Credentials
- Navigation Menu
- Collecting log information
- Credential Offer
- Proof Request
- Proof Proposal
- Proof
- Question
- Invite Action
- Settings
- Feedback
- Application information
- Physical Document Verification
- Splash screen and app icon
- Credential attachments
- Examples
- Advanced
- Application
The base application settings should be specified in app.js
file.
-
APP_NAME
- (string, Mandatory) name of the applicationexport const APP_NAME = 'AppName'
-
APP_ICON
- (image source, Optional) application icon-
to use default MSDK icon
export const APP_ICON = null
-
to use custom
export const APP_ICON = require('path/to/app_icon.png')
-
-
APP_LOGO
- (image source, Optional) small application logo used on several screens.- to use default MSDK logo
export const APP_LOGO = null
- to use custom
export const APP_LOGO = require('path/to/logo_app.png')
- to use default MSDK logo
-
COMPANY_NAME
- (string, Optional) name of a company built app.- to omit
export const APP_LOGO = null
- to use custom
export const COMPANY_NAME = 'Company'
- to omit
-
COMPANY_LOGO
- (image source, Optional) logo of a company built application.- to omit
export const COMPANY_LOGO = null
- to use custom
export const COMPANY_LOGO = require('path/to/app_company.png')
- to omit
-
DEFAULT_USER_AVATAR
- (image source, Optional) default user avatar placeholder.- to use default avatar
export const DEFAULT_USER_AVATAR = null
- to use custom
export const DEFAULT_USER_AVATAR = require('path/to/user_avatar.png')
- to use default avatar
-
DEEP_LINK
- (string, Optional) Branch.io Deep link address.-
to omit
export const DEEP_LINK = null
-
to use custom
export const DEEP_LINK = 'https://address.com'
-
-
PUSH_NOTIFICATION_PERMISSION_SCREEN_IMAGE_IOS
- (image source, Optional) For iOS side we have push notification permission screen with image which simulate ConnectMe notification by default. It's way for customization this screen.- to omit
export const PUSH_NOTIFICATION_PERMISSION_SCREEN_IMAGE_IOS = null
- to use custom
export const PUSH_NOTIFICATION_PERMISSION_SCREEN_IMAGE_IOS = require('iphoneX.png')
- to use custom
- to omit
You should set up an environment to be used by your application in the provision.js
module.
-
Application environment
-
DEFAULT_SERVER_ENVIRONMENT
- the name of environment to use.There are several predefined environments:
// use default combination - DEMO for debug and PROD for releases builds export const DEFAULT_SERVER_ENVIRONMENT = null // use Demo env // Agency: `https://agency.pps.evernym.com` and `Sovrin Staging Net` export const DEFAULT_SERVER_ENVIRONMENT = 'DEMO' // use Production env // Agency: `https://agency.evernym.com` and `Sovrin Live Net` export const DEFAULT_SERVER_ENVIRONMENT = 'PROD' // use Staging env // Agency: `https://agency.pstg.evernym.com` and `Sovrin Staging Net` export const DEFAULT_SERVER_ENVIRONMENT = 'STAGING'
-
-
SERVER_ENVIRONMENTS
- (object) additional custom server configurations:- to use default environments
export const SERVER_ENVIRONMENTS = {}
- to add custom environment
export const SERVER_ENVIRONMENTS = { 'CUSTOM': { agencyUrl: 'ahency_url', agencyDID: 'did', agencyVerificationKey: 'verkey', poolConfig: [{ key: 'staging', genesis: 'genesis_transactions' }], } }
You can provide and use your custom environment using a combination of
SERVER_ENVIRONMENTS
andDEFAULT_SERVER_ENVIRONMENT
variables:export const SERVER_ENVIRONMENTS = { 'CUSTOM': { agencyUrl: 'ahency_url', agencyDID: 'did', agencyVerificationKey: 'verkey', poolConfig: [{ key: 'staging', genesis: 'genesis_transactions' }], } } export const DEFAULT_SERVER_ENVIRONMENT = 'CUSTOM' ```
- to use default environments
-
Information used for application provisioning
-
GET_PROVISION_TOKEN_FUNC
- function to be called to get provisioning token./// example export const GET_PROVISION_TOKEN_FUNC = async (): [error: string | null, token: string | null] => { try { const response = fetch_api(your_endpoint) /// process response return [null, response.token] } catch (error) { return [error.message, null] } }
-
SPONSOR_ID
- An ID given to you from Evernym's Support Team after the Sponsor onboarding process is complete.```javascript export const SPONSOR_ID = 'sponsorid' ```
-
There are two strategies regarding receiving messages by an application:
-
Polling - app once in a while calls Cloud Agent to get all new messages for all existing connections.
-
Push Notifications - There is configured Push Notification service which notifies the application about new messages.
By default, app uses Polling strategy which follows rules:
- Download messages every 2 seconds for 60 seconds after the user performs an action like:
- Accept a connection invitation
- Accept a credential offer
- Share a proof
- Answer a question
- Download messages every 3 seconds for another 2 minutes after first point.
- Download messages every 15 seconds the all rest time.
- Download messages when a user navigates to
Home
screen. - Download messages by manual pulling screen down
You can change timeouts by setting up POOLING_INTERVALS
variable.
POOLING_INTERVALS
- (object, Optional) timeouts (in milliseconds) to trigger message downloading.- to use default -
{ 'short': 2000, // 2 seconds 'medium': 3000, // 3 seconds 'long': 15000, // 15 seconds }
export const POOLING_INTERVALS = null
- to enable
export const POOLING_INTERVALS = { 'short': 5000, // 5 seconds 'medium': 6000, // 6 seconds 'long': 30000, // 30 seconds }
- to use default -
If you wish to use Push Notifications strategy you need to set variable USE_PUSH_NOTIFICATION
in the app.js
module:
-
USE_PUSH_NOTIFICATION
- (boolean, Optional) whether you want to enable push notifications logic.- to use default - false
export const USE_PUSH_NOTIFICATION = null
- to enable
export const USE_PUSH_NOTIFICATION = true
- to use default - false
-
VCX_PUSH_TYPE
- (number) type of push communication method to use.export const VCX_PUSH_TYPE = 3
-
Type 3 - Forward Push
This method is used when the sponsor wants to handle sending of actual push notifications.
Mobile app is supposed to register its own push notification details.
Associated sponsor’s configuration should have been already configured with an endpoint.
In this case, Verity will use the associated sponsor’s configured endpoint and send the meta data (msg type, sponsee details, recipient DID etc.) to the associated sponsor’s endpoint. The sponsor is supposed to notify the mobile app about the message, and then, the mobile app is supposed to download the message from its agent by calling the relevant api.
-
Type 4 - Sponsor Push
If using this method, Verity will handle sending of actual push notifications.
Associated sponsor’s configuration should have been already configured with an endpoint.
Verity will use the associated sponsor’s configured push notification account detail instead of Evernym’s push notification account detail.
Currently, only Firebase Cloud Messaging is supported.
-
NOTE that if you decided to enable Push Notifications you MUST configure Firebase for target build platforms!
Application color theme is set by a group of constants provided in colors.js
configuration module.
It is used throughout the whole application.
COLORS
- (object, Optional) color palette to use.- to use default
Default:
export const COMPANY_LOGO = null
{ main: '#86B93B', secondary: 'rgba(134, 185, 59, 0.15)', green1: '#86B93B', green2: '#6C8E3A', green3: 'rgba(134, 185, 59, 0.15)', red: '#CE0B24', orange: '#EB9B2D', white: '#FFFFFF', gray5: '#F2F2F2', gray4: '#EAEAEA', gray3: '#A5A5A5', gray2: '#777777', gray1: '#505050', gray0: '#404040', black: '#000000', blue: '#236BAE', }
- to use custom
export const COMPANY_LOGO = { main: '#236BAE', secondary: '#11ABAE', ... }
- to use default
You can specify the font which will be used in the app inside the font.js
module.
-
FONT_FAMILY
- (string, Optional) font family to use.- to use default -
Lato
export const FONT_FAMILY = null
- to use custom
export const FONT_FAMILY = 'Roboto'
- to use default -
-
FONT_SIZES
- (string, Optional) grid to use for fonts.- to use default -
Lato
Default:export const FONT_SIZES = null
{ size0: 42, size1: 26, size2: 23, size3: 19, size4: 17, size5: 15, size6: 14, size7: 13, size8: 11, size9: 10, size10: 9, size11: 8, }
- to use custom
export const FONT_SIZES = { size0: 36, size1: 22, ...
- to use default -
You can configure EULA and privacy terms inside the eula.js
module.
-
TERMS_AND_CONDITIONS_TITLE
- (string, Optional) the text which will be used for the label.- to use default -
Terms and Conditions
export const TERMS_AND_CONDITIONS_TITLE = null
- to use custom
export const TERMS_AND_CONDITIONS_TITLE = 'Custom Terms and Conditions'
- to use default -
-
PRIVACY_POLICY_TITLE
- (string, Optional) the text which will be used for the label.- to use default -
Privacy Policy
export const PRIVACY_POLICY_TITLE = null
- to use custom
export const PRIVACY_POLICY_TITLE = 'Custom Privacy Policy'
- to use default -
-
CustomEulaScreen
- (React Component) custom component for Eula screen rendering (instead of predefined one).- to use default
export const CustomEulaScreen = null
- to use custom
export const CustomEulaScreen = () => <Text>Custom Eula</Text>
- to use default
There are two type variables used for specifying documents location:
-
URL - url address leading to web document version (is used by default)
-
ANDROID_EULA_URL
- (string, Optional) url leading to EULA for android app- to use default -
https://www.connect.me/google.html
export const ANDROID_EULA_URL = null
- to use custom
export const ANDROID_EULA_URL = 'https://www.custom./androud_eula.html'
- to use default -
-
IOS_EULA_URL
-(string, Optional) url leading to EULA for ios app- to use default -
https://www.connect.me/ios_eula.html
export const IOS_EULA_URL = null
- to use custom
export const IOS_EULA_URL = 'https://www.custom.me/ios_eula.html'
- to use default -
-
PRIVACY_POLICY_URL
- (string, Optional) url leading to Privacy policy document- to use default -
https://www.connect.me/privacy.html
export const PRIVACY_POLICY_URL = null
- to use custom
export const PRIVACY_POLICY_URL = 'https://www.connect.me/privacy.html'
- to use default -
-
-
LOCAL - path to local asset
-
ANDROID_EULA_LOCAL
- (string, Optional) path to local EULA file for android app- to use default -
None
export const ANDROID_EULA_LOCAL = null
- to use custom
export const ANDROID_EULA_LOCAL = 'file:///eula_android.html'
- to use default -
-
IOS_EULA_LOCAL
- (string, Optional) path to local EULA file for ios app- to use default -
None
export const IOS_EULA_LOCAL = null
- to use custom
export const IOS_EULA_LOCAL = './eula_ios.html'
- to use default -
-
ANDROID_PRIVACY_POLICY_LOCAL
- (string, Optional) path to local Privacy policy document for android app- to use default -
None
export const ANDROID_PRIVACY_POLICY_LOCAL = null
- to use custom
export const ANDROID_PRIVACY_POLICY_LOCAL = 'file:///privacy.html'
- to use default -
-
IOS_PRIVACY_POLICY_LOCAL
- (string, Optional) path to local Privacy policy document for ios app- to use default -
None
export const IOS_PRIVACY_POLICY_LOCAL = null
- to use custom
export const IOS_PRIVACY_POLICY_LOCAL = './privacy.html'
- to use default -
-
Note: By default, MSDK tries to use web versions of documents. Local assets will be used when there are connectivity issues.
You can configure application startup wizard which is shown for the newly installed application inside the startup.js
module.
-
BACKGROUND_IMAGE
- (image source, Optional) image to use as a background:- to use default MSDK star up background
export const BACKGROUND_IMAGE = null
- to use custom
export const BACKGROUND_IMAGE = require('path/to/setup.png')
- to use default MSDK star up background
-
CustomStartUpScreen
- (React Component) custom component for Start Up screen rendering (instead of predefined one).- to use default
export const CustomStartUpScreen = null
- to use custom
export const CustomStartUpScreen = () => <Text>Custom Start Up</Text>
- to use default
-
ANDROID_DEVICE_CHECK_API_KEY
- (Android device verification API key generated from Google cloud console). This SDK also provides option to secure your app such that your app is authorized to run only on non-rooted devices and only on real devices. if you set this key this, SDK will check for device integrity and will show messages as you configure them using below constants. Although the name of variable has Android in it, SDK will run check on both Android and iOS if this key is set. Ifnull
is passed, then device verification will not be done on both Android and iOS. -
deviceSecurityCheckFailedMessage
- This message is shown if device is rooted or if release build is running on simulator/emulator. -
devicePlayServiceUpdateRequiredMessage
- This message is shown only for Android devices. If an Android device has incompatible version or old version of Play Service, then this message will be show to user with an option to update play service. -
devicePlayServiceRequiredMessage
- This message is shown only for Android devices. If an Android device has play services disabled, then this message will be shown along with an option to enable fromSettings
.
You can configure application locking screens (set up / enter / change password) inside the lock.js
module.
LockHeader
- (React Component, Optional)component which will be displayed as the header (above password input):- to omit
export const LockHeader = null
- to use custom
export const LockHeader = () => <Text>Hello</Text>
- to omit
You can configure application Home
screen inside the home.js
module.
-
HEADLINE
- (string, Optional) the text which will be used for the header.- to use default -
Home
export const HEADLINE = null
- to use custom
export const HEADLINE = 'Custom Home'
- to use default -
-
HomeViewEmptyState
- (React Component, Optional) component to be displayed at the home screen in cases of no recent notifications.This will usually happen after new installation of the application.
You can provide a greeting message as in this example:
- to use default
export const HomeViewEmptyState = null
- to omit
export const HomeViewEmptyState = () => null
- to use custom
export const HomeViewEmptyState = () => { return ( <Text>Hello, you now have a digital wallet!</Text> ) }
- to use default
-
SHOW_EVENTS_HISTORY
- (boolean, Optional) a flag indicating whether you want to show the history of events on the Home view.- to use default -
show
export const SHOW_EVENTS_HISTORY = null
- to use custom
export const SHOW_EVENTS_HISTORY = true
- to use default -
-
CustomMyConnectionsScreen
- (React Component) custom component for Connections screen rendering (instead of predefined one).- to use default
export const CustomHomeScreen = null
- to use custom
export const CustomHomeScreen = () => <Text>Custom Home</Text>
- to use default
You can configure application Connections
screen inside the connections.js
module.
-
HEADLINE
- (string, Optional) the text which will be used for the header.- to use default -
show
export const HEADLINE = null
- to use custom
export const HEADLINE = 'Custom Connections'
- to use default -
-
MyConnectionsViewEmptyState
- (React Component, Optional) component to be displayed at the connections screen in cases of no connections made yet.- to use default
export const MyConnectionsViewEmptyState = null
- to omit
export const MyConnectionsViewEmptyState = () => null
- to use custom
export const MyConnectionsViewEmptyState = () => { return ( <Text>You do not have connections yet!</Text> ) }
- to use default
-
SHOW_CAMERA_BUTTON
- (boolean, Optional) flag indicating whether you want to show camera button.- to use default -
true
export const SHOW_CAMERA_BUTTON = null
- to use custom
export const SHOW_CAMERA_BUTTON = false
- to use default -
-
CustomMyConnectionsScreen
- (React Component) custom component for Connections screen rendering (instead of predefined one).- to use default
export const CustomMyConnectionsScreen = null
- to use custom
export const CustomMyConnectionsScreen = () => <Text>Custom Connections</Text>
- to use default
-
CustomConnectionDetailsScreen
- (React Component) custom component for Connection Details screen rendering (instead of predefined one).- to use default
export const CustomConnectionDetailsScreen = null
- to use custom
export const CustomConnectionDetailsScreen = () => <Text>Custom Connection Details</Text>
- to use default
You can configure application Credentials
screen inside the credentials.js
module.
-
HEADLINE
- (string, Optional) the text which will be used for the header.- to use default -
show
export const HEADLINE = null
- to use custom
export const HEADLINE = 'Custom Credentials'
- to use default -
-
MyCredentialsViewEmptyState
- (React Component, Optional) component to be displayed at the credentials screen in cases of no credentials made yet.- to use default
export const MyCredentialsViewEmptyState = null
- to omit
export const MyCredentialsViewEmptyState = () => null
- to use custom
export const MyCredentialsViewEmptyState = () => { return ( <Text>You do not have credentials yet!</Text> ) }
- to use default
-
SHOW_CAMERA_BUTTON
- (boolean, Optional) flag indicating whether you want to show camera button.- to use default -
true
export const SHOW_CAMERA_BUTTON = null
- to use custom
export const SHOW_CAMERA_BUTTON = false
- to use default -
-
CustomMyCredentialsScreen
- (React Component) custom component for Credentials screen rendering (instead of predefined one).- to use default
export const CustomMyCredentialsScreen = null
- to use custom
export const CustomMyCredentialsScreen = () => <Text>Custom Credentials</Text>
- to use default
-
CustomCredentialDetailsScreen
- (React Component) custom component for Credential Details screen rendering (instead of predefined one).- to use default
export const CustomCredentialDetailsScreen = null
- to use custom
export const CustomCredentialDetailsScreen = () => <Text>Custom Credential Details</Text>
- to use default
You can also configure application Show Credentail
modal dialog or disable this feature.
-
SHOW_CREDENTIAL
- (boolean, Optional) whether you want to use the feature of presenting a credential (it reveals credential data to Verifier scanning QR code).- to use default -
true
export const SHOW_CREDENTIAL = null
- to use custom
export const SHOW_CREDENTIAL = false
- to use default -
-
AUTO_ACCEPT_CREDENTIAL_PRESENTATION_REQUEST
- (boolean, Optional) whether you want to automatically accept followingpresentation request
and generate proof or show it to user for manually accepting.NOTE: acceptably if
SHOW_CREDENTIAL
feature is enable- to use default -
false
export const AUTO_ACCEPT_CREDENTIAL_PRESENTATION_REQUEST = null
- to use custom
export const AUTO_ACCEPT_CREDENTIAL_PRESENTATION_REQUEST = true
- to use default -
-
SHOW_CREDENTIAL_HEADLINE
- (string, Optional) the text which will be used for the header.- to use default -
show
export const SHOW_CREDENTIAL_HEADLINE = null
- to use custom
export const SHOW_CREDENTIAL_HEADLINE = 'Custom Show Credential'
- to use default -
-
CustomShowCredentialModal
- (React Component) custom component for Show Credential modal window rendering (instead of predefined one).- to use default
export const CustomShowCredentialModal = null
- to use custom
export const CustomShowCredentialModal = { screen: () => <Text>Custom Proof CustomShowCredentialModal</Text>, // Optional, React Component navigationOptions: {}, // Optional, ModalStack.Screen Options - https://reactnavigation.org/docs/screen-options }
- to use default
You can configure navigation menu and app navigation inside the navigator.js
module.
-
MENU_NAVIGATION_OPTIONS
- (object) The set of navigation options (and their labels) to be shown.- to use default
Default tabs:
export const MENU_NAVIGATION_OPTIONS = null
- Home
- Connections
- Credentials
- Settings
- to change predefined (for predefined routes
name, label, route, icon
are optional fields / defaults will be used if they are not specified)// Menu contains Home and Connections tabs export const MENU_NAVIGATION_OPTIONS = [ { name: 'Connections', label: 'Other Connection Label' } ]
- to change order
export const MENU_NAVIGATION_OPTIONS = [ { name: 'Settings', }, { name: 'Connections', }, { name: 'Credentials', } ]
- to add new route
// Menu contains Home and My Route tabs export const Component = () => { return <Text style={{color: colors.black}}>MY SCREEN</Text> } export const MENU_NAVIGATION_OPTIONS = [ { name: 'My Route', // id label: 'My Route', // label to show route: 'route', // route name icon: <Icon name="my" />, // icon to use component: Component // React Component to render } ]
Note -
Home
screen is always included. - to use default
-
DrawerHeaderContent
- (React Component) You can provide component to be displayed in the navigation drawer at the top, above the navigation section.- to use default
export const DrawerHeaderContent = null
- to omit
export const DrawerHeaderContent = () => null
- to use custom
export const DrawerHeaderContent = (props: { height: number, width: number, fill: string, }) => <Text>You are using sdk-app</Text>
- to use default
-
DrawerFooterContent
- (React Component) You can provide component to be displayed in the navigation drawer at the bottom, below the navigation section.- to use default
export const DrawerFooterContent = null
- to omit
export const DrawerFooterContent = () => null
- to use custom
export const DrawerFooterContent = () => <Text>You are using wallet 1.0.0</Text>
- to use default
-
EXTRA_SCREENS
- (object) additional routes need to be registered in the app navigator inside Screens Stack Navigator (see https://reactnavigation.org/docs/stack-navigator/)- to use default
export const EXTRA_SCREENS = null export const EXTRA_SCREENS = []
- to add custom
export const Component = () => { return <Text style={{color: colors.black}}>MY SCREEN</Text> } export const EXTRA_SCREENS = [ { route: 'route', // route name component: Component, // React Component to render options: { title: 'Awesome app' } // see https://reactnavigation.org/docs/screen-options } ]
- to use default
-
EXTRA_MODALS
- (object) additional routes need to be registered in the app navigator inside Modal Windows Stack Navigator (see https://reactnavigation.org/docs/stack-navigator/)- to use default
export const EXTRA_MODALS = null export const EXTRA_MODALS = []
- to add custom
export const Component = () => { return <Text style={{color: colors.black}}>MY SCREEN</Text> } export const EXTRA_MODALS = [ { route: 'route', // route name component: Component, // React Component to render options: { title: 'Awesome app' } // see https://reactnavigation.org/docs/screen-options } ]
- to use default
You can configure data used for logging in the logs.js
module.
You can receive encrypted log file by email.
-
SEND_LOGS_EMAIL
- (string) - email to send logs.- to use default -
[email protected]
export const SEND_LOGS_EMAIL = null
- to use custom
export const SEND_LOGS_EMAIL = '[email protected]'
- to use default -
-
CUSTOM_LOG_UTILS
- (object) key or URL to the file containing key used for log encryption.export let CUSTOM_LOG_UTILS = { publicKeyUrl: '...', encryptionKey: '...', }
You can customize Credential Offer
dialog in the credential-offer.js
module.
-
HEADLINE
- (string) the text which will be used for the header.- to use default -
Credential Offer
export const HEADLINE = null
- to use custom
export const HEADLINE = 'Custom Credential Offer'
- to use default -
-
ACCEPT_BUTTON_TEXT
- (string) the text which will be used for top (accept) button.- to use default -
Accept Credential
export const ACCEPT_BUTTON_TEXT = null
- to use custom
export const ACCEPT_BUTTON_TEXT = 'Accept'
- to use default -
-
DENY_BUTTON_TEXT
- (string) the text which will be used for bottom (deny) button.- to use default -
Reject
export const DENY_BUTTON_TEXT = null
- to use custom
export const DENY_BUTTON_TEXT = 'Deny'
- to use default -
-
CustomCredentialOfferModal
- (React Component) custom component for Credential Offer dialog rendering (instead of predefined one).- to use default
export const CustomCredentialOfferModal = null
- to use custom
export const CustomCredentialOfferModal = { screen: () => <Text>Custom Credential Offer Dialog</Text>, // Optional, React Component navigationOptions: {}, // Optional, ModalStack.Screen Options - https://reactnavigation.org/docs/screen-options }
- to use default
-
CustomCredentialOfferModal
- (React Component) custom component for received Credential dialog rendering (instead of predefined one).- to use default
export const CustomCredentialModal = null
- to use custom
export const CustomCredentialModal = { screen: () => <Text>Custom Credential Dialog</Text>, // Optional, React Component navigationOptions: {}, // ModalStack.Screen Options - https://reactnavigation.org/docs/screen-options }
- to use default
You can customize Proof Request
dialog in the proof-request.js
module.
-
HEADLINE
- (string) the text which will be used for the header.- to use default -
Proof Request
export const HEADLINE = null
- to use custom
export const HEADLINE = 'Custom Request'
- to use default -
-
ACCEPT_BUTTON_TEXT
- (string) the text which will be used for top (accept) button.- to use default -
Share Attributes
export const ACCEPT_BUTTON_TEXT = null
- to use custom
export const ACCEPT_BUTTON_TEXT = 'Accept'
- to use default -
-
DENY_BUTTON_TEXT
- (string) the text which will be used for bottom (deny) button.- to use default -
Reject
export const DENY_BUTTON_TEXT = null
- to use custom
export const DENY_BUTTON_TEXT = 'Deny'
- to use default -
-
CustomProofRequestModal
- (React Component) custom component for received Proof Request dialog rendering (instead of predefined one).- to use default
export const CustomProofRequestModal = null
- to use custom
export const CustomProofRequestModal = { screen: () => <Text>Custom Proof Request Dialog</Text>, // Optional, React Component navigationOptions: {}, // Optional, ModalStack.Screen Options - https://reactnavigation.org/docs/screen-options }
- to use default
-
CustomSelectAttributeValueModal
- (React Component) custom component for selecting a credential for filling a requested attribute in Proof (instead of predefined one).- to use default
export const CustomSelectAttributeValueModal = null
- to use custom
export const CustomSelectAttributeValueModal = { screen: () => <Text>Custom Dialog</Text>, // Optional, React Component navigationOptions: {}, // Optional, ModalStack.Screen Options - https://reactnavigation.org/docs/screen-options }
- to use default
-
CustomSelectAttributesValuesModal
- (React Component) custom component for selecting a credential for filling a requested attribute group in Proof (instead of predefined one).- to use default
export const CustomSelectAttributesValuesModal = null
- to use custom
export const CustomSelectAttributesValuesModal = { screen: () => <Text>Custom Dialog</Text>, // Optional, React Component navigationOptions: {}, // Optional, ModalStack.Screen Options - https://reactnavigation.org/docs/screen-options }
- to use default
-
CustomEnterAttributeValueModal
- (React Component) custom component for entering a custom value for a requested attribute which can be self attested in Proof (instead of predefined one).- to use default
export const CustomEnterAttributeValueModal = null
- to use custom
export const CustomEnterAttributeValueModal = { screen: () => <Text>Custom Dialog</Text>, // Optional, React Component navigationOptions: {}, // Optional, ModalStack.Screen Options - https://reactnavigation.org/docs/screen-options }
- to use default
You can customize Proof Proposal
dialog in the proof-proposal.js
module.
-
HEADLINE
- (string) the text which will be used for the header.- to use default -
Proof Proposal
export const HEADLINE = null
- to use custom
export const HEADLINE = 'Custom Proposal'
- to use default -
-
ACCEPT_BUTTON_TEXT
- (string) the text which will be used for top (accept) button.- to use default -
Accept
export const ACCEPT_BUTTON_TEXT = null
- to use custom
export const ACCEPT_BUTTON_TEXT = 'Ok'
- to use default -
-
DENY_BUTTON_TEXT
- (string) the text which will be used for bottom (deny) button.- to use default -
Cancel
export const DENY_BUTTON_TEXT = null
- to use custom
export const DENY_BUTTON_TEXT = 'Deny'
- to use default -
-
CustomProofProposalModal
- (React Component) custom component for received Proof Proposal dialog rendering (instead of predefined one).- to use default
export const CustomProofProposalModal = null
- to use custom
export const CustomProofProposalModal = { screen: () => <Text>Custom Proof Proposal Dialog</Text>, // Optional, React Component navigationOptions: {}, // Optional, ModalStack.Screen Options - https://reactnavigation.org/docs/screen-options }
- to use default
You can customize Shared Proof
and Received Proof
dialogs in the proof.js
module.
-
SHARED_PROOF_HEADLINE
- (string) the text which will be used for the header ofShared Proof
dialog.- to use default -
Proof
export const SHARED_PROOF_HEADLINE = null
- to use custom
export const SHARED_PROOF_HEADLINE = 'Custom Header'
- to use default -
-
CustomSharedProofModal
- (React Component) custom component for received Shared Proof dialog rendering (instead of predefined one).- to use default
export const CustomSharedProofModal = null
- to use custom
export const CustomSharedProofModal = { screen: () => <Text>Custom Proof Dialog</Text>, // Optional, React Component navigationOptions: {}, // Optional, ModalStack.Screen Options - https://reactnavigation.org/docs/screen-options }
- to use default
-
RECEIVED_PROOF_HEADLINE
- (string) the text which will be used for the header ofReceived Proof
dialog.- to use default -
Proof
export const RECEIVED_PROOF_HEADLINE = null
- to use custom
export const RECEIVED_PROOF_HEADLINE = 'Custom Header'
- to use default -
-
CustomProofProposalModal
- (React Component) custom component for received Proof dialog rendering (instead of predefined one).- to use default
export const CustomProofProposalModal = null
- to use custom
export const CustomReceivedProofModal = { screen: () => <Text>Custom Proof Dialog</Text>, // Optional, React Component navigationOptions: {}, // Optional, ModalStack.Screen Options - https://reactnavigation.org/docs/screen-options }
- to use default
You can customize Question
dialog in the question-dialog.js
module.
-
HEADLINE
- (string) the text which will be used for the header.- to use default -
Proof Request
export const HEADLINE = null
- to use custom
export const HEADLINE = 'Custom Request'
- to use default -
-
CustomQuestionModal
- (React Component) custom component for rendering of Question dialog (instead of predefined one).- to use default
export const CustomQuestionModal = null
- to use custom
export const CustomQuestionModal = { screen: () => <Text>Custom Question Dialog</Text>, // Optional, React Component navigationOptions: {}, // Optional, ModalStack.Screen Options - https://reactnavigation.org/docs/screen-options }
- to use default
You can customize Invite Action
dialog in the invite-action.js
module.
-
HEADLINE
- (string) the text which will be used for the header.- to use default -
New Message
export const HEADLINE = null
- to use custom
export const HEADLINE = 'Custom Headline'
- to use default -
-
ACCEPT_BUTTON_TEXT
- (string) the text which will be used for top (accept) button.- to use default -
Accept
export const ACCEPT_BUTTON_TEXT = null
- to use custom
export const ACCEPT_BUTTON_TEXT = 'Ok'
- to use default -
-
DENY_BUTTON_TEXT
- (string) the text which will be used for bottom (deny) button.- to use default -
Reject
export const DENY_BUTTON_TEXT = null
- to use custom
export const DENY_BUTTON_TEXT = 'Deny'
- to use default -
-
CustomInviteActionModal
- (React Component) custom component for Invite Action dialog rendering (instead of predefined one).- to use default
export const CustomInviteActionModal = null
- to use custom
export const CustomInviteActionModal = { screen: () => <Text>Custom Question Dialog</Text>, // Optional, React Component navigationOptions: {}, // Optional, ModalStack.Screen Options - https://reactnavigation.org/docs/screen-options }
- to use default
You can customize Settings
view in the settings.js
module.
-
HEADLINE
- (string) the text which will be used for the header.- to use default -
Settings
export const HEADLINE = null
- to use custom
export const HEADLINE = 'Custom Settings'
- to use default -
-
settingsOptions
- (object) The set of options to be shown. Optionally, you can also specify option title, subtitle, and icon.-
to use default
export const SETTINGS_OPTIONS = null
Defaults:
- Biometrics
- Passcode
- Logs
- About
Predefined Options:
Biometrics
- enable/disable using finger or face to secure appPasscode
- change your app passcodeLogs
- send logs to development teamAbout
- application informationFeedback
- give the app a feedbackManualBackup
- ability to create the application local backup file which can be used for restoring or sharing the app. Note, that you may need to configure document picker: https://github.com/rnmods/react-native-document-picker/blob/4053e6106440a8f711a0e82a74949c7e51213105/install-old.mdViewRecoveryPassphrase
- show passphrase used for local backup generation (works only ifManualBackup
is enabled).
-
to change predefined (for predefined options
title, subtitle, avatar, rightIcon, onPress
are optional fields / defaults will be used if they are not specified)// Menu contains Home and Connections tabs export const SETTINGS_OPTIONS = [ { name: 'Biometrics', label: 'Other Biometrics Label' } ]
-
to change order
export const SETTINGS_OPTIONS = [ { name: 'About', }, { name: 'Biometrics', }, { name: 'Logs', } ]
-
to add new setting
// Settings contains Biometrics and Custom settings export const SETTINGS_OPTIONS = [ { name: 'Biometrics', }, { name: 'Custom', title: 'Custom Option', // title subtitle: null, // (optional) - description avatar: null, // (optional) - icon to show on the left rightIcon: null, // (optional) - icon to show on the right onPress: null, // (optional) - handler on touch }, ]
-
-
SHOW_CAMERA_BUTTON
- (boolean, Optional) flag indicating whether you want to show camera button.- to use default -
true
export const SHOW_CAMERA_BUTTON = null
- to use custom
export const SHOW_CAMERA_BUTTON = false
- to use default -
-
CustomSettingsScreen
- (React Component) custom component for rendering of Settings screen (instead of predefined one).- to use default
export const CustomSettingsScreen = null
- to use custom
export const CustomSettingsScreen = () => <Text>Custom Settings</Text>
- to use default
In order to gather application feedback is used Apptentive
.
You can provide credentials to be used for setting up Apptentive
module in feedback.js
file.
Note: This variable is mandatory if you want to enable feedback
option on Settings
screen.
export const APPTENTIVE_CREDENTIALS = Platform.select({
ios: {
apptentiveKey: '-',
apptentiveSignature: '-',
},
android: {
apptentiveKey: '-',
apptentiveSignature: '-',
},
})
The information about the application which will be shown on About
screen can be configured in app.js
file.
-
INFO
- (object) object specifying which information need to be show-
to use default -
appLogo, appName, appVersion, appEnvironment, builtBy, poweredBy, termsAndConditions, privacyPolicy
export const INFO = null
-
to use custom
export const INFO = { appLogo: true, // show application logo appName: true, // show application name }
Options:
appLogo
- show application logoappName
- show application nameappVersion
- show application versionappEnvironment
- show application environmentbuiltBy
- show company label/name built applicationpoweredBy
- powered by Evernym labeltermsAndConditions
- end user license agreementprivacyPolicy
- privacy policy document
-
-
AdditionalInfo
- (React Component) some additional information which will be show on the screen- to omit
export const AdditionalInfo = null
- to use custom
export const AdditionalInfo = () => <Text>Extra data</Text>
- to omit
-
CustomAboutAppScreen
- (React Component) custom component for rendering of About screen (instead of predefined one).- to use default
export const CustomAboutAppScreen = null
- to use custom
export const CustomAboutAppScreen = () => <Text>Custom About</Text>
- to use default
This feature allows users to get a verifiable credential after scanning and processing physical documents.
Three kinds of documents are currently supported (vary depending on the selected country):
- Passport
- Driver License
- Identity card
In order to enable Physical Document Verification feature you need to add PhysicalDocumentVerification
item into MENU_NAVIGATION_OPTIONS
(inside the navigator.js
module).
export const MENU_NAVIGATION_OPTIONS = [
// other options
{
name: 'PhysicalDocumentVerification',
}
]
Flow:
- At the start, User will be requested for giving Camera permissions.
- User select country document belongs to.
- User select document type.
- Scan document (for some documents two sides need to be scanned).
- Scan User face.
- After the document processing, User will receive Credential Offer from Evernym's Issuer Service.
Additional configuration for this feature, can be done in physical-document-verification.js
file.
IOS_GET_DEVICE_CHECK_JWT
- Function that would be called to verify device check token of ios app. Below are the details to fulfill this functionality:- Generate an API Key from developer.apple.com which only has access to call Device Check APIs
- In your developer account, go to Certificates, Identifiers & Profiles.
- Under Keys, select All and click the Add button (+) in the upper-right corner.
- Under Key Description, enter a unique name for the signing key.
- Under Key Services, select the Device Check, then click Continue.
- Review the key configuration, then click Confirm.
- Take note of key id.
- Click Download to generate and download the key now. If you download the key, it is saved as a text file with a .p8 file extension. Save the file in a secure place because the key is not saved in your developer account and you won’t be able to download it again.
- Click Done.
- On server side below code can be used to generate JWT. Note: The below functionality can be done inside the app itself. For example: when you are in development phase. But, we would recommend to do it on server side. You can use your sponsor provision backend to add one more API endpoint to generate ios specific JWT.
// server side api.js, function to get JWT const { getToken } = require('@sagi.io/workers-jwt') // team id or issuer-id const iss = '' // key-id const kid = '' const privateKeyPEM = '' const jwtPayload = { iss, iat: Math.floor(Date.now() / 1000) } const jwt = await getToken({ privateKeyPEM, payload: jwtPayload, alg: 'ES256', headerAdditions: { kid } })
- Mobile app can call above API via http method and get the JWT. Make sure to add proper Authorization/Authentication as per your own app logic for above API call
export const IOS_GET_DEVICE_CHECK_JWT = async function getDeviceCheckJwt(): Promise<[typeof Error | null, string | null]> { try { // call your backend api to get ios platform jwt, this API call is the one that contains above server side code const response = await fetch(`<your-app-backend-url>/get-ios-jwt`, { method: 'POST', headers: { 'Content-Type': 'application/json', 'Authorization': 'Bearer <your own auth strategy>' }, body: '{}', }) let responseText = await response.json() if (!response.ok) { return [ responseText.errorMessage || responseText.message || responseText, null, ] } // please ensure that response from this function is an array which has first value as Error or null and second value as jwt or null // assuming that your API call response was JSON and had a property `jwt` which contains jwt token return [null, responseText.jwt] } catch (e) { return [e, null] } }
- The function that call above API is the same function that we need to pass to
IOS_GET_DEVICE_CHECK_JWT
- Generate an API Key from developer.apple.com which only has access to call Device Check APIs
These are configured inside your application for specific platforms.
-
Android:
-
Splash Screen:
- Added following code into your
MainActivity.java
file:import org.devio.rn.splashscreen.SplashScreen; import android.os.Bundle; ... public class MainActivity extends ReactActivity { ... @Override protected void onCreate(Bundle savedInstanceState) { SplashScreen.show(this); super.onCreate(savedInstanceState); } ... } ```
- copy
files/layout
andfiles/drawable-mdpi
directories into yourandroid/app/src/main/res
directory.
- Added following code into your
-
Application icon: replace file
ic_launcher.png
inandroid/app/src/main/res/mipmap-hdpi
directory with a desired one.
-
-
iOS: TODO
When app gets an attribute with _link
postfix (example Photo_link
), it tries to render its value as attachment according to defined mime type.
Supported mime types:
- Photo types:
image/jpeg
image/png
image/jpg
- MS Word types:
application/msword
application/vnd.openxmlformats-officedocument.wordprocessingml.document
application/vnd.openxmlformats-officedocument.wordprocessingml.template
application/vnd.ms-word.document.macroEnabled.12
application/vnd.ms-word.template.macroEnabled.12
- MS Excel types:
application/vnd.ms-excel
application/vnd.openxmlformats-officedocument.spreadsheetml.sheet
application/vnd.openxmlformats-officedocument.spreadsheetml.template
application/vnd.ms-excel.sheet.macroEnabled.12
application/vnd.ms-excel.template.macroEnabled.12
application/vnd.ms-excel.addin.macroEnabled.12
application/vnd.ms-excel.sheet.binary.macroEnabled.12
- MS Powerpoint types:
application/vnd.ms-powerpoint
application/vnd.openxmlformats-officedocument.presentationml.presentation
application/vnd.openxmlformats-officedocument.presentationml.template
application/vnd.openxmlformats-officedocument.presentationml.slideshow
application/vnd.ms-powerpoint.addin.macroEnabled.12
application/vnd.ms-powerpoint.presentation.macroEnabled.12
application/vnd.ms-powerpoint.template.macroEnabled.12
application/vnd.ms-powerpoint.slideshow.macroEnabled.12
- CSV:
text/csv
- PDF types:
application/pdf
- Audio and video types:
audio/mp4
audio/mpeg
audio/mp3
video/mp4
Credential containing attachments:
{
'First Name': 'Faber',
'Photo_link': '{\"mime-type\": \"image/jpeg\", \"extension\": \"jpeg\", \"name\": \"my_photo.jpeg\",\"data\": { \"base64\": \"\"}}',
'PDF_link': '{"mime-type": "application/pdf", "extension": "pdf", "name": "my_pdf.pdf","data": { "base64": "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CmVuZG9iagoKOSAwIG9iago8PC9UeXBlL0ZvbnQvU3VidHlwZS9UcnVlVHlwZS9CYXNlRm9udC9CQUFBQUErQXJpYWwtQm9sZE1UCi9GaXJzdENoYXIgMAovTGFzdENoYXIgMTEKL1dpZHRoc1s3NTAgNzIyIDYxMCA4ODkgNTU2IDI3NyA2NjYgNjEwIDMzMyAyNzcgMjc3IDU1NiBdCi9Gb250RGVzY3JpcHRvciA3IDAgUgovVG9Vbmljb2RlIDggMCBSCj4+CmVuZG9iagoKMTAgMCBvYmoKPDwKL0YxIDkgMCBSCj4+CmVuZG9iagoKMTEgMCBvYmoKPDwvRm9udCAxMCAwIFIKL1Byb2NTZXRbL1BERi9UZXh0XT4+CmVuZG9iagoKMSAwIG9iago8PC9UeXBlL1BhZ2UvUGFyZW50IDQgMCBSL1Jlc291cmNlcyAxMSAwIFIvTWVkaWFCb3hbMCAwIDU5NSA4NDJdL0dyb3VwPDwvUy9UcmFuc3BhcmVuY3kvQ1MvRGV2aWNlUkdCL0kgdHJ1ZT4+L0NvbnRlbnRzIDIgMCBSPj4KZW5kb2JqCgoxMiAwIG9iago8PC9Db3VudCAxL0ZpcnN0IDEzIDAgUi9MYXN0IDEzIDAgUgo+PgplbmRvYmoKCjEzIDAgb2JqCjw8L1RpdGxlPEZFRkYwMDQ0MDA3NTAwNkQwMDZEMDA3OTAwMjAwMDUwMDA0NDAwNDYwMDIwMDA2NjAwNjkwMDZDMDA2NT4KL0Rlc3RbMSAwIFIvWFlaIDU2LjcgNzczLjMgMF0vUGFyZW50IDEyIDAgUj4+CmVuZG9iagoKNCAwIG9iago8PC9UeXBlL1BhZ2VzCi9SZXNvdXJjZXMgMTEgMCBSCi9NZWRpYUJveFsgMCAwIDU5NSA4NDIgXQovS2lkc1sgMSAwIFIgXQovQ291bnQgMT4+CmVuZG9iagoKMTQgMCBvYmoKPDwvVHlwZS9DYXRhbG9nL1BhZ2VzIDQgMCBSCi9PdXRsaW5lcyAxMiAwIFIKPj4KZW5kb2JqCgoxNSAwIG9iago8PC9BdXRob3I8RkVGRjAwNDUwMDc2MDA2MTAwNkUwMDY3MDA2NTAwNkMwMDZGMDA3MzAwMjAwMDU2MDA2QzAwNjEwMDYzMDA2ODAwNkYwMDY3MDA2OTAwNjEwMDZFMDA2RTAwNjkwMDczPgovQ3JlYXRvcjxGRUZGMDA1NzAwNzIwMDY5MDA3NDAwNjUwMDcyPgovUHJvZHVjZXI8RkVGRjAwNEYwMDcwMDA2NTAwNkUwMDRGMDA2NjAwNjYwMDY5MDA2MzAwNjUwMDJFMDA2RjAwNzIwMDY3MDAyMDAwMzIwMDJFMDAzMT4KL0NyZWF0aW9uRGF0ZShEOjIwMDcwMjIzMTc1NjM3KzAyJzAwJyk+PgplbmRvYmoKCnhyZWYKMCAxNgowMDAwMDAwMDAwIDY1NTM1IGYgCjAwMDAwMTE5OTcgMDAwMDAgbiAKMDAwMDAwMDAxOSAwMDAwMCBuIAowMDAwMDAwMjI0IDAwMDAwIG4gCjAwMDAwMTIzMzAgMDAwMDAgbiAKMDAwMDAwMDI0NCAwMDAwMCBuIAowMDAwMDExMTU0IDAwMDAwIG4gCjAwMDAwMTExNzYgMDAwMDAgbiAKMDAwMDAxMTM2OCAwMDAwMCBuIAowMDAwMDExNzA5IDAwMDAwIG4gCjAwMDAwMTE5MTAgMDAwMDAgbiAKMDAwMDAxMTk0MyAwMDAwMCBuIAowMDAwMDEyMTQwIDAwMDAwIG4gCjAwMDAwMTIxOTYgMDAwMDAgbiAKMDAwMDAxMjQyOSAwMDAwMCBuIAowMDAwMDEyNDk0IDAwMDAwIG4gCnRyYWlsZXIKPDwvU2l6ZSAxNi9Sb290IDE0IDAgUgovSW5mbyAxNSAwIFIKL0lEIFsgPEY3RDc3QjNEMjJCOUY5MjgyOUQ0OUZGNUQ3OEI4RjI4Pgo8RjdENzdCM0QyMkI5RjkyODI5RDQ5RkY1RDc4QjhGMjg+IF0KPj4Kc3RhcnR4cmVmCjEyNzg3CiUlRU9GCg=="}}',
'DOCX_link': '{"mime-type": "application/msword", "extension": "docx", "name": "my_docx.docx","data": { "base64": "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"}}',
'CSV_link': '{"mime-type": "text/csv", "extension": "csv", "name": "my_csv.csv","data": { "base64": "VXNlcm5hbWU7IElkZW50aWZpZXI7Rmlyc3QgbmFtZTtMYXN0IG5hbWUKYm9va2VyMTI7OTAxMjtSYWNoZWw7Qm9va2VyCmdyZXkwNzsyMDcwO0xhdXJhO0dyZXkKam9obnNvbjgxOzQwODE7Q3JhaWc7Sm9obnNvbgpqZW5raW5zNDY7OTM0NjtNYXJ5O0plbmtpbnMKc21pdGg3OTs1MDc5O0phbWllO1NtaXRoCgo="}}',
}
Proof Request containing attachments:
{
"requested_attributes": {
"attr_1": {
"name": "First Name"
},
"attr_2": {
"name": "Photo_link"
}
}
}
For advanced customizations, you can refer to this document describing MSDK internals.