diff --git a/app/Gemfile.lock b/app/Gemfile.lock index 4069d41b6..8b97a10a4 100644 --- a/app/Gemfile.lock +++ b/app/Gemfile.lock @@ -25,7 +25,7 @@ GEM artifactory (3.0.17) atomos (0.1.3) aws-eventstream (1.4.0) - aws-partitions (1.1172.0) + aws-partitions (1.1173.0) aws-sdk-core (3.233.0) aws-eventstream (~> 1, >= 1.3.0) aws-partitions (~> 1, >= 1.992.0) @@ -34,10 +34,10 @@ GEM bigdecimal jmespath (~> 1, >= 1.6.1) logger - aws-sdk-kms (1.113.0) + aws-sdk-kms (1.114.0) aws-sdk-core (~> 3, >= 3.231.0) aws-sigv4 (~> 1.5) - aws-sdk-s3 (1.199.1) + aws-sdk-s3 (1.200.0) aws-sdk-core (~> 3, >= 3.231.0) aws-sdk-kms (~> 1) aws-sigv4 (~> 1.5) diff --git a/app/metro.config.cjs b/app/metro.config.cjs index 768039970..74fcb51fd 100644 --- a/app/metro.config.cjs +++ b/app/metro.config.cjs @@ -66,11 +66,18 @@ const config = { new RegExp('packages/mobile-sdk-alpha/node_modules/react(/|$)'), new RegExp('packages/mobile-sdk-alpha/node_modules/react-dom(/|$)'), new RegExp('packages/mobile-sdk-alpha/node_modules/react-native(/|$)'), + new RegExp( + 'packages/mobile-sdk-alpha/node_modules/lottie-react-native(/|$)', + ), new RegExp('packages/mobile-sdk-alpha/node_modules/scheduler(/|$)'), + new RegExp( + 'packages/mobile-sdk-alpha/node_modules/react-native-svg(/|$)', + ), new RegExp('packages/mobile-sdk-demo/node_modules/react(/|$)'), new RegExp('packages/mobile-sdk-demo/node_modules/react-dom(/|$)'), new RegExp('packages/mobile-sdk-demo/node_modules/react-native(/|$)'), new RegExp('packages/mobile-sdk-demo/node_modules/scheduler(/|$)'), + new RegExp('packages/mobile-sdk-demo/node_modules/react-native-svg(/|$)'), ], // Enable automatic workspace package resolution enableGlobalPackages: true, @@ -95,6 +102,10 @@ const config = { assert: require.resolve('assert'), events: require.resolve('events'), process: require.resolve('process'), + 'react-native-svg': path.resolve( + projectRoot, + 'node_modules/react-native-svg', + ), // App-specific alias '@': path.join(__dirname, 'src'), }, diff --git a/app/src/components/loading/LoadingUI.tsx b/app/src/components/loading/LoadingUI.tsx index 2b746034b..9ae720f71 100644 --- a/app/src/components/loading/LoadingUI.tsx +++ b/app/src/components/loading/LoadingUI.tsx @@ -7,7 +7,8 @@ import React from 'react'; import { useSafeAreaInsets } from 'react-native-safe-area-context'; import { Text, View, XStack, YStack } from 'tamagui'; -import { DelayedLottieView } from '@/components/DelayedLottieView'; +import { DelayedLottieView } from '@selfxyz/mobile-sdk-alpha'; + import CloseWarningIcon from '@/images/icons/close-warning.svg'; import Plus from '@/images/icons/plus_slate600.svg'; import { diff --git a/app/src/layouts/ExpandableBottomLayout.tsx b/app/src/layouts/ExpandableBottomLayout.tsx index 2b46f6800..db660c08c 100644 --- a/app/src/layouts/ExpandableBottomLayout.tsx +++ b/app/src/layouts/ExpandableBottomLayout.tsx @@ -3,51 +3,32 @@ // NOTE: Converts to Apache-2.0 on 2029-06-11 per LICENSE. import React from 'react'; -import { - Dimensions, - PixelRatio, - Platform, - ScrollView, - StyleSheet, -} from 'react-native'; import { SystemBars } from 'react-native-edge-to-edge'; import { useSafeAreaInsets } from 'react-native-safe-area-context'; -import type { ViewProps } from 'tamagui'; -import { View } from 'tamagui'; -import { black, white } from '@/utils/colors'; -import { extraYPadding } from '@/utils/constants'; +import type { + BottomSectionProps, + FullSectionProps, + LayoutProps, + TopSectionProps, +} from '@selfxyz/mobile-sdk-alpha'; +import { ExpandableBottomLayout as BaseExpandableBottomLayout } from '@selfxyz/mobile-sdk-alpha'; -// Get the current font scale factor -const fontScale = PixelRatio.getFontScale(); -// fontScale > 1 means the user has increased text size in accessibility settings -const isLargerTextEnabled = fontScale > 1.3; +import { black } from '@/utils/colors'; -interface ExpandableBottomLayoutProps extends ViewProps { - children: React.ReactNode; - backgroundColor: string; -} - -interface TopSectionProps extends ViewProps { - children: React.ReactNode; - backgroundColor: string; - roundTop?: boolean; -} - -interface BottomSectionProps extends ViewProps { - children: React.ReactNode; - backgroundColor: string; -} - -const Layout: React.FC = ({ +const Layout: React.FC = ({ children, backgroundColor, + ...props }) => { return ( - + {children} - + ); }; @@ -57,24 +38,18 @@ const TopSection: React.FC = ({ ...props }) => { const { top } = useSafeAreaInsets(); - const { roundTop, ...restProps } = props; + return ( - {children} - + ); }; -type FullSectionProps = ViewProps; /* * Rather than using a top and bottom section, this component is te entire thing. * It leave space for the safe area insets and provides basic padding @@ -85,75 +60,38 @@ const FullSection: React.FC = ({ ...props }: FullSectionProps) => { const { top, bottom } = useSafeAreaInsets(); + return ( - {children} - + ); }; const BottomSection: React.FC = ({ children, - style, ...props }) => { const { bottom: safeAreaBottom } = useSafeAreaInsets(); - const incomingBottom = props.paddingBottom ?? 0; - const minBottom = safeAreaBottom + extraYPadding; - const totalBottom = - typeof incomingBottom === 'number' ? minBottom + incomingBottom : minBottom; - - let panelHeight: number | 'auto' = 'auto'; - // set bottom section height to 38% of screen height - // and wrap children in a scroll view if larger text is enabled - if (isLargerTextEnabled) { - const windowHeight = Dimensions.get('window').height; - panelHeight = windowHeight * 0.38; - children = ( - - {children} - - ); - } return ( - {children} - + ); }; /** - * This component is a layout that has a top and bottom section. Bottom section - * automatically expands to as much space as it needs while the top section - * takes up the remaining space. - * - * Usage: - * - * import { ExpandableBottomLayout } from '../components/ExpandableBottomLayout'; - * - * - * - * <...top section content...> - * - * - * <...bottom section content...> - * - * + * This component is a wrapper around the ExpandableBottomLayout component from the mobile SDK + * pacakge. It handles the safe area insets and system bars. */ export const ExpandableBottomLayout = { Layout, @@ -161,32 +99,3 @@ export const ExpandableBottomLayout = { FullSection, BottomSection, }; - -const styles = StyleSheet.create({ - roundTop: { - marginTop: 12, - overflow: 'hidden', - borderTopRightRadius: 30, - borderTopLeftRadius: 30, - }, - layout: { - height: '100%', - flexDirection: 'column', - }, - topSection: { - alignSelf: 'stretch', - flexGrow: 1, - flexShrink: Platform.select({ web: 0, default: 1 }), - alignItems: 'center', - justifyContent: 'center', - backgroundColor: black, - overflow: 'hidden', - padding: 20, - }, - bottomSection: { - backgroundColor: white, - paddingTop: 30, - paddingLeft: 20, - paddingRight: 20, - }, -}); diff --git a/app/src/providers/selfClientProvider.tsx b/app/src/providers/selfClientProvider.tsx index 47b100c7f..1f5c9592e 100644 --- a/app/src/providers/selfClientProvider.tsx +++ b/app/src/providers/selfClientProvider.tsx @@ -6,17 +6,15 @@ import type { PropsWithChildren } from 'react'; import { useMemo } from 'react'; import { Platform } from 'react-native'; -import type { - Adapters, - TrackEventParams, - WsConn, -} from '@selfxyz/mobile-sdk-alpha'; import { + type Adapters, createListenersMap, reactNativeScannerAdapter, SdkEvents, SelfClientProvider as SDKSelfClientProvider, + type TrackEventParams, webNFCScannerShim, + type WsConn, } from '@selfxyz/mobile-sdk-alpha'; import type { RootStackParamList } from '@/navigation'; diff --git a/app/src/screens/app/SplashScreen.tsx b/app/src/screens/app/SplashScreen.tsx index 530031a87..dc6d4d060 100644 --- a/app/src/screens/app/SplashScreen.tsx +++ b/app/src/screens/app/SplashScreen.tsx @@ -8,12 +8,12 @@ import { useNavigation } from '@react-navigation/native'; import type { NativeStackNavigationProp } from '@react-navigation/native-stack'; import { + DelayedLottieView, hasAnyValidRegisteredDocument, useSelfClient, } from '@selfxyz/mobile-sdk-alpha'; import splashAnimation from '@/assets/animations/splash.json'; -import { DelayedLottieView } from '@/components/DelayedLottieView'; import type { RootStackParamList } from '@/navigation'; import { migrateToSecureKeychain, useAuth } from '@/providers/authProvider'; import { diff --git a/app/src/screens/documents/scanning/DocumentCameraScreen.tsx b/app/src/screens/documents/scanning/DocumentCameraScreen.tsx index 0bf74f621..04ebfd57a 100644 --- a/app/src/screens/documents/scanning/DocumentCameraScreen.tsx +++ b/app/src/screens/documents/scanning/DocumentCameraScreen.tsx @@ -8,6 +8,7 @@ import { View, XStack, YStack } from 'tamagui'; import { useIsFocused } from '@react-navigation/native'; import { + DelayedLottieView, hasAnyValidRegisteredDocument, useSelfClient, } from '@selfxyz/mobile-sdk-alpha'; @@ -24,7 +25,6 @@ import { } from '@selfxyz/mobile-sdk-alpha/onboarding/read-mrz'; import passportScanAnimation from '@/assets/animations/passport_scan.json'; -import { DelayedLottieView } from '@/components/DelayedLottieView'; import { PassportCamera } from '@/components/native/PassportCamera'; import useHapticNavigation from '@/hooks/useHapticNavigation'; import Scan from '@/images/icons/passport_camera_scan.svg'; diff --git a/app/src/screens/documents/selection/ConfirmBelongingScreen.tsx b/app/src/screens/documents/selection/ConfirmBelongingScreen.tsx index 97d4b3eca..92dcdee8c 100644 --- a/app/src/screens/documents/selection/ConfirmBelongingScreen.tsx +++ b/app/src/screens/documents/selection/ConfirmBelongingScreen.tsx @@ -7,7 +7,11 @@ import type { StaticScreenProps } from '@react-navigation/native'; import { usePreventRemove } from '@react-navigation/native'; import type { DocumentCategory } from '@selfxyz/common/utils/types'; -import { loadSelectedDocument, useSelfClient } from '@selfxyz/mobile-sdk-alpha'; +import { + DelayedLottieView, + loadSelectedDocument, + useSelfClient, +} from '@selfxyz/mobile-sdk-alpha'; import { Description, PrimaryButton, @@ -20,7 +24,6 @@ import { import { getPreRegistrationDescription } from '@selfxyz/mobile-sdk-alpha/onboarding/confirm-identification'; import successAnimation from '@/assets/animations/loading/success.json'; -import { DelayedLottieView } from '@/components/DelayedLottieView'; import useHapticNavigation from '@/hooks/useHapticNavigation'; import { ExpandableBottomLayout } from '@/layouts/ExpandableBottomLayout'; import { styles } from '@/screens/verification/ProofRequestStatusScreen'; diff --git a/app/src/screens/onboarding/AccountVerifiedSuccessScreen.tsx b/app/src/screens/onboarding/AccountVerifiedSuccessScreen.tsx index 1469839d1..ec1627902 100644 --- a/app/src/screens/onboarding/AccountVerifiedSuccessScreen.tsx +++ b/app/src/screens/onboarding/AccountVerifiedSuccessScreen.tsx @@ -7,6 +7,7 @@ import { YStack } from 'tamagui'; import { useNavigation } from '@react-navigation/native'; import type { NativeStackNavigationProp } from '@react-navigation/native-stack'; +import { DelayedLottieView } from '@selfxyz/mobile-sdk-alpha'; import { Description, PrimaryButton, @@ -15,7 +16,6 @@ import { import { BackupEvents } from '@selfxyz/mobile-sdk-alpha/constants/analytics'; import proofSuccessAnimation from '@/assets/animations/proof_success.json'; -import { DelayedLottieView } from '@/components/DelayedLottieView'; import { ExpandableBottomLayout } from '@/layouts/ExpandableBottomLayout'; import type { RootStackParamList } from '@/navigation'; import { styles } from '@/screens/verification/ProofRequestStatusScreen'; diff --git a/app/src/screens/onboarding/DisclaimerScreen.tsx b/app/src/screens/onboarding/DisclaimerScreen.tsx index 28a68253e..5f8ceb562 100644 --- a/app/src/screens/onboarding/DisclaimerScreen.tsx +++ b/app/src/screens/onboarding/DisclaimerScreen.tsx @@ -8,6 +8,7 @@ import { YStack } from 'tamagui'; import { useNavigation } from '@react-navigation/native'; import type { NativeStackNavigationProp } from '@react-navigation/native-stack'; +import { DelayedLottieView } from '@selfxyz/mobile-sdk-alpha'; import { Caution, PrimaryButton, @@ -16,7 +17,6 @@ import { import { AppEvents } from '@selfxyz/mobile-sdk-alpha/constants/analytics'; import warningAnimation from '@/assets/animations/warning.json'; -import { DelayedLottieView } from '@/components/DelayedLottieView'; import { ExpandableBottomLayout } from '@/layouts/ExpandableBottomLayout'; import type { RootStackParamList } from '@/navigation'; import { useSettingStore } from '@/stores/settingStore'; diff --git a/packages/mobile-sdk-alpha/package.json b/packages/mobile-sdk-alpha/package.json index 478fc71fc..b97185255 100644 --- a/packages/mobile-sdk-alpha/package.json +++ b/packages/mobile-sdk-alpha/package.json @@ -137,6 +137,7 @@ "eslint-plugin-simple-import-sort": "^12.1.1", "eslint-plugin-sort-exports": "^0.9.1", "jsdom": "^25.0.1", + "lottie-react-native": "7.2.2", "prettier": "^3.5.3", "react": "^18.3.1", "react-dom": "^18.3.1", @@ -149,6 +150,7 @@ "vitest": "^2.1.8" }, "peerDependencies": { + "lottie-react-native": "7.2.2", "react": "^18.3.1", "react-native": "0.76.9", "react-native-haptic-feedback": "*", diff --git a/packages/mobile-sdk-alpha/src/animations/passport_scan.json b/packages/mobile-sdk-alpha/src/animations/passport_scan.json new file mode 100644 index 000000000..1d5fd4871 --- /dev/null +++ b/packages/mobile-sdk-alpha/src/animations/passport_scan.json @@ -0,0 +1,2366 @@ +{ + "assets": [ + { + "h": 252, + "id": "0", + "p": "", + "u": "", + "w": 373, + "e": 1 + }, + { + "id": "8", + "layers": [ + { "ind": 7, "ty": 2, "parent": 6, "ks": {}, "ip": 0, "op": 121, "st": 0, "refId": "0" }, + { + "ind": 6, + "ty": 3, + "ks": { "p": { "a": 0, "k": [-0.029, 0] }, "s": { "a": 0, "k": [94.84, 94.84] } }, + "ip": 0, + "op": 121, + "st": 0 + } + ] + }, + { + "id": "12", + "layers": [ + { + "ind": 10, + "ty": 0, + "ks": { "p": { "a": 0, "k": [20, 20] } }, + "w": 353.7, + "h": 239, + "ip": 0, + "op": 121, + "st": 0, + "refId": "8" + }, + { + "ind": 11, + "ty": 4, + "ks": {}, + "ip": 0, + "op": 121, + "st": 0, + "shapes": [ + { + "ty": "rc", + "p": { "a": 0, "k": [196.85, 139.5] }, + "r": { "a": 0, "k": 0 }, + "s": { "a": 0, "k": [393.7, 279] } + }, + { "ty": "fl", "c": { "a": 0, "k": [0, 0, 0, 0] }, "o": { "a": 0, "k": 0 } } + ] + } + ] + }, + { + "id": "15", + "layers": [ + { + "ind": 14, + "ty": 0, + "parent": 5, + "ks": { "p": { "a": 0, "k": [-20, -20] } }, + "ef": [ + { + "ty": 29, + "ef": [ + { + "ty": 0, + "nm": "", + "v": { + "a": 1, + "k": [ + { "t": 0, "s": [33.33], "h": 1 }, + { "t": 24.6, "s": [33.33], "i": { "x": 0.174, "y": 0.759 }, "o": { "x": 0.011, "y": 0.227 } }, + { "t": 57, "s": [3.67], "i": { "x": 0.722, "y": 1 }, "o": { "x": 0.35, "y": 0.828 } }, + { "t": 89.4, "s": [0], "h": 1 }, + { "t": 120, "s": [0], "h": 1 } + ] + } + }, + { "ty": 7, "nm": "", "v": { "a": 0, "k": 1 } }, + { "ty": 7, "nm": "", "v": { "a": 0, "k": 0 } } + ] + } + ], + "w": 393.7, + "h": 279, + "ip": 0, + "op": 121, + "st": 0, + "refId": "12" + }, + { "ind": 5, "ty": 3, "ks": { "p": { "a": 0, "k": [21, 21] } }, "ip": 0, "op": 121, "st": 0 } + ] + }, + { + "id": "19", + "layers": [ + { + "ind": 17, + "ty": 0, + "ks": { + "a": { "a": 0, "k": [21, 21] }, + "o": { + "a": 1, + "k": [ + { "t": 0, "s": [0], "h": 1 }, + { "t": 24.6, "s": [0], "i": { "x": 0, "y": 1 }, "o": { "x": 0, "y": 0 } }, + { "t": 89.4, "s": [100], "h": 1 }, + { "t": 120, "s": [100], "h": 1 } + ] + }, + "p": { "a": 0, "k": [41, 41] } + }, + "w": 395, + "h": 280, + "ip": 0, + "op": 121, + "st": 0, + "refId": "15" + }, + { + "ind": 18, + "ty": 4, + "ks": {}, + "ip": 0, + "op": 121, + "st": 0, + "shapes": [ + { + "ty": "rc", + "p": { "a": 0, "k": [227.85, 170.5] }, + "r": { "a": 0, "k": 0 }, + "s": { "a": 0, "k": [455.7, 341] } + }, + { "ty": "fl", "c": { "a": 0, "k": [0, 0, 0, 0] }, "o": { "a": 0, "k": 0 } } + ] + } + ] + }, + { + "id": "22", + "layers": [ + { + "ind": 21, + "ty": 0, + "parent": 4, + "ks": { "p": { "a": 0, "k": [-41, -41] } }, + "ef": [ + { + "ty": 29, + "ef": [ + { + "ty": 0, + "nm": "", + "v": { + "a": 1, + "k": [ + { "t": 0, "s": [0], "h": 1 }, + { "t": 72, "s": [0], "i": { "x": 0.65, "y": 0.172 }, "o": { "x": 0.278, "y": 0 } }, + { "t": 96, "s": [3.67], "i": { "x": 0.989, "y": 0.773 }, "o": { "x": 0.826, "y": 0.241 } }, + { "t": 120, "s": [33.33], "h": 1 } + ] + } + }, + { "ty": 7, "nm": "", "v": { "a": 0, "k": 1 } }, + { "ty": 7, "nm": "", "v": { "a": 0, "k": 0 } } + ] + } + ], + "w": 455.7, + "h": 341, + "ip": 0, + "op": 121, + "st": 0, + "refId": "19" + }, + { "ind": 4, "ty": 3, "ks": { "p": { "a": 0, "k": [41, 41] } }, "ip": 0, "op": 121, "st": 0 } + ] + }, + { + "id": "28", + "layers": [ + { + "ind": 27, + "ty": 4, + "ks": {}, + "ip": 0, + "op": 121, + "st": 0, + "shapes": [ + { + "ty": "rc", + "p": { "a": 0, "k": [176.5, 116] }, + "r": { "a": 0, "k": 0 }, + "s": { "a": 0, "k": [353, 232] } + }, + { "ty": "fl", "c": { "a": 0, "k": [0, 0, 0, 0] }, "o": { "a": 0, "k": 0 } } + ] + }, + { + "ind": 0, + "ty": 4, + "ks": { "s": { "a": 0, "k": [133.33, 133.33] } }, + "ip": 0, + "op": 121, + "st": 0, + "shapes": [ + { + "ty": "gr", + "it": [ + { + "ty": "gr", + "it": [ + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, -2.21], + [0, 0], + [10.49, 0], + [0, 0], + [0, 10.49], + [0, 0], + [-2.21, 0] + ], + "o": [ + [0, 0], + [2.21, 0], + [0, 0], + [0, 10.49], + [0, 0], + [-10.49, 0], + [0, 0], + [0, -2.21], + [0, 0] + ], + "v": [ + [5, 1], + [348, 1], + [352, 5], + [352, 212], + [333, 231], + [20, 231], + [1, 212], + [1, 5], + [5, 1] + ] + } + } + }, + { + "ty": "st", + "c": { "a": 0, "k": [1, 1, 1, 1] }, + "lc": 1, + "lj": 1, + "ml": 4, + "o": { "a": 0, "k": 100 }, + "w": { "a": 0, "k": 1 } + }, + { "ty": "tr", "o": { "a": 0, "k": 100 }, "s": { "a": 0, "k": [75, 75] } } + ] + }, + { "ty": "tr", "o": { "a": 0, "k": 100 } } + ] + } + ] + } + ] + }, + { + "id": "59", + "layers": [ + { + "ind": 58, + "ty": 4, + "parent": 57, + "ks": {}, + "ip": 0, + "op": 121, + "st": 0, + "shapes": [ + { + "ty": "rc", + "p": { + "a": 1, + "k": [ + { "t": 0, "s": [165.5, 109], "i": { "x": [1, 1], "y": [1, 1] }, "o": { "x": [0, 0], "y": [0, 0] } }, + { + "t": 60, + "s": [165.5, 109], + "i": { "x": [0, 1], "y": [1, 1] }, + "o": { "x": [0.5, 0], "y": [0, 0] } + }, + { "t": 120, "s": [0, 109], "h": 1 } + ] + }, + "r": { "a": 0, "k": 0 }, + "s": { + "a": 1, + "k": [ + { "t": 0, "s": [331, 218], "i": { "x": [1, 1], "y": [1, 1] }, "o": { "x": [0, 0], "y": [0, 0] } }, + { "t": 60, "s": [331, 218], "i": { "x": [0, 1], "y": [1, 1] }, "o": { "x": [0.5, 0], "y": [0, 0] } }, + { "t": 120, "s": [0, 218], "h": 1 } + ] + } + }, + { "ty": "fl", "c": { "a": 0, "k": [0, 0, 0] }, "o": { "a": 0, "k": 100 } } + ] + }, + { "ind": 57, "ty": 3, "parent": 56, "ks": {}, "ip": 0, "op": 121, "st": 0 }, + { "ind": 56, "ty": 3, "ks": { "p": { "a": 0, "k": [166, 0] } }, "ip": 0, "op": 121, "st": 0 } + ] + }, + { + "id": "63", + "layers": [ + { + "ind": 61, + "ty": 0, + "parent": 55, + "ks": { "a": { "a": 0, "k": [166, 0] } }, + "w": 497, + "h": 218, + "ip": 0, + "op": 121, + "st": 0, + "refId": "59" + }, + { + "ind": 62, + "ty": 4, + "parent": 55, + "ks": { + "o": { + "a": 1, + "k": [ + { "t": 0, "s": [100], "h": 1 }, + { "t": 60, "s": [100], "h": 1 }, + { "t": 60, "s": [0], "h": 1 }, + { "t": 120, "s": [0], "h": 1 }, + { "t": 120, "s": [100], "h": 1 } + ] + } + }, + "ip": 0, + "op": 121, + "st": 0, + "shapes": [ + { + "ty": "rc", + "p": { "a": 0, "k": [176.5, 116] }, + "r": { "a": 0, "k": 0 }, + "s": { "a": 0, "k": [353, 232] } + }, + { "ty": "fl", "c": { "a": 0, "k": [0, 0, 0] }, "o": { "a": 0, "k": 100 } } + ] + }, + { "ind": 55, "ty": 3, "ks": { "p": { "a": 0, "k": [166, 0] } }, "ip": 0, "op": 121, "st": 0 } + ] + }, + { + "id": "45", + "layers": [ + { + "ind": 44, + "ty": 4, + "parent": 43, + "ks": {}, + "ip": 0, + "op": 121, + "st": 0, + "shapes": [ + { + "ty": "rc", + "p": { + "a": 1, + "k": [ + { "t": 0, "s": [0, 109], "i": { "x": [0, 1], "y": [1, 1] }, "o": { "x": [0.5, 0], "y": [0, 0] } }, + { "t": 60, "s": [165.5, 109], "i": { "x": [1, 1], "y": [1, 1] }, "o": { "x": [0, 0], "y": [0, 0] } }, + { "t": 120, "s": [165.5, 109], "h": 1 } + ] + }, + "r": { "a": 0, "k": 0 }, + "s": { + "a": 1, + "k": [ + { "t": 0, "s": [0, 218], "i": { "x": [0, 1], "y": [1, 1] }, "o": { "x": [0.5, 0], "y": [0, 0] } }, + { "t": 60, "s": [331, 218], "i": { "x": [1, 1], "y": [1, 1] }, "o": { "x": [0, 0], "y": [0, 0] } }, + { "t": 120, "s": [331, 218], "h": 1 } + ] + } + }, + { "ty": "fl", "c": { "a": 0, "k": [0, 0, 0] }, "o": { "a": 0, "k": 100 } } + ] + }, + { + "ind": 43, + "ty": 3, + "parent": 42, + "ks": { + "p": { + "a": 1, + "k": [ + { "t": 0, "s": [331, 0], "i": { "x": [0, 1], "y": [1, 1] }, "o": { "x": [0.5, 0], "y": [0, 0] } }, + { "t": 60, "s": [0, 0], "i": { "x": [1, 1], "y": [1, 1] }, "o": { "x": [0, 0], "y": [0, 0] } }, + { "t": 120, "s": [0, 0], "h": 1 } + ] + } + }, + "ip": 0, + "op": 121, + "st": 0 + }, + { "ind": 42, "ty": 3, "ks": { "p": { "a": 0, "k": [166, 0] } }, "ip": 0, "op": 121, "st": 0 } + ] + }, + { + "id": "49", + "layers": [ + { + "ind": 47, + "ty": 0, + "parent": 41, + "ks": { "a": { "a": 0, "k": [166, 0] } }, + "w": 828, + "h": 218, + "ip": 0, + "op": 121, + "st": 0, + "refId": "45" + }, + { + "ind": 48, + "ty": 4, + "parent": 41, + "ks": { + "o": { + "a": 1, + "k": [ + { "t": 0, "s": [0], "h": 1 }, + { "t": 60, "s": [0], "h": 1 }, + { "t": 60, "s": [100], "h": 1 }, + { "t": 120, "s": [100], "h": 1 } + ] + } + }, + "ip": 0, + "op": 121, + "st": 0, + "shapes": [ + { + "ty": "rc", + "p": { "a": 0, "k": [176.5, 116] }, + "r": { "a": 0, "k": 0 }, + "s": { "a": 0, "k": [353, 232] } + }, + { "ty": "fl", "c": { "a": 0, "k": [0, 0, 0] }, "o": { "a": 0, "k": 100 } } + ] + }, + { "ind": 41, "ty": 3, "ks": { "p": { "a": 0, "k": [166, 0] } }, "ip": 0, "op": 121, "st": 0 } + ] + }, + { + "id": "35", + "layers": [ + { + "ind": 34, + "ty": 4, + "ks": {}, + "ip": 0, + "op": 121, + "st": 0, + "shapes": [ + { + "ty": "rc", + "p": { "a": 0, "k": [176.5, 116] }, + "r": { "a": 0, "k": 0 }, + "s": { "a": 0, "k": [353, 232] } + }, + { "ty": "fl", "c": { "a": 0, "k": [0, 0, 0, 0] }, "o": { "a": 0, "k": 0 } } + ] + }, + { + "ind": 0, + "ty": 4, + "ks": { "s": { "a": 0, "k": [133.33, 133.33] } }, + "ip": 0, + "op": 121, + "st": 0, + "shapes": [ + { + "ty": "gr", + "it": [ + { + "ty": "gr", + "it": [ + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [18.52, 151.45], + [18.52, 149.89], + [24.98, 146.28], + [24.98, 147.87], + [20.01, 150.61], + [20.01, 150.73], + [24.98, 153.46], + [24.98, 155.07] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [29.02, 151.45], + [29.02, 149.89], + [35.47, 146.28], + [35.47, 147.87], + [30.5, 150.61], + [30.5, 150.73], + [35.47, 153.46], + [35.47, 155.07] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [39.51, 151.45], + [39.51, 149.89], + [45.96, 146.28], + [45.96, 147.87], + [41, 150.61], + [41, 150.73], + [45.96, 153.46], + [45.96, 155.07] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [50.01, 151.45], + [50.01, 149.89], + [56.46, 146.28], + [56.46, 147.87], + [51.49, 150.61], + [51.49, 150.73], + [56.46, 153.46], + [56.46, 155.07] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [60.5, 151.45], + [60.5, 149.89], + [66.95, 146.28], + [66.95, 147.87], + [61.99, 150.61], + [61.99, 150.73], + [66.95, 153.46], + [66.95, 155.07] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [71, 151.45], + [71, 149.89], + [77.45, 146.28], + [77.45, 147.87], + [72.48, 150.61], + [72.48, 150.73], + [77.45, 153.46], + [77.45, 155.07] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [81.49, 151.45], + [81.49, 149.89], + [87.94, 146.28], + [87.94, 147.87], + [82.98, 150.61], + [82.98, 150.73], + [87.94, 153.46], + [87.94, 155.07] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [91.98, 151.45], + [91.98, 149.89], + [98.43, 146.28], + [98.43, 147.87], + [93.47, 150.61], + [93.47, 150.73], + [98.43, 153.46], + [98.43, 155.07] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [102.48, 151.45], + [102.48, 149.89], + [108.93, 146.28], + [108.93, 147.87], + [103.96, 150.61], + [103.96, 150.73], + [108.93, 153.46], + [108.93, 155.07] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [112.97, 151.45], + [112.97, 149.89], + [119.42, 146.28], + [119.42, 147.87], + [114.46, 150.61], + [114.46, 150.73], + [119.42, 153.46], + [119.42, 155.07] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [123.46, 151.45], + [123.46, 149.89], + [129.92, 146.28], + [129.92, 147.87], + [124.95, 150.61], + [124.95, 150.73], + [129.92, 153.46], + [129.92, 155.07] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [133.96, 151.45], + [133.96, 149.89], + [140.41, 146.28], + [140.41, 147.87], + [135.45, 150.61], + [135.45, 150.73], + [140.41, 153.46], + [140.41, 155.07] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [144.45, 151.45], + [144.45, 149.89], + [150.91, 146.28], + [150.91, 147.87], + [145.94, 150.61], + [145.94, 150.73], + [150.91, 153.46], + [150.91, 155.07] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [154.95, 151.45], + [154.95, 149.89], + [161.4, 146.28], + [161.4, 147.87], + [156.43, 150.61], + [156.43, 150.73], + [161.4, 153.46], + [161.4, 155.07] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [165.44, 151.45], + [165.44, 149.89], + [171.89, 146.28], + [171.89, 147.87], + [166.93, 150.61], + [166.93, 150.73], + [171.89, 153.46], + [171.89, 155.07] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [175.94, 151.45], + [175.94, 149.89], + [182.39, 146.28], + [182.39, 147.87], + [177.42, 150.61], + [177.42, 150.73], + [182.39, 153.46], + [182.39, 155.07] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [186.43, 151.45], + [186.43, 149.89], + [192.88, 146.28], + [192.88, 147.87], + [187.92, 150.61], + [187.92, 150.73], + [192.88, 153.46], + [192.88, 155.07] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [196.93, 151.45], + [196.93, 149.89], + [203.38, 146.28], + [203.38, 147.87], + [198.41, 150.61], + [198.41, 150.73], + [203.38, 153.46], + [203.38, 155.07] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [207.42, 151.45], + [207.42, 149.89], + [213.87, 146.28], + [213.87, 147.87], + [208.9, 150.61], + [208.9, 150.73], + [213.87, 153.46], + [213.87, 155.07] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [217.91, 151.45], + [217.91, 149.89], + [224.36, 146.28], + [224.36, 147.87], + [219.4, 150.61], + [219.4, 150.73], + [224.36, 153.46], + [224.36, 155.07] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [228.41, 151.45], + [228.41, 149.89], + [234.86, 146.28], + [234.86, 147.87], + [229.89, 150.61], + [229.89, 150.73], + [234.86, 153.46], + [234.86, 155.07] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [238.9, 151.45], + [238.9, 149.89], + [245.35, 146.28], + [245.35, 147.87], + [240.39, 150.61], + [240.39, 150.73], + [245.35, 153.46], + [245.35, 155.07] + ] + } + } + }, + { "ty": "fl", "c": { "a": 0, "k": [0, 1, 0.71, 1] }, "o": { "a": 0, "k": 100 } }, + { "ty": "tr", "o": { "a": 0, "k": 100 } } + ] + }, + { + "ty": "gr", + "it": [ + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [18.52, 132.33], + [18.52, 130.77], + [24.98, 127.16], + [24.98, 128.75], + [20.01, 131.49], + [20.01, 131.61], + [24.98, 134.34], + [24.98, 135.95] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [29.02, 132.33], + [29.02, 130.77], + [35.47, 127.16], + [35.47, 128.75], + [30.5, 131.49], + [30.5, 131.61], + [35.47, 134.34], + [35.47, 135.95] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [39.51, 132.33], + [39.51, 130.77], + [45.96, 127.16], + [45.96, 128.75], + [41, 131.49], + [41, 131.61], + [45.96, 134.34], + [45.96, 135.95] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [50.01, 132.33], + [50.01, 130.77], + [56.46, 127.16], + [56.46, 128.75], + [51.49, 131.49], + [51.49, 131.61], + [56.46, 134.34], + [56.46, 135.95] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [60.5, 132.33], + [60.5, 130.77], + [66.95, 127.16], + [66.95, 128.75], + [61.99, 131.49], + [61.99, 131.61], + [66.95, 134.34], + [66.95, 135.95] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [71, 132.33], + [71, 130.77], + [77.45, 127.16], + [77.45, 128.75], + [72.48, 131.49], + [72.48, 131.61], + [77.45, 134.34], + [77.45, 135.95] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [81.49, 132.33], + [81.49, 130.77], + [87.94, 127.16], + [87.94, 128.75], + [82.98, 131.49], + [82.98, 131.61], + [87.94, 134.34], + [87.94, 135.95] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [91.98, 132.33], + [91.98, 130.77], + [98.43, 127.16], + [98.43, 128.75], + [93.47, 131.49], + [93.47, 131.61], + [98.43, 134.34], + [98.43, 135.95] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [102.48, 132.33], + [102.48, 130.77], + [108.93, 127.16], + [108.93, 128.75], + [103.96, 131.49], + [103.96, 131.61], + [108.93, 134.34], + [108.93, 135.95] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [112.97, 132.33], + [112.97, 130.77], + [119.42, 127.16], + [119.42, 128.75], + [114.46, 131.49], + [114.46, 131.61], + [119.42, 134.34], + [119.42, 135.95] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [123.46, 132.33], + [123.46, 130.77], + [129.92, 127.16], + [129.92, 128.75], + [124.95, 131.49], + [124.95, 131.61], + [129.92, 134.34], + [129.92, 135.95] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [133.96, 132.33], + [133.96, 130.77], + [140.41, 127.16], + [140.41, 128.75], + [135.45, 131.49], + [135.45, 131.61], + [140.41, 134.34], + [140.41, 135.95] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [144.45, 132.33], + [144.45, 130.77], + [150.91, 127.16], + [150.91, 128.75], + [145.94, 131.49], + [145.94, 131.61], + [150.91, 134.34], + [150.91, 135.95] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [154.95, 132.33], + [154.95, 130.77], + [161.4, 127.16], + [161.4, 128.75], + [156.43, 131.49], + [156.43, 131.61], + [161.4, 134.34], + [161.4, 135.95] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [165.44, 132.33], + [165.44, 130.77], + [171.89, 127.16], + [171.89, 128.75], + [166.93, 131.49], + [166.93, 131.61], + [171.89, 134.34], + [171.89, 135.95] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [175.94, 132.33], + [175.94, 130.77], + [182.39, 127.16], + [182.39, 128.75], + [177.42, 131.49], + [177.42, 131.61], + [182.39, 134.34], + [182.39, 135.95] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [186.43, 132.33], + [186.43, 130.77], + [192.88, 127.16], + [192.88, 128.75], + [187.92, 131.49], + [187.92, 131.61], + [192.88, 134.34], + [192.88, 135.95] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [196.93, 132.33], + [196.93, 130.77], + [203.38, 127.16], + [203.38, 128.75], + [198.41, 131.49], + [198.41, 131.61], + [203.38, 134.34], + [203.38, 135.95] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [207.42, 132.33], + [207.42, 130.77], + [213.87, 127.16], + [213.87, 128.75], + [208.9, 131.49], + [208.9, 131.61], + [213.87, 134.34], + [213.87, 135.95] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [217.91, 132.33], + [217.91, 130.77], + [224.36, 127.16], + [224.36, 128.75], + [219.4, 131.49], + [219.4, 131.61], + [224.36, 134.34], + [224.36, 135.95] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [228.41, 132.33], + [228.41, 130.77], + [234.86, 127.16], + [234.86, 128.75], + [229.89, 131.49], + [229.89, 131.61], + [234.86, 134.34], + [234.86, 135.95] + ] + } + } + }, + { + "ty": "sh", + "ks": { + "a": 0, + "k": { + "c": true, + "i": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "o": [ + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ], + "v": [ + [238.9, 132.33], + [238.9, 130.77], + [245.35, 127.16], + [245.35, 128.75], + [240.39, 131.49], + [240.39, 131.61], + [245.35, 134.34], + [245.35, 135.95] + ] + } + } + }, + { "ty": "fl", "c": { "a": 0, "k": [0, 1, 0.71, 1] }, "o": { "a": 0, "k": 100 } }, + { "ty": "tr", "o": { "a": 0, "k": 100 } } + ] + }, + { "ty": "tr", "o": { "a": 0, "k": 100 } } + ] + } + ] + } + ] + }, + { + "id": "38", + "layers": [ + { "ind": 37, "ty": 0, "parent": 33, "ks": {}, "w": 353, "h": 232, "ip": 0, "op": 121, "st": 0, "refId": "35" }, + { + "ind": 33, + "ty": 3, + "parent": 32, + "ks": { "s": { "a": 0, "k": [93.77, 93.97] } }, + "ip": 0, + "op": 121, + "st": 0 + }, + { "ind": 32, "ty": 3, "ks": {}, "ip": 0, "op": 121, "st": 0 } + ] + }, + { + "id": "52", + "layers": [ + { + "ind": 51, + "ty": 0, + "td": 1, + "parent": 31, + "ks": { "a": { "a": 0, "k": [166, 0] } }, + "w": 994, + "h": 232, + "ip": 0, + "op": 121, + "st": 0, + "refId": "49" + }, + { + "ind": 40, + "ty": 0, + "tt": 1, + "parent": 31, + "ks": {}, + "w": 353, + "h": 232, + "ip": 0, + "op": 121, + "st": 0, + "refId": "38" + }, + { "ind": 31, "ty": 3, "ks": {}, "ip": 0, "op": 121, "st": 0 } + ] + }, + { + "id": "66", + "layers": [ + { + "ind": 65, + "ty": 0, + "td": 1, + "ks": { "a": { "a": 0, "k": [166, 0] } }, + "w": 663, + "h": 232, + "ip": 0, + "op": 121, + "st": 0, + "refId": "63" + }, + { "ind": 54, "ty": 0, "tt": 1, "ks": {}, "w": 353, "h": 232, "ip": 0, "op": 121, "st": 0, "refId": "52" } + ] + } + ], + "fr": 60, + "h": 398, + "ip": 0, + "layers": [ + { + "ind": 24, + "ty": 0, + "parent": 3, + "ks": { + "a": { "a": 0, "k": [41, 41] }, + "o": { + "a": 1, + "k": [ + { "t": 0, "s": [100], "h": 1 }, + { "t": 72, "s": [100], "i": { "x": 1, "y": 1 }, "o": { "x": 1, "y": 0 } }, + { "t": 120, "s": [0], "h": 1 } + ] + } + }, + "w": 456, + "h": 341, + "ip": 0, + "op": 121, + "st": 0, + "refId": "22" + }, + { "ind": 3, "ty": 3, "parent": 2, "ks": { "p": { "a": 0, "k": [20, 82] } }, "ip": 0, "op": 121, "st": 0 }, + { "ind": 30, "ty": 0, "parent": 26, "ks": {}, "w": 353, "h": 232, "ip": 0, "op": 121, "st": 0, "refId": "28" }, + { "ind": 26, "ty": 3, "parent": 25, "ks": { "s": { "a": 0, "k": [93.2, 93.1] } }, "ip": 0, "op": 121, "st": 0 }, + { "ind": 25, "ty": 3, "parent": 2, "ks": { "p": { "a": 0, "k": [32, 94] } }, "ip": 0, "op": 121, "st": 0 }, + { + "ind": 68, + "ty": 0, + "parent": 2, + "ks": { + "o": { + "a": 1, + "k": [ + { "t": 0, "s": [100], "h": 1 }, + { "t": 120, "s": [100], "h": 1 }, + { "t": 120, "s": [0], "h": 1 } + ] + }, + "p": { "a": 0, "k": [31, 93] } + }, + "w": 353, + "h": 232, + "ip": 0, + "op": 121, + "st": 0, + "refId": "66" + }, + { "ind": 2, "ty": 3, "parent": 1, "ks": {}, "ip": 0, "op": 121, "st": 0 }, + { "ind": 1, "ty": 3, "ks": {}, "ip": 0, "op": 121, "st": 0 } + ], + "meta": { "g": "https://jitter.video" }, + "op": 120, + "v": "5.7.4", + "w": 393 +} diff --git a/packages/mobile-sdk-alpha/src/browser.ts b/packages/mobile-sdk-alpha/src/browser.ts index e2482ed4b..e8b57181a 100644 --- a/packages/mobile-sdk-alpha/src/browser.ts +++ b/packages/mobile-sdk-alpha/src/browser.ts @@ -34,7 +34,18 @@ export type { PassportValidationCallbacks } from './validation/document'; export type { SDKEvent, SDKEventMap } from './types/events'; export type { SdkErrorCategory } from './errors'; +export { + type BottomSectionProps, + ExpandableBottomLayout, + type FullSectionProps, + type LayoutProps, + type TopSectionProps, +} from './layouts/ExpandableBottomLayout'; + +export { DelayedLottieView } from './components/DelayedLottieView.web'; + export { type ProvingStateType } from './proving/provingMachine'; + export { SCANNER_ERROR_CODES, notImplemented, sdkError } from './errors'; export { SdkEvents } from './types/events'; diff --git a/app/src/components/DelayedLottieView.tsx b/packages/mobile-sdk-alpha/src/components/DelayedLottieView.tsx similarity index 53% rename from app/src/components/DelayedLottieView.tsx rename to packages/mobile-sdk-alpha/src/components/DelayedLottieView.tsx index e05ffc737..e00e98197 100644 --- a/app/src/components/DelayedLottieView.tsx +++ b/packages/mobile-sdk-alpha/src/components/DelayedLottieView.tsx @@ -4,7 +4,8 @@ import type { LottieViewProps } from 'lottie-react-native'; import LottieView from 'lottie-react-native'; -import React, { forwardRef, useEffect, useRef } from 'react'; +import type React from 'react'; +import { forwardRef, useEffect, useRef } from 'react'; /** * Wrapper around LottieView that fixes iOS native module initialization timing. @@ -20,29 +21,30 @@ import React, { forwardRef, useEffect, useRef } from 'react'; * @example * */ -export const DelayedLottieView = forwardRef( - (props, forwardedRef) => { - const internalRef = useRef(null); - const ref = (forwardedRef as React.RefObject) || internalRef; - - useEffect(() => { - // Only auto-trigger for autoPlay animations - if (props.autoPlay) { - const timer = setTimeout(() => { - ref.current?.play(); - }, 100); - - return () => clearTimeout(timer); - } - }, [props.autoPlay, ref]); - - // For autoPlay animations, disable native autoPlay and control it ourselves - const modifiedProps = props.autoPlay - ? { ...props, autoPlay: false } - : props; - - return ; - }, -); +export const DelayedLottieView = forwardRef((props, forwardedRef) => { + // If LottieView is undefined (peer dependency not installed), return null + if (typeof LottieView === 'undefined') { + return null; + } + + const internalRef = useRef(null); + const ref = (forwardedRef as React.RefObject) || internalRef; + + useEffect(() => { + // Only auto-trigger for autoPlay animations + if (props.autoPlay) { + const timer = setTimeout(() => { + ref.current?.play(); + }, 100); + + return () => clearTimeout(timer); + } + }, [props.autoPlay, ref]); + + // For autoPlay animations, disable native autoPlay and control it ourselves + const modifiedProps = props.autoPlay ? { ...props, autoPlay: false } : props; + + return ; +}); DelayedLottieView.displayName = 'DelayedLottieView'; diff --git a/packages/mobile-sdk-alpha/src/components/DelayedLottieView.web.tsx b/packages/mobile-sdk-alpha/src/components/DelayedLottieView.web.tsx new file mode 100644 index 000000000..907521a0b --- /dev/null +++ b/packages/mobile-sdk-alpha/src/components/DelayedLottieView.web.tsx @@ -0,0 +1,10 @@ +// SPDX-FileCopyrightText: 2025 Social Connect Labs, Inc. +// SPDX-License-Identifier: BUSL-1.1 +// NOTE: Converts to Apache-2.0 on 2029-06-11 per LICENSE. + +/** + * DelayedLottieView for web placeholder component. + */ +export const DelayedLottieView = () => { + return
; +}; diff --git a/packages/mobile-sdk-alpha/src/components/MRZScannerView.tsx b/packages/mobile-sdk-alpha/src/components/MRZScannerView.tsx index 3abc5cc93..86076f114 100644 --- a/packages/mobile-sdk-alpha/src/components/MRZScannerView.tsx +++ b/packages/mobile-sdk-alpha/src/components/MRZScannerView.tsx @@ -4,7 +4,7 @@ import { useCallback, useRef } from 'react'; import type { DimensionValue, NativeSyntheticEvent, ViewProps, ViewStyle } from 'react-native'; -import { NativeModules, PixelRatio, Platform, requireNativeComponent, StyleSheet, View } from 'react-native'; +import { NativeModules, PixelRatio, Platform, requireNativeComponent, View } from 'react-native'; import { extractMRZInfo, formatDateToYYMMDD } from '../mrz'; import type { MRZInfo } from '../types/public'; @@ -99,7 +99,6 @@ export const MRZScannerView: React.FC = ({ ); const containerStyle = [ - styles.container, height !== undefined && { height }, width !== undefined && { width }, aspectRatio !== undefined && { aspectRatio }, @@ -112,8 +111,8 @@ export const MRZScannerView: React.FC = ({ = ({ isMounted={true} style={{ height: PixelRatio.getPixelSizeForLayoutSize(800), - width: PixelRatio.getPixelSizeForLayoutSize(800), + width: PixelRatio.getPixelSizeForLayoutSize(400), }} onError={handleError} onPassportRead={handleMRZDetected} @@ -139,18 +138,4 @@ export const MRZScannerView: React.FC = ({ } }; -// TODO Check this -const styles = StyleSheet.create({ - container: { - width: '100%', - minHeight: 200, - aspectRatio: 1, - }, - scanner: { - flex: 1, - width: '100%', - height: '100%', - }, -}); - export const SelfMRZScannerModule = NativeModules.SelfMRZScannerModule; diff --git a/packages/mobile-sdk-alpha/src/components/layout/View.tsx b/packages/mobile-sdk-alpha/src/components/layout/View.tsx index 4f37f881a..83f163bc4 100644 --- a/packages/mobile-sdk-alpha/src/components/layout/View.tsx +++ b/packages/mobile-sdk-alpha/src/components/layout/View.tsx @@ -198,6 +198,7 @@ export const View: React.FC = ({ flexGrow, flexShrink, width, + maxWidth, height, flexDirection, justifyContent, @@ -226,6 +227,7 @@ export const View: React.FC = ({ ...(flexGrow !== undefined && { flexGrow }), ...(flexShrink !== undefined && { flexShrink }), ...(width !== undefined && { width }), + ...(maxWidth !== undefined && { maxWidth }), ...(height !== undefined && { height }), ...(flexDirection && { flexDirection }), ...(justifyContent && { justifyContent }), diff --git a/packages/mobile-sdk-alpha/src/constants/layout.ts b/packages/mobile-sdk-alpha/src/constants/layout.ts new file mode 100644 index 000000000..9c15d3da1 --- /dev/null +++ b/packages/mobile-sdk-alpha/src/constants/layout.ts @@ -0,0 +1,5 @@ +// SPDX-FileCopyrightText: 2025 Social Connect Labs, Inc. +// SPDX-License-Identifier: BUSL-1.1 +// NOTE: Converts to Apache-2.0 on 2029-06-11 per LICENSE. + +export const extraYPadding = 15; diff --git a/packages/mobile-sdk-alpha/src/flows/onboarding/document-camera-screen.tsx b/packages/mobile-sdk-alpha/src/flows/onboarding/document-camera-screen.tsx new file mode 100644 index 000000000..e9ec1ee4e --- /dev/null +++ b/packages/mobile-sdk-alpha/src/flows/onboarding/document-camera-screen.tsx @@ -0,0 +1,119 @@ +// SPDX-FileCopyrightText: 2025 Social Connect Labs, Inc. +// SPDX-License-Identifier: BUSL-1.1 +// NOTE: Converts to Apache-2.0 on 2029-06-11 per LICENSE. + +import { useCallback, useRef } from 'react'; +import { StyleSheet } from 'react-native'; +import passportScanAnimation from 'src/animations/passport_scan.json'; +import { Additional, Description, SecondaryButton, Title, View, XStack, YStack } from 'src/components'; +import { DelayedLottieView } from 'src/components/DelayedLottieView'; +import { MRZScannerView } from 'src/components/MRZScannerView'; +import { PassportEvents } from 'src/constants/analytics'; +import { black, slate400, slate800, white } from 'src/constants/colors'; +import { useSelfClient } from 'src/context'; +import { mrzReadInstructions, useReadMRZ } from 'src/flows/onboarding/read-mrz'; +import type { SafeAreaInsets } from 'src/layouts/ExpandableBottomLayout'; +import { ExpandableBottomLayout } from 'src/layouts/ExpandableBottomLayout'; +import { SdkEvents } from 'src/types/events'; +import type { MRZInfo } from 'src/types/public'; + +import Scan from '../../../svgs/icons/passport_camera_scan.svg'; +import { dinot } from '../../constants/fonts'; + +type Props = { + onBack?: () => void; + onSuccess?: () => void; + safeAreaInsets?: SafeAreaInsets; +}; + +export const DocumentCameraScreen = ({ onBack, onSuccess, safeAreaInsets }: Props) => { + const scanStartTimeRef = useRef(Date.now()); + const selfClient = useSelfClient(); + const { onPassportRead } = useReadMRZ(scanStartTimeRef); + + const handleMRZDetected = useCallback( + (mrzData: MRZInfo) => { + onPassportRead(null, mrzData); + + onSuccess?.(); + }, + [onPassportRead], + ); + + const handleScannerError = useCallback( + (error: string) => { + selfClient.emit(SdkEvents.ERROR, new Error(`MRZ scanner error: ${error}`)); + }, + [selfClient], + ); + + return ( + + + + + + + + + Scan your ID + + + + + + Open to the photograph page + {mrzReadInstructions()} + + + + + SELF WILL NOT CAPTURE AN IMAGE OF YOUR PASSPORT. + + {})}> + Cancel + + + + + ); +}; + +const styles = StyleSheet.create({ + animation: { + position: 'absolute', + width: '100%', + height: '100%', + }, + subheader: { + color: slate800, + textAlign: 'left', + textAlignVertical: 'top', + }, + description: { + textAlign: 'left', + }, + disclaimer: { + fontFamily: dinot, + textAlign: 'center', + fontSize: 11, + color: slate400, + textTransform: 'uppercase', + width: '100%', + alignSelf: 'center', + letterSpacing: 0.44, + marginTop: 0, + marginBottom: 10, + }, +}); diff --git a/packages/mobile-sdk-alpha/src/index.ts b/packages/mobile-sdk-alpha/src/index.ts index 8ed35acd5..aa1c2279d 100644 --- a/packages/mobile-sdk-alpha/src/index.ts +++ b/packages/mobile-sdk-alpha/src/index.ts @@ -49,6 +49,17 @@ export type { SDKEvent, SDKEventMap } from './types/events'; export type { SdkErrorCategory } from './errors'; // Screen Components (React Native-based) export type { provingMachineCircuitType } from './proving/provingMachine'; + +export { + type BottomSectionProps, + ExpandableBottomLayout, + type FullSectionProps, + type LayoutProps, + type TopSectionProps, +} from './layouts/ExpandableBottomLayout'; + +export { DelayedLottieView } from './components/DelayedLottieView'; + export { InitError, LivenessError, @@ -59,10 +70,8 @@ export { notImplemented, sdkError, } from './errors'; -export { NFCScannerScreen } from './components/screens/NFCScannerScreen'; -// Context and Client -export { PassportCameraScreen } from './components/screens/PassportCameraScreen'; +export { NFCScannerScreen } from './components/screens/NFCScannerScreen'; export { type ProvingStateType } from './proving/provingMachine'; // Components diff --git a/packages/mobile-sdk-alpha/src/layouts/ExpandableBottomLayout.tsx b/packages/mobile-sdk-alpha/src/layouts/ExpandableBottomLayout.tsx new file mode 100644 index 000000000..baee68309 --- /dev/null +++ b/packages/mobile-sdk-alpha/src/layouts/ExpandableBottomLayout.tsx @@ -0,0 +1,175 @@ +// SPDX-FileCopyrightText: 2025 Social Connect Labs, Inc. +// SPDX-License-Identifier: BUSL-1.1 +// NOTE: Converts to Apache-2.0 on 2029-06-11 per LICENSE. + +import type React from 'react'; +import { Dimensions, PixelRatio, Platform, ScrollView, StyleSheet } from 'react-native'; + +import type { ViewProps } from '../components'; +import { View } from '../components'; +import { black, white } from '../constants/colors'; +import { extraYPadding } from '../constants/layout'; + +const SAFE_AREA_TOP_DEFAULT = 0; +const SAFE_AREA_BOTTOM_DEFAULT = 0; + +// Get the current font scale factor +const fontScale = PixelRatio.getFontScale(); +// fontScale > 1 means the user has increased text size in accessibility settings +const isLargerTextEnabled = fontScale > 1.3; + +export interface BottomSectionProps extends ViewProps { + children: React.ReactNode; + backgroundColor: string; + safeAreaBottom?: number; +} + +export type FullSectionProps = ViewProps & { + safeAreaTop?: number; + safeAreaBottom?: number; +}; + +export interface LayoutProps extends ViewProps { + children: React.ReactNode; + backgroundColor: string; + safeAreaTop?: number; + safeAreaBottom?: number; +} + +const Layout: React.FC = ({ children, backgroundColor }) => { + return ( + + {children} + + ); +}; + +const TopSection: React.FC = ({ children, backgroundColor, ...props }) => { + const { safeAreaTop = SAFE_AREA_TOP_DEFAULT } = props; + const { roundTop, ...restProps } = props; + + return ( + + {children} + + ); +}; + +/** + * This component is a layout that has a top and bottom section. Bottom section + * automatically expands to as much space as it needs while the top section + * takes up the remaining space. + * + * Usage: + * + * import { ExpandableBottomLayout } from '../components/ExpandableBottomLayout'; + * + * + * + * <...top section content...> + * + * + * <...bottom section content...> + * + * + */ +export type SafeAreaInsets = { + top: number; + bottom: number; +}; +/* + * Rather than using a top and bottom section, this component is te entire thing. + * It leave space for the safe area insets and provides basic padding + */ +const FullSection: React.FC = ({ children, backgroundColor, ...props }: FullSectionProps) => { + const { safeAreaTop = SAFE_AREA_TOP_DEFAULT, safeAreaBottom = SAFE_AREA_BOTTOM_DEFAULT } = props; + return ( + + {children} + + ); +}; + +const BottomSection: React.FC = ({ children, style, ...props }) => { + const incomingBottom = props.paddingBottom ?? 0; + const { safeAreaBottom = SAFE_AREA_BOTTOM_DEFAULT } = props; + const minBottom = safeAreaBottom + extraYPadding; + const totalBottom = typeof incomingBottom === 'number' ? minBottom + incomingBottom : minBottom; + + let panelHeight: number | 'auto' = 'auto'; + // set bottom section height to 38% of screen height + // and wrap children in a scroll view if larger text is enabled + if (isLargerTextEnabled) { + const windowHeight = Dimensions.get('window').height; + panelHeight = windowHeight * 0.38; + children = ( + + {children} + + ); + } + + return ( + + {children} + + ); +}; + +export interface TopSectionProps extends ViewProps { + children: React.ReactNode; + backgroundColor: string; + roundTop?: boolean; + safeAreaTop?: number; +} + +export const ExpandableBottomLayout = { + Layout, + TopSection, + FullSection, + BottomSection, +}; + +const styles = StyleSheet.create({ + roundTop: { + marginTop: 12, + overflow: 'hidden', + borderTopRightRadius: 30, + borderTopLeftRadius: 30, + }, + layout: { + height: '100%', + flexDirection: 'column', + }, + topSection: { + alignSelf: 'stretch', + flexGrow: 1, + flexShrink: Platform.select({ web: 0, default: 1 }), + alignItems: 'center', + justifyContent: 'center', + backgroundColor: black, + overflow: 'hidden', + padding: 20, + }, + bottomSection: { + backgroundColor: white, + paddingTop: 30, + paddingLeft: 20, + paddingRight: 20, + }, +}); diff --git a/packages/mobile-sdk-alpha/svgs/icons/passport_camera_scan.svg b/packages/mobile-sdk-alpha/svgs/icons/passport_camera_scan.svg new file mode 100644 index 000000000..cf47854f6 --- /dev/null +++ b/packages/mobile-sdk-alpha/svgs/icons/passport_camera_scan.svg @@ -0,0 +1,3 @@ + + + diff --git a/packages/mobile-sdk-alpha/tests/setup.ts b/packages/mobile-sdk-alpha/tests/setup.ts index 5c81ee4ba..5eeea990e 100644 --- a/packages/mobile-sdk-alpha/tests/setup.ts +++ b/packages/mobile-sdk-alpha/tests/setup.ts @@ -67,6 +67,12 @@ vi.mock('react-native', () => ({ Dimensions: { get: vi.fn(() => ({ width: 375, height: 667 })), }, + PixelRatio: { + get: vi.fn(() => 2), + getFontScale: vi.fn(() => 1), + getPixelSizeForLayoutSize: vi.fn((size: number) => size * 2), + roundToNearestPixel: vi.fn((size: number) => Math.round(size)), + }, StatusBar: { setBarStyle: vi.fn(), }, @@ -192,3 +198,56 @@ vi.mock('react-native-localize', () => ({ isRTL: false, }), })); + +// Mock react-native-svg +vi.mock('react-native-svg', () => ({ + default: 'svg', + Svg: 'svg', + Circle: 'circle', + Ellipse: 'ellipse', + G: 'g', + Text: 'text', + TSpan: 'tspan', + TextPath: 'textPath', + Path: 'path', + Polygon: 'polygon', + Polyline: 'polyline', + Line: 'line', + Rect: 'rect', + Use: 'use', + Image: 'image', + Symbol: 'symbol', + Defs: 'defs', + LinearGradient: 'linearGradient', + RadialGradient: 'radialGradient', + Stop: 'stop', + ClipPath: 'clipPath', + Pattern: 'pattern', + Mask: 'mask', +})); + +// Mock react-native-svg-circle-country-flags +vi.mock('react-native-svg-circle-country-flags', () => ({ + default: {}, +})); + +// Mock lottie-react-native +vi.mock('lottie-react-native', () => ({ + default: 'div', +})); + +// Mock react-native-haptic-feedback +vi.mock('react-native-haptic-feedback', () => ({ + default: { + trigger: vi.fn(), + }, + HapticFeedbackTypes: { + impactLight: 'impactLight', + impactMedium: 'impactMedium', + impactHeavy: 'impactHeavy', + notificationSuccess: 'notificationSuccess', + notificationWarning: 'notificationWarning', + notificationError: 'notificationError', + selection: 'selection', + }, +})); diff --git a/packages/mobile-sdk-demo/metro.config.cjs b/packages/mobile-sdk-demo/metro.config.cjs index 8c96308c2..f4f46d152 100644 --- a/packages/mobile-sdk-demo/metro.config.cjs +++ b/packages/mobile-sdk-demo/metro.config.cjs @@ -67,8 +67,11 @@ const config = { unstable_conditionNames: ['react-native', 'import', 'require'], unstable_enableSymlinks: true, nodeModulesPaths: [path.resolve(projectRoot, 'node_modules'), path.resolve(workspaceRoot, 'node_modules')], + + // SVG support assetExts: assetExts.filter(ext => ext !== 'svg'), sourceExts: [...sourceExts, 'svg'], + extraNodeModules: { // Add workspace packages for proper resolution '@selfxyz/common': path.resolve(workspaceRoot, 'common'), diff --git a/packages/mobile-sdk-demo/package.json b/packages/mobile-sdk-demo/package.json index f4a27ac3a..d7b4ec8f7 100644 --- a/packages/mobile-sdk-demo/package.json +++ b/packages/mobile-sdk-demo/package.json @@ -38,6 +38,8 @@ "buffer": "^6.0.3", "constants-browserify": "^1.0.0", "ethers": "^6.11.0", + "lottie-react": "^2.4.1", + "lottie-react-native": "7.2.2", "react": "^18.3.1", "react-native": "0.76.9", "react-native-get-random-values": "^1.11.0", diff --git a/packages/mobile-sdk-demo/src/hooks/useMRZScanner.ts b/packages/mobile-sdk-demo/src/hooks/useMRZScanner.ts deleted file mode 100644 index 39b5ba4f0..000000000 --- a/packages/mobile-sdk-demo/src/hooks/useMRZScanner.ts +++ /dev/null @@ -1,183 +0,0 @@ -// SPDX-FileCopyrightText: 2025 Social Connect Labs, Inc. -// SPDX-License-Identifier: BUSL-1.1 -// NOTE: Converts to Apache-2.0 on 2029-06-11 per LICENSE. - -import { useCallback, useEffect, useMemo, useRef, useState } from 'react'; -import { AccessibilityInfo, PermissionsAndroid, Platform } from 'react-native'; - -import type { MRZInfo } from '@selfxyz/mobile-sdk-alpha'; -import { useReadMRZ } from '@selfxyz/mobile-sdk-alpha/onboarding/read-mrz'; - -import type { NormalizedMRZResult } from '../utils/camera'; -import { normalizeMRZPayload } from '../utils/camera'; - -type PermissionState = 'loading' | 'granted' | 'denied'; -type ScanState = 'idle' | 'scanning' | 'success' | 'error'; - -function announceForAccessibility(message: string) { - if (!message) { - return; - } - - try { - AccessibilityInfo.announceForAccessibility?.(message); - } catch { - // Intentionally swallow to avoid crashing accessibility users on announce failures. - } -} - -export interface DocumentScannerCopy { - instructions: string; - success: string; - error: string; - permissionDenied: string; - resetAnnouncement: string; -} - -export interface DocumentScannerState { - permissionStatus: PermissionState; - scanState: ScanState; - mrzResult: NormalizedMRZResult | null; - error: string | null; - requestPermission: () => Promise; - handleMRZDetected: (payload: MRZInfo) => void; - handleScannerError: (error: string) => void; - handleScanAgain: () => void; -} - -export function useMRZScanner(copy: DocumentScannerCopy): DocumentScannerState { - const [permissionStatus, setPermissionStatus] = useState('loading'); - const [scanState, setScanState] = useState('idle'); - const [mrzResult, setMrzResult] = useState(null); - const [error, setError] = useState(null); - - const scanStartTimeRef = useRef(Date.now()); - const { onPassportRead } = useReadMRZ(scanStartTimeRef); - - const requestPermission = useCallback(async () => { - setPermissionStatus('loading'); - setError(null); - - if (Platform.OS === 'android') { - try { - const result = await PermissionsAndroid.request(PermissionsAndroid.PERMISSIONS.CAMERA, { - title: 'Camera permission', - message: 'We need your permission to access the camera for MRZ scanning.', - buttonPositive: 'Allow', - buttonNegative: 'Cancel', - buttonNeutral: 'Ask me later', - }); - - if (result === PermissionsAndroid.RESULTS.GRANTED) { - setPermissionStatus('granted'); - } else { - setPermissionStatus('denied'); - } - } catch { - setPermissionStatus('denied'); - setError('Camera permission request failed. Please try again.'); - } - } else { - setPermissionStatus('granted'); - } - }, []); - - useEffect(() => { - requestPermission(); - }, [requestPermission]); - - useEffect(() => { - if (permissionStatus === 'granted') { - announceForAccessibility(copy.instructions); - setScanState(current => { - if (current === 'success') { - return current; - } - scanStartTimeRef.current = Date.now(); - return 'scanning'; - }); - } else if (permissionStatus === 'denied') { - announceForAccessibility(copy.permissionDenied); - setScanState('idle'); - } - }, [copy.instructions, copy.permissionDenied, permissionStatus]); - - useEffect(() => { - if (scanState === 'success') { - announceForAccessibility(copy.success); - } else if (scanState === 'error') { - announceForAccessibility(copy.error); - } - }, [copy.error, copy.success, scanState]); - - useEffect(() => { - if (error) { - announceForAccessibility(error); - } - }, [error]); - - const handleMRZDetected = useCallback( - (payload: MRZInfo) => { - setError(null); - - setScanState(current => { - if (current === 'success') { - return current; - } - return 'scanning'; - }); - - try { - const normalized = normalizeMRZPayload(payload); - setMrzResult(normalized); - setScanState('success'); - onPassportRead(null, normalized.info); - } catch { - setScanState('error'); - setError('Unable to validate the MRZ data from the scan.'); - } - }, - [onPassportRead], - ); - - const handleScannerError = useCallback((scannerError: string) => { - setScanState('error'); - setError(scannerError || 'An unexpected camera error occurred.'); - }, []); - - const handleScanAgain = useCallback(() => { - if (permissionStatus === 'denied') { - requestPermission(); - return; - } - - scanStartTimeRef.current = Date.now(); - setMrzResult(null); - setError(null); - setScanState('scanning'); - announceForAccessibility(copy.resetAnnouncement); - }, [copy.resetAnnouncement, permissionStatus, requestPermission]); - - return useMemo( - () => ({ - permissionStatus, - scanState, - mrzResult, - error, - requestPermission, - handleMRZDetected, - handleScannerError, - handleScanAgain, - }), - [ - error, - handleMRZDetected, - handleScanAgain, - handleScannerError, - mrzResult, - permissionStatus, - requestPermission, - scanState, - ], - ); -} diff --git a/packages/mobile-sdk-demo/src/screens/DocumentCamera.tsx b/packages/mobile-sdk-demo/src/screens/DocumentCamera.tsx index a065b4070..192b4f554 100644 --- a/packages/mobile-sdk-demo/src/screens/DocumentCamera.tsx +++ b/packages/mobile-sdk-demo/src/screens/DocumentCamera.tsx @@ -2,225 +2,19 @@ // SPDX-License-Identifier: BUSL-1.1 // NOTE: Converts to Apache-2.0 on 2029-06-11 per LICENSE. -import React, { useCallback } from 'react'; -import { ActivityIndicator, Alert, StyleSheet, Text, TouchableOpacity, View } from 'react-native'; +import React from 'react'; -import { MRZScannerView } from '@selfxyz/mobile-sdk-alpha/onboarding/read-mrz'; - -import ScreenLayout from '../components/ScreenLayout'; -import DocumentScanResultCard from '../components/DocumentScanResultCard'; -import { useMRZScanner } from '../hooks/useMRZScanner'; +import { DocumentCameraScreen } from '@selfxyz/mobile-sdk-alpha/onboarding/document-camera-screen'; type Props = { onBack: () => void; + onSuccess: () => void; }; -const instructionsText = 'Align the machine-readable text with the frame and hold steady while we scan.'; - -const successMessage = 'Document scan successful. Review the details below.'; -const errorMessage = 'We could not read your document. Adjust lighting and try again.'; -const permissionDeniedMessage = 'Camera access was denied. Enable permissions to scan your document.'; - -export default function DocumentCamera({ onBack }: Props) { - const scannerCopy = { - instructions: instructionsText, - success: successMessage, - error: errorMessage, - permissionDenied: permissionDeniedMessage, - resetAnnouncement: 'Ready to scan again. Align the document in the viewfinder.', - } as const; - - const { - permissionStatus, - scanState, - mrzResult, - error, - requestPermission, - handleMRZDetected, - handleScannerError, - handleScanAgain, - } = useMRZScanner(scannerCopy); - - const handleSaveDocument = useCallback(() => { - if (!mrzResult) { - Alert.alert('Save Document', 'Scan a document before attempting to save.'); - return; - } - - Alert.alert( - 'Save Document', - 'Document storage will be available in a future release. Your scan is ready when you need it.', - ); - }, [mrzResult]); - - const renderPermissionDenied = () => ( - - {permissionDeniedMessage} - - Request Permission - - - ); - - const renderLoading = () => ( - - - Preparing camera… - - ); - +export default function DocumentCamera({ onBack, onSuccess }: Props) { return ( - { - onBack(); - }} - contentStyle={styles.screenContent} - > - {permissionStatus === 'loading' && renderLoading()} - {permissionStatus === 'denied' && renderPermissionDenied()} - - {permissionStatus === 'granted' && ( - - - Position your document - {instructionsText} - - - - - - - - {scanState === 'scanning' && !error && ( - - - Scanning for MRZ data… - - )} - - {scanState === 'success' && mrzResult && ( - {successMessage} - )} - - {scanState === 'error' && error && {error}} - - - - - Scan Again - - - - Save Document - - - - {mrzResult && } - - )} - + <> + + ); } - -const styles = StyleSheet.create({ - screenContent: { - gap: 16, - }, - contentWrapper: { - flex: 1, - }, - instructionsContainer: { - marginBottom: 16, - }, - instructionsTitle: { - color: '#0f172a', - fontWeight: '600', - fontSize: 16, - marginBottom: 4, - }, - instructionsText: { - color: '#475569', - fontSize: 14, - lineHeight: 20, - }, - cameraWrapper: { - backgroundColor: '#0f172a', - borderRadius: 16, - overflow: 'hidden', - minHeight: 260, - marginBottom: 16, - }, - scanner: { - flex: 1, - width: '100%', - height: '100%', - backgroundColor: '#0f172a', - }, - statusContainer: { - marginBottom: 16, - alignItems: 'center', - }, - statusRow: { - flexDirection: 'row', - alignItems: 'center', - justifyContent: 'center', - gap: 8, - }, - statusText: { - color: '#0f172a', - fontSize: 14, - marginTop: 8, - }, - successText: { - color: '#15803d', - fontWeight: '600', - }, - errorText: { - color: '#b91c1c', - fontWeight: '600', - }, - actions: { - flexDirection: 'row', - gap: 12, - }, - actionsWithResult: { - marginBottom: 16, - }, - primaryButton: { - flex: 1, - backgroundColor: '#0f172a', - paddingVertical: 12, - borderRadius: 8, - alignItems: 'center', - }, - primaryButtonText: { - color: '#ffffff', - fontSize: 15, - fontWeight: '600', - }, - secondaryButton: { - flex: 1, - backgroundColor: '#e2e8f0', - paddingVertical: 12, - borderRadius: 8, - alignItems: 'center', - }, - secondaryButtonText: { - color: '#0f172a', - fontSize: 15, - fontWeight: '600', - }, - centeredState: { - flex: 1, - alignItems: 'center', - justifyContent: 'center', - gap: 12, - }, - permissionText: { - color: '#0f172a', - textAlign: 'center', - fontSize: 15, - lineHeight: 22, - }, -}); diff --git a/packages/mobile-sdk-demo/src/screens/index.ts b/packages/mobile-sdk-demo/src/screens/index.ts index 4be4e64bb..ee37bb2fb 100644 --- a/packages/mobile-sdk-demo/src/screens/index.ts +++ b/packages/mobile-sdk-demo/src/screens/index.ts @@ -74,7 +74,10 @@ export const screenDescriptors: ScreenDescriptor[] = [ sectionTitle: '📸 Scanning', status: 'placeholder', load: () => require('./DocumentCamera').default, - getProps: ({ navigate }) => ({ onBack: () => navigate('home') }), + getProps: ({ navigate }) => ({ + onBack: () => navigate('home'), + onSuccess: () => navigate('nfc'), + }), }, { id: 'nfc', diff --git a/packages/mobile-sdk-demo/tests/screens/DocumentCamera.test.tsx b/packages/mobile-sdk-demo/tests/screens/DocumentCamera.test.tsx index f4be57707..9bfe15419 100644 --- a/packages/mobile-sdk-demo/tests/screens/DocumentCamera.test.tsx +++ b/packages/mobile-sdk-demo/tests/screens/DocumentCamera.test.tsx @@ -8,8 +8,9 @@ import DocumentCamera from '../../src/screens/DocumentCamera'; describe('DocumentCamera screen', () => { it('shows placeholder messaging and handles back navigation', async () => { const onBack = vi.fn(); + const onSuccess = vi.fn(); - render(); + render(); expect(screen.getByText('Document Camera')).toBeInTheDocument(); expect(screen.getByText(/camera-based document scanning/i)).toBeInTheDocument(); diff --git a/yarn.lock b/yarn.lock index dfc5f28b7..d79760da9 100644 --- a/yarn.lock +++ b/yarn.lock @@ -7457,6 +7457,7 @@ __metadata: eslint-plugin-simple-import-sort: "npm:^12.1.1" eslint-plugin-sort-exports: "npm:^0.9.1" jsdom: "npm:^25.0.1" + lottie-react-native: "npm:7.2.2" node-forge: "npm:^1.3.1" prettier: "npm:^3.5.3" react: "npm:^18.3.1" @@ -7474,6 +7475,7 @@ __metadata: xstate: "npm:^5.20.2" zustand: "npm:^4.5.2" peerDependencies: + lottie-react-native: 7.2.2 react: ^18.3.1 react-native: 0.76.9 react-native-haptic-feedback: "*" @@ -23818,6 +23820,8 @@ __metadata: ethers: "npm:^6.11.0" find-yarn-workspace-root: "npm:^2.0.0" jsdom: "npm:^25.0.1" + lottie-react: "npm:^2.4.1" + lottie-react-native: "npm:7.2.2" metro-react-native-babel-preset: "npm:0.76.9" prettier: "npm:^3.6.2" react: "npm:^18.3.1"