Skip to content

Latest commit

 

History

History
1596 lines (1382 loc) · 135 KB

Customization.md

File metadata and controls

1596 lines (1382 loc) · 135 KB

Configuration

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:

Application

The base application settings should be specified in app.js file.

  • APP_NAME - (string, Mandatory) name of the application

    export 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')
  • 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'
  • 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')
  • 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')
  • 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')

Environment

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 and DEFAULT_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' 
      ```
  • 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'
      ```
      

Receiving Message

There are two strategies regarding receiving messages by an application:

  1. Polling - app once in a while calls Cloud Agent to get all new messages for all existing connections.

  2. 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
      }

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
  • 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!

Color theme

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
      export const COMPANY_LOGO = null
      Default:
      {
        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',
       ...
      }

Font

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'
  • FONT_SIZES - (string, Optional) grid to use for fonts.

    • to use default - Lato
      export const FONT_SIZES = null
      Default:
      {
        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,
        ...
      

End User License Agreement

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'
  • 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'
  • 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>

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'
    • 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'
    • 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'
  • 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'
    • 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'
    • 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'
    • 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'

Note: By default, MSDK tries to use web versions of documents. Local assets will be used when there are connectivity issues.

Start up

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')
  • 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>
  • 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. If null 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 from Settings.

Lock

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>

Home

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'
  • 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>
        )
      }
  • 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
  • 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>

Connections

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'
  • 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>
        )
      }
  • 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
  • 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>
  • 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>

Credentials

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'
  • 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>
        )
      }
  • 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
  • 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>
  • 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>

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
  • AUTO_ACCEPT_CREDENTIAL_PRESENTATION_REQUEST - (boolean, Optional) whether you want to automatically accept following presentation 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
  • 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'
  • 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
      } 

Navigation Menu

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
      export const MENU_NAVIGATION_OPTIONS = null
      Default tabs:
      • 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.

  • 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>
  • 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>
  • 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
        }
      ]
  • 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
        }
      ]

Collecting log information

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.

  • CUSTOM_LOG_UTILS - (object) key or URL to the file containing key used for log encryption.

    export let CUSTOM_LOG_UTILS = {
      publicKeyUrl: '...',
      encryptionKey: '...',
    }

Credential Offer

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'
  • 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'
  • 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'
  • 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
      } 
  • 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
      }        

Proof Request

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'
  • 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'
  • 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'
  • 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
      } 
  • 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
      }         
  • 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
      }          
  • 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
      }         

Proof Proposal

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'
  • 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'
  • 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'
  • 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
      } 

Proof

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 of Shared Proof dialog.

    • to use default - Proof
      export const SHARED_PROOF_HEADLINE = null
    • to use custom
      export const SHARED_PROOF_HEADLINE = 'Custom Header'
  • 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
      } 
  • RECEIVED_PROOF_HEADLINE - (string) the text which will be used for the header of Received Proof dialog.

    • to use default - Proof
      export const RECEIVED_PROOF_HEADLINE = null
    • to use custom
      export const RECEIVED_PROOF_HEADLINE = 'Custom Header'
  • 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
      } 

Question

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'
  • 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
      }         

Invite Action

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'
  • 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'
  • 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'
  • 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
      }        

Settings

You can customize Settings view in the settings.js module.

  1. 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'
  2. 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 app
      • Passcode - change your app passcode
      • Logs - send logs to development team
      • About - application information
      • Feedback - give the app a feedback
      • ManualBackup - 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.md
      • ViewRecoveryPassphrase - show passphrase used for local backup generation (works only if ManualBackup 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
  • 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>

Feedback

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: '-',
  },
})

Application information

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 logo
      • appName - show application name
      • appVersion - show application version
      • appEnvironment - show application environment
      • builtBy - show company label/name built application
      • poweredBy - powered by Evernym label
      • termsAndConditions - end user license agreement
      • privacyPolicy - 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>
  • 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>

Physical Document Verification

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:

  1. At the start, User will be requested for giving Camera permissions.
  2. User select country document belongs to.
  3. User select document type.
  4. Scan document (for some documents two sides need to be scanned).
  5. Scan User face.
  6. 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

Splash screen and app icon

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 and files/drawable-mdpi directories into your android/app/src/main/res directory.
    • Application icon: replace file ic_launcher.png in android/app/src/main/res/mipmap-hdpi directory with a desired one.

  • iOS: TODO

Credential attachments

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

Examples

Credential

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

Proof Request containing attachments:

{
  "requested_attributes": {
    "attr_1": {
      "name": "First Name"
    },
    "attr_2": {
      "name": "Photo_link"
    }
  }
}

Advanced

For advanced customizations, you can refer to this document describing MSDK internals.