@onairos/react-native 3.1.15 → 3.1.17

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (191) hide show
  1. package/README.md +404 -0
  2. package/lib/commonjs/assets/images/Checkbox.svg +3 -3
  3. package/lib/commonjs/assets/images/EnochE.svg +19 -19
  4. package/lib/commonjs/assets/images/Personalityprofile.svg +3 -3
  5. package/lib/commonjs/assets/images/Personalitytraits.svg +3 -3
  6. package/lib/commonjs/assets/images/Userpreferences.svg +3 -3
  7. package/lib/commonjs/assets/images/arrow.svg +20 -20
  8. package/lib/commonjs/assets/images/basicproficon.svg +43 -43
  9. package/lib/commonjs/assets/images/basicprofile.svg +3 -3
  10. package/lib/commonjs/assets/images/checkmark.svg +4 -4
  11. package/lib/commonjs/assets/images/contentanalysis.svg +3 -3
  12. package/lib/commonjs/assets/images/contenticon.svg +23 -23
  13. package/lib/commonjs/assets/images/personalityicon.svg +18 -18
  14. package/lib/commonjs/assets/images/x-close.svg +3 -3
  15. package/lib/commonjs/components/OnairosSignInButton.js +32 -74
  16. package/lib/commonjs/components/OnairosSignInButton.js.map +1 -1
  17. package/lib/commonjs/components/UniversalOnboarding.js +4 -4
  18. package/lib/commonjs/config/api.js +2 -2
  19. package/lib/commonjs/hooks/useConnections.js +6 -6
  20. package/lib/commonjs/hooks/useUserConnections.js +10 -10
  21. package/lib/commonjs/index.js +5 -12
  22. package/lib/commonjs/index.js.map +1 -1
  23. package/lib/commonjs/services/apiClient.js +35 -35
  24. package/lib/commonjs/services/apiKeyService.js +99 -99
  25. package/lib/commonjs/services/authService.js +82 -82
  26. package/lib/commonjs/services/biometricPinService.js +10 -10
  27. package/lib/commonjs/services/connectedAccountsService.js +32 -32
  28. package/lib/commonjs/services/googleAuthService.js +15 -15
  29. package/lib/commonjs/services/imageCompressionService.js +15 -15
  30. package/lib/commonjs/services/jwtStorageService.js +59 -59
  31. package/lib/commonjs/services/mobileTrainingService.js +14 -14
  32. package/lib/commonjs/services/pinEncryptionService.js +10 -10
  33. package/lib/commonjs/services/pinStorageUtils.js +15 -15
  34. package/lib/commonjs/services/platformAuthService.js +47 -47
  35. package/lib/commonjs/services/storageService.js +31 -31
  36. package/lib/commonjs/services/trainingApiHelpers.js +33 -33
  37. package/lib/commonjs/services/userConnectionsService.js +24 -24
  38. package/lib/commonjs/utils/Portal.js +4 -4
  39. package/lib/commonjs/utils/api.js +24 -24
  40. package/lib/commonjs/utils/auth.js +18 -18
  41. package/lib/commonjs/utils/crypto.js +13 -13
  42. package/lib/commonjs/utils/encryption.js +12 -12
  43. package/lib/commonjs/utils/eventUtils.js +52 -52
  44. package/lib/commonjs/utils/programmaticFlow.js +16 -16
  45. package/lib/commonjs/utils/retryHelper.js +27 -27
  46. package/lib/module/assets/images/Checkbox.svg +3 -3
  47. package/lib/module/assets/images/EnochE.svg +19 -19
  48. package/lib/module/assets/images/Personalityprofile.svg +3 -3
  49. package/lib/module/assets/images/Personalitytraits.svg +3 -3
  50. package/lib/module/assets/images/Userpreferences.svg +3 -3
  51. package/lib/module/assets/images/arrow.svg +20 -20
  52. package/lib/module/assets/images/basicproficon.svg +43 -43
  53. package/lib/module/assets/images/basicprofile.svg +3 -3
  54. package/lib/module/assets/images/checkmark.svg +4 -4
  55. package/lib/module/assets/images/contentanalysis.svg +3 -3
  56. package/lib/module/assets/images/contenticon.svg +23 -23
  57. package/lib/module/assets/images/personalityicon.svg +18 -18
  58. package/lib/module/assets/images/x-close.svg +3 -3
  59. package/lib/module/components/OnairosSignInButton.js +32 -74
  60. package/lib/module/components/OnairosSignInButton.js.map +1 -1
  61. package/lib/module/components/UniversalOnboarding.js +4 -4
  62. package/lib/module/config/api.js +2 -2
  63. package/lib/module/hooks/useConnections.js +6 -6
  64. package/lib/module/hooks/useUserConnections.js +10 -10
  65. package/lib/module/index.js +5 -6
  66. package/lib/module/index.js.map +1 -1
  67. package/lib/module/services/apiClient.js +35 -35
  68. package/lib/module/services/apiKeyService.js +99 -99
  69. package/lib/module/services/authService.js +82 -82
  70. package/lib/module/services/biometricPinService.js +10 -10
  71. package/lib/module/services/connectedAccountsService.js +32 -32
  72. package/lib/module/services/googleAuthService.js +15 -15
  73. package/lib/module/services/imageCompressionService.js +15 -15
  74. package/lib/module/services/jwtStorageService.js +59 -59
  75. package/lib/module/services/mobileTrainingService.js +14 -14
  76. package/lib/module/services/pinEncryptionService.js +10 -10
  77. package/lib/module/services/pinStorageUtils.js +15 -15
  78. package/lib/module/services/platformAuthService.js +47 -47
  79. package/lib/module/services/storageService.js +31 -31
  80. package/lib/module/services/trainingApiHelpers.js +33 -33
  81. package/lib/module/services/userConnectionsService.js +24 -24
  82. package/lib/module/utils/Portal.js +4 -4
  83. package/lib/module/utils/api.js +24 -24
  84. package/lib/module/utils/auth.js +18 -18
  85. package/lib/module/utils/crypto.js +13 -13
  86. package/lib/module/utils/encryption.js +12 -12
  87. package/lib/module/utils/eventUtils.js +52 -52
  88. package/lib/module/utils/programmaticFlow.js +16 -16
  89. package/lib/module/utils/retryHelper.js +27 -27
  90. package/lib/typescript/components/OnairosSignInButton.d.ts.map +1 -1
  91. package/lib/typescript/index.d.ts +0 -1
  92. package/lib/typescript/index.d.ts.map +1 -1
  93. package/package.json +163 -163
  94. package/src/api/index.ts +151 -151
  95. package/src/assets/images/Checkbox.svg +3 -3
  96. package/src/assets/images/EnochE.svg +19 -19
  97. package/src/assets/images/Personalityprofile.svg +3 -3
  98. package/src/assets/images/Personalitytraits.svg +3 -3
  99. package/src/assets/images/Userpreferences.svg +3 -3
  100. package/src/assets/images/arrow.svg +20 -20
  101. package/src/assets/images/basicproficon.svg +43 -43
  102. package/src/assets/images/basicprofile.svg +3 -3
  103. package/src/assets/images/checkmark.svg +4 -4
  104. package/src/assets/images/contentanalysis.svg +3 -3
  105. package/src/assets/images/contenticon.svg +23 -23
  106. package/src/assets/images/personalityicon.svg +18 -18
  107. package/src/assets/images/x-close.svg +3 -3
  108. package/src/components/BodyText.tsx +33 -33
  109. package/src/components/BrandMark.tsx +62 -62
  110. package/src/components/CodeInput.tsx +32 -32
  111. package/src/components/DataRequestScreen.tsx +355 -355
  112. package/src/components/EmailInput.tsx +31 -31
  113. package/src/components/EmailVerificationModal.tsx +363 -363
  114. package/src/components/ExistingUserDataConfirmation.tsx +506 -506
  115. package/src/components/GoogleButton.tsx +55 -55
  116. package/src/components/HeadingGroup.tsx +49 -49
  117. package/src/components/ModalHeader.tsx +125 -125
  118. package/src/components/ModalSheet.tsx +57 -57
  119. package/src/components/Onairos.tsx +422 -422
  120. package/src/components/OnairosButton.tsx +339 -339
  121. package/src/components/OnairosSignInButton.tsx +130 -166
  122. package/src/components/Overlay.tsx +506 -506
  123. package/src/components/PersonaImage.tsx +79 -79
  124. package/src/components/PersonaLoadingScreen.tsx +201 -201
  125. package/src/components/PersonalizationConsentScreen.tsx +410 -410
  126. package/src/components/PinCreationScreen.tsx +492 -492
  127. package/src/components/PinInput.tsx +555 -555
  128. package/src/components/PlatformConnectorsStep.tsx +891 -891
  129. package/src/components/PlatformList.tsx +144 -144
  130. package/src/components/PlatformToggle.tsx +226 -226
  131. package/src/components/PrimaryButton.tsx +213 -213
  132. package/src/components/SignInMatchAnimation.tsx +225 -225
  133. package/src/components/SignInStep.tsx +217 -217
  134. package/src/components/TrainingModal.tsx +1047 -1047
  135. package/src/components/UniversalOnboarding.tsx +2887 -2887
  136. package/src/components/VerificationStep.tsx +198 -198
  137. package/src/components/WelcomeScreen.tsx +473 -473
  138. package/src/components/icons/Basicproficon.tsx +30 -30
  139. package/src/components/icons/Basicprofile.tsx +17 -17
  140. package/src/components/icons/Checkbox.tsx +17 -17
  141. package/src/components/icons/Checkmark.tsx +24 -24
  142. package/src/components/icons/Contentanalysis.tsx +17 -17
  143. package/src/components/icons/Contenticon.tsx +30 -30
  144. package/src/components/icons/EnochE.tsx +39 -39
  145. package/src/components/icons/Personalityicon.tsx +22 -22
  146. package/src/components/icons/Personalityprofile.tsx +17 -17
  147. package/src/components/icons/Personalitytraits.tsx +17 -17
  148. package/src/components/icons/Userpreferences.tsx +17 -17
  149. package/src/components/icons/index.ts +12 -12
  150. package/src/components/onboarding/OAuthWebView.tsx +232 -232
  151. package/src/config/api.ts +25 -25
  152. package/src/context/AuthContext.tsx +393 -393
  153. package/src/hooks/useConnectedAccounts.ts +138 -138
  154. package/src/hooks/useConnections.ts +161 -161
  155. package/src/hooks/useCredentials.ts +174 -174
  156. package/src/hooks/useUserConnections.ts +165 -165
  157. package/src/index.js +14 -14
  158. package/src/index.ts +94 -95
  159. package/src/services/apiClient.ts +336 -336
  160. package/src/services/apiKeyService.ts +919 -919
  161. package/src/services/authService.ts +1008 -1008
  162. package/src/services/biometricPinService.ts +192 -192
  163. package/src/services/connectedAccountsService.ts +289 -289
  164. package/src/services/googleAuthService.ts +279 -279
  165. package/src/services/imageCompressionService.ts +302 -302
  166. package/src/services/jwtStorageService.ts +256 -256
  167. package/src/services/mobileTrainingService.ts +203 -203
  168. package/src/services/pinEncryptionService.ts +75 -75
  169. package/src/services/pinStorageUtils.ts +96 -96
  170. package/src/services/platformAuthService.ts +1346 -1346
  171. package/src/services/storageService.ts +451 -451
  172. package/src/services/trainingApiHelpers.ts +66 -66
  173. package/src/services/userConnectionsService.ts +556 -556
  174. package/src/services/youtubeMigrationService.ts +453 -453
  175. package/src/theme/index.ts +239 -239
  176. package/src/types/ambient.d.ts +28 -28
  177. package/src/types/index.ts +265 -265
  178. package/src/types/node-fix.d.ts +18 -18
  179. package/src/types/node-override.d.ts +23 -23
  180. package/src/types/opacity.d.ts +15 -15
  181. package/src/types/types.d.ts +17 -17
  182. package/src/utils/Portal.tsx +82 -82
  183. package/src/utils/api.js +111 -111
  184. package/src/utils/auth.js +103 -103
  185. package/src/utils/crypto.js +59 -59
  186. package/src/utils/encryption.ts +68 -68
  187. package/src/utils/eventUtils.ts +302 -302
  188. package/src/utils/haptics.ts +58 -58
  189. package/src/utils/imagePreloader.ts +2 -2
  190. package/src/utils/programmaticFlow.ts +112 -112
  191. package/src/utils/retryHelper.ts +274 -274
@@ -1,193 +1,193 @@
1
- import * as Keychain from 'react-native-keychain';
2
- import { Platform } from 'react-native';
3
-
4
- // Service key for PIN storage
5
- const PIN_SERVICE_KEY = 'OnairosEventsPIN';
6
- const PIN_USERNAME = 'user_pin';
7
-
8
- export interface BiometricPinService {
9
- storePinWithBiometric: (pin: string) => Promise<boolean>;
10
- retrievePinWithBiometric: () => Promise<string | null>;
11
- isPinStored: () => Promise<boolean>;
12
- removePinFromStorage: () => Promise<boolean>;
13
- isBiometricAvailable: () => Promise<boolean>;
14
- }
15
-
16
- /**
17
- * Check if biometric authentication is available on the device
18
- */
19
- export const isBiometricAvailable = async (): Promise<boolean> => {
20
- try {
21
- const biometryType = await Keychain.getSupportedBiometryType();
22
- console.log('📱 Biometric support:', biometryType);
23
-
24
- // Check for Face ID on iOS or any biometric on Android
25
- return biometryType === Keychain.BIOMETRY_TYPE.FACE_ID ||
26
- biometryType === Keychain.BIOMETRY_TYPE.TOUCH_ID ||
27
- biometryType === Keychain.BIOMETRY_TYPE.FINGERPRINT ||
28
- biometryType === Keychain.BIOMETRY_TYPE.FACE ||
29
- biometryType === Keychain.BIOMETRY_TYPE.IRIS;
30
- } catch (error) {
31
- console.error('❌ Error checking biometric availability:', error);
32
- return false;
33
- }
34
- };
35
-
36
- /**
37
- * Store PIN with biometric authentication
38
- */
39
- export const storePinWithBiometric = async (pin: string): Promise<boolean> => {
40
- try {
41
- console.log('🔐 Starting Face ID authentication for PIN storage...');
42
-
43
- // Check if biometric is available - REQUIRED for this implementation
44
- const biometricAvailable = await isBiometricAvailable();
45
- if (!biometricAvailable) {
46
- console.error('❌ Face ID/Touch ID not available on this device');
47
- return false;
48
- }
49
-
50
- console.log('📱 Face ID available, showing authentication prompt...');
51
-
52
- // Configure keychain options for MANDATORY biometric authentication
53
- const keychainOptions = {
54
- accessControl: Keychain.ACCESS_CONTROL.BIOMETRY_CURRENT_SET,
55
- authenticationType: Keychain.AUTHENTICATION_TYPE.BIOMETRICS,
56
- touchID: true,
57
- showModal: true,
58
- kLocalizedFallbackTitle: '', // Disable fallback to prevent passcode option
59
- // Add explicit prompts for better UX
60
- ...(Platform.OS === 'ios' && {
61
- localizedPrompt: 'Authenticate with Face ID to secure your PIN',
62
- localizedCancel: 'Cancel',
63
- }),
64
- ...(Platform.OS === 'android' && {
65
- promptMessage: 'Authenticate with biometric to secure your PIN',
66
- cancelButtonText: 'Cancel',
67
- })
68
- } as const;
69
-
70
- // Store the PIN - this will trigger the Face ID prompt
71
- console.log('👤 Requesting Face ID authentication...');
72
- const result = await Keychain.setInternetCredentials(
73
- PIN_SERVICE_KEY,
74
- PIN_USERNAME,
75
- pin,
76
- keychainOptions
77
- );
78
-
79
- console.log('✅ Face ID authentication successful - PIN stored securely');
80
- console.log('🔍 Keychain storage result:', result);
81
- return true;
82
-
83
- } catch (error: any) {
84
- console.error('❌ Face ID authentication failed:', error);
85
- console.error('❌ Error details:', {
86
- message: error.message,
87
- code: error.code,
88
- domain: error.domain,
89
- userInfo: error.userInfo
90
- });
91
-
92
- // Handle specific error cases
93
- if (error?.message?.includes('UserCancel') || error?.code === -128) {
94
- console.log('👤 User cancelled Face ID authentication');
95
- return false;
96
- } else if (error?.message?.includes('BiometryNotAvailable') || error?.code === -6) {
97
- console.log('📱 Face ID/Touch ID not available');
98
- return false;
99
- } else if (error?.message?.includes('AuthenticationFailed') || error?.code === -1) {
100
- console.log('🚫 Face ID authentication failed');
101
- return false;
102
- } else if (error?.message?.includes('BiometryLockout') || error?.code === -8) {
103
- console.log('🔒 Face ID locked out - too many failed attempts');
104
- return false;
105
- }
106
-
107
- console.error('❌ Unknown Face ID error:', error.message);
108
- return false;
109
- }
110
- };
111
-
112
- /**
113
- * Retrieve PIN with biometric authentication
114
- */
115
- export const retrievePinWithBiometric = async (): Promise<string | null> => {
116
- try {
117
- console.log('🔓 Retrieving PIN with biometric authentication...');
118
-
119
- // Retrieve the PIN (will prompt for biometric if configured)
120
- const credentials = await Keychain.getInternetCredentials(PIN_SERVICE_KEY);
121
-
122
- if (credentials && credentials.password) {
123
- console.log('✅ PIN retrieved successfully');
124
- return credentials.password;
125
- } else {
126
- console.log('⚠️ No PIN found in secure storage');
127
- return null;
128
- }
129
-
130
- } catch (error: any) {
131
- console.error('❌ Error retrieving PIN with biometric:', error);
132
-
133
- // Handle specific error cases
134
- if (error?.message?.includes('UserCancel')) {
135
- console.log('👤 User cancelled biometric authentication');
136
- return null;
137
- } else if (error?.message?.includes('BiometryNotAvailable')) {
138
- console.log('📱 Biometric not available');
139
- return null;
140
- }
141
-
142
- return null;
143
- }
144
- };
145
-
146
- /**
147
- * Check if PIN is stored in secure storage
148
- */
149
- export const isPinStored = async (): Promise<boolean> => {
150
- try {
151
- const credentials = await Keychain.getInternetCredentials(PIN_SERVICE_KEY);
152
- return !!(credentials && credentials.password);
153
- } catch (error: any) {
154
- console.error('❌ Error checking PIN storage:', error);
155
- return false;
156
- }
157
- };
158
-
159
- /**
160
- * Remove PIN from secure storage
161
- */
162
- export const removePinFromStorage = async (): Promise<boolean> => {
163
- try {
164
- console.log('🗑️ Removing PIN from secure storage...');
165
-
166
- // For v10.0.0, we need to use resetGenericPassword instead of resetInternetCredentials
167
- // or handle the API difference properly
168
- try {
169
- await Keychain.resetGenericPassword({ service: PIN_SERVICE_KEY });
170
- console.log('✅ PIN removed successfully');
171
- return true;
172
- } catch (resetError) {
173
- // Fallback: try the old API
174
- console.log('🔄 Trying alternative reset method...');
175
- const result = await (Keychain as any).resetInternetCredentials(PIN_SERVICE_KEY);
176
- console.log('✅ PIN removed successfully with fallback method');
177
- return true;
178
- }
179
-
180
- } catch (error: any) {
181
- console.error('❌ Error removing PIN from storage:', error);
182
- return false;
183
- }
184
- };
185
-
186
- // Export the service object
187
- export const biometricPinService: BiometricPinService = {
188
- storePinWithBiometric,
189
- retrievePinWithBiometric,
190
- isPinStored,
191
- removePinFromStorage,
192
- isBiometricAvailable,
1
+ import * as Keychain from 'react-native-keychain';
2
+ import { Platform } from 'react-native';
3
+
4
+ // Service key for PIN storage
5
+ const PIN_SERVICE_KEY = 'OnairosEventsPIN';
6
+ const PIN_USERNAME = 'user_pin';
7
+
8
+ export interface BiometricPinService {
9
+ storePinWithBiometric: (pin: string) => Promise<boolean>;
10
+ retrievePinWithBiometric: () => Promise<string | null>;
11
+ isPinStored: () => Promise<boolean>;
12
+ removePinFromStorage: () => Promise<boolean>;
13
+ isBiometricAvailable: () => Promise<boolean>;
14
+ }
15
+
16
+ /**
17
+ * Check if biometric authentication is available on the device
18
+ */
19
+ export const isBiometricAvailable = async (): Promise<boolean> => {
20
+ try {
21
+ const biometryType = await Keychain.getSupportedBiometryType();
22
+ console.log('📱 Biometric support:', biometryType);
23
+
24
+ // Check for Face ID on iOS or any biometric on Android
25
+ return biometryType === Keychain.BIOMETRY_TYPE.FACE_ID ||
26
+ biometryType === Keychain.BIOMETRY_TYPE.TOUCH_ID ||
27
+ biometryType === Keychain.BIOMETRY_TYPE.FINGERPRINT ||
28
+ biometryType === Keychain.BIOMETRY_TYPE.FACE ||
29
+ biometryType === Keychain.BIOMETRY_TYPE.IRIS;
30
+ } catch (error) {
31
+ console.error('❌ Error checking biometric availability:', error);
32
+ return false;
33
+ }
34
+ };
35
+
36
+ /**
37
+ * Store PIN with biometric authentication
38
+ */
39
+ export const storePinWithBiometric = async (pin: string): Promise<boolean> => {
40
+ try {
41
+ console.log('🔐 Starting Face ID authentication for PIN storage...');
42
+
43
+ // Check if biometric is available - REQUIRED for this implementation
44
+ const biometricAvailable = await isBiometricAvailable();
45
+ if (!biometricAvailable) {
46
+ console.error('❌ Face ID/Touch ID not available on this device');
47
+ return false;
48
+ }
49
+
50
+ console.log('📱 Face ID available, showing authentication prompt...');
51
+
52
+ // Configure keychain options for MANDATORY biometric authentication
53
+ const keychainOptions = {
54
+ accessControl: Keychain.ACCESS_CONTROL.BIOMETRY_CURRENT_SET,
55
+ authenticationType: Keychain.AUTHENTICATION_TYPE.BIOMETRICS,
56
+ touchID: true,
57
+ showModal: true,
58
+ kLocalizedFallbackTitle: '', // Disable fallback to prevent passcode option
59
+ // Add explicit prompts for better UX
60
+ ...(Platform.OS === 'ios' && {
61
+ localizedPrompt: 'Authenticate with Face ID to secure your PIN',
62
+ localizedCancel: 'Cancel',
63
+ }),
64
+ ...(Platform.OS === 'android' && {
65
+ promptMessage: 'Authenticate with biometric to secure your PIN',
66
+ cancelButtonText: 'Cancel',
67
+ })
68
+ } as const;
69
+
70
+ // Store the PIN - this will trigger the Face ID prompt
71
+ console.log('👤 Requesting Face ID authentication...');
72
+ const result = await Keychain.setInternetCredentials(
73
+ PIN_SERVICE_KEY,
74
+ PIN_USERNAME,
75
+ pin,
76
+ keychainOptions
77
+ );
78
+
79
+ console.log('✅ Face ID authentication successful - PIN stored securely');
80
+ console.log('🔍 Keychain storage result:', result);
81
+ return true;
82
+
83
+ } catch (error: any) {
84
+ console.error('❌ Face ID authentication failed:', error);
85
+ console.error('❌ Error details:', {
86
+ message: error.message,
87
+ code: error.code,
88
+ domain: error.domain,
89
+ userInfo: error.userInfo
90
+ });
91
+
92
+ // Handle specific error cases
93
+ if (error?.message?.includes('UserCancel') || error?.code === -128) {
94
+ console.log('👤 User cancelled Face ID authentication');
95
+ return false;
96
+ } else if (error?.message?.includes('BiometryNotAvailable') || error?.code === -6) {
97
+ console.log('📱 Face ID/Touch ID not available');
98
+ return false;
99
+ } else if (error?.message?.includes('AuthenticationFailed') || error?.code === -1) {
100
+ console.log('🚫 Face ID authentication failed');
101
+ return false;
102
+ } else if (error?.message?.includes('BiometryLockout') || error?.code === -8) {
103
+ console.log('🔒 Face ID locked out - too many failed attempts');
104
+ return false;
105
+ }
106
+
107
+ console.error('❌ Unknown Face ID error:', error.message);
108
+ return false;
109
+ }
110
+ };
111
+
112
+ /**
113
+ * Retrieve PIN with biometric authentication
114
+ */
115
+ export const retrievePinWithBiometric = async (): Promise<string | null> => {
116
+ try {
117
+ console.log('🔓 Retrieving PIN with biometric authentication...');
118
+
119
+ // Retrieve the PIN (will prompt for biometric if configured)
120
+ const credentials = await Keychain.getInternetCredentials(PIN_SERVICE_KEY);
121
+
122
+ if (credentials && credentials.password) {
123
+ console.log('✅ PIN retrieved successfully');
124
+ return credentials.password;
125
+ } else {
126
+ console.log('⚠️ No PIN found in secure storage');
127
+ return null;
128
+ }
129
+
130
+ } catch (error: any) {
131
+ console.error('❌ Error retrieving PIN with biometric:', error);
132
+
133
+ // Handle specific error cases
134
+ if (error?.message?.includes('UserCancel')) {
135
+ console.log('👤 User cancelled biometric authentication');
136
+ return null;
137
+ } else if (error?.message?.includes('BiometryNotAvailable')) {
138
+ console.log('📱 Biometric not available');
139
+ return null;
140
+ }
141
+
142
+ return null;
143
+ }
144
+ };
145
+
146
+ /**
147
+ * Check if PIN is stored in secure storage
148
+ */
149
+ export const isPinStored = async (): Promise<boolean> => {
150
+ try {
151
+ const credentials = await Keychain.getInternetCredentials(PIN_SERVICE_KEY);
152
+ return !!(credentials && credentials.password);
153
+ } catch (error: any) {
154
+ console.error('❌ Error checking PIN storage:', error);
155
+ return false;
156
+ }
157
+ };
158
+
159
+ /**
160
+ * Remove PIN from secure storage
161
+ */
162
+ export const removePinFromStorage = async (): Promise<boolean> => {
163
+ try {
164
+ console.log('🗑️ Removing PIN from secure storage...');
165
+
166
+ // For v10.0.0, we need to use resetGenericPassword instead of resetInternetCredentials
167
+ // or handle the API difference properly
168
+ try {
169
+ await Keychain.resetGenericPassword({ service: PIN_SERVICE_KEY });
170
+ console.log('✅ PIN removed successfully');
171
+ return true;
172
+ } catch (resetError) {
173
+ // Fallback: try the old API
174
+ console.log('🔄 Trying alternative reset method...');
175
+ const result = await (Keychain as any).resetInternetCredentials(PIN_SERVICE_KEY);
176
+ console.log('✅ PIN removed successfully with fallback method');
177
+ return true;
178
+ }
179
+
180
+ } catch (error: any) {
181
+ console.error('❌ Error removing PIN from storage:', error);
182
+ return false;
183
+ }
184
+ };
185
+
186
+ // Export the service object
187
+ export const biometricPinService: BiometricPinService = {
188
+ storePinWithBiometric,
189
+ retrievePinWithBiometric,
190
+ isPinStored,
191
+ removePinFromStorage,
192
+ isBiometricAvailable,
193
193
  };