@onairos/react-native 3.1.16 → 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 (198) 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/OnairosButton.js +290 -0
  16. package/lib/commonjs/components/OnairosButton.js.map +1 -0
  17. package/lib/commonjs/components/OnairosSignInButton.js +30 -8
  18. package/lib/commonjs/components/OnairosSignInButton.js.map +1 -1
  19. package/lib/commonjs/components/UniversalOnboarding.js +4 -4
  20. package/lib/commonjs/config/api.js +2 -2
  21. package/lib/commonjs/hooks/useConnections.js +6 -6
  22. package/lib/commonjs/hooks/useUserConnections.js +10 -10
  23. package/lib/commonjs/index.js +9 -10
  24. package/lib/commonjs/index.js.map +1 -1
  25. package/lib/commonjs/services/apiClient.js +35 -35
  26. package/lib/commonjs/services/apiKeyService.js +99 -99
  27. package/lib/commonjs/services/authService.js +82 -82
  28. package/lib/commonjs/services/biometricPinService.js +10 -10
  29. package/lib/commonjs/services/connectedAccountsService.js +32 -32
  30. package/lib/commonjs/services/googleAuthService.js +15 -15
  31. package/lib/commonjs/services/imageCompressionService.js +15 -15
  32. package/lib/commonjs/services/jwtStorageService.js +59 -59
  33. package/lib/commonjs/services/mobileTrainingService.js +14 -14
  34. package/lib/commonjs/services/pinEncryptionService.js +10 -10
  35. package/lib/commonjs/services/pinStorageUtils.js +15 -15
  36. package/lib/commonjs/services/platformAuthService.js +47 -47
  37. package/lib/commonjs/services/storageService.js +31 -31
  38. package/lib/commonjs/services/trainingApiHelpers.js +33 -33
  39. package/lib/commonjs/services/userConnectionsService.js +24 -24
  40. package/lib/commonjs/utils/Portal.js +4 -4
  41. package/lib/commonjs/utils/api.js +24 -24
  42. package/lib/commonjs/utils/auth.js +18 -18
  43. package/lib/commonjs/utils/crypto.js +13 -13
  44. package/lib/commonjs/utils/encryption.js +12 -12
  45. package/lib/commonjs/utils/eventUtils.js +52 -52
  46. package/lib/commonjs/utils/programmaticFlow.js +16 -16
  47. package/lib/commonjs/utils/retryHelper.js +27 -27
  48. package/lib/module/assets/images/Checkbox.svg +3 -3
  49. package/lib/module/assets/images/EnochE.svg +19 -19
  50. package/lib/module/assets/images/Personalityprofile.svg +3 -3
  51. package/lib/module/assets/images/Personalitytraits.svg +3 -3
  52. package/lib/module/assets/images/Userpreferences.svg +3 -3
  53. package/lib/module/assets/images/arrow.svg +20 -20
  54. package/lib/module/assets/images/basicproficon.svg +43 -43
  55. package/lib/module/assets/images/basicprofile.svg +3 -3
  56. package/lib/module/assets/images/checkmark.svg +4 -4
  57. package/lib/module/assets/images/contentanalysis.svg +3 -3
  58. package/lib/module/assets/images/contenticon.svg +23 -23
  59. package/lib/module/assets/images/personalityicon.svg +18 -18
  60. package/lib/module/assets/images/x-close.svg +3 -3
  61. package/lib/module/components/OnairosButton.js +282 -0
  62. package/lib/module/components/OnairosButton.js.map +1 -0
  63. package/lib/module/components/OnairosSignInButton.js +30 -8
  64. package/lib/module/components/OnairosSignInButton.js.map +1 -1
  65. package/lib/module/components/UniversalOnboarding.js +4 -4
  66. package/lib/module/config/api.js +2 -2
  67. package/lib/module/hooks/useConnections.js +6 -6
  68. package/lib/module/hooks/useUserConnections.js +10 -10
  69. package/lib/module/index.js +8 -10
  70. package/lib/module/index.js.map +1 -1
  71. package/lib/module/services/apiClient.js +35 -35
  72. package/lib/module/services/apiKeyService.js +99 -99
  73. package/lib/module/services/authService.js +82 -82
  74. package/lib/module/services/biometricPinService.js +10 -10
  75. package/lib/module/services/connectedAccountsService.js +32 -32
  76. package/lib/module/services/googleAuthService.js +15 -15
  77. package/lib/module/services/imageCompressionService.js +15 -15
  78. package/lib/module/services/jwtStorageService.js +59 -59
  79. package/lib/module/services/mobileTrainingService.js +14 -14
  80. package/lib/module/services/pinEncryptionService.js +10 -10
  81. package/lib/module/services/pinStorageUtils.js +15 -15
  82. package/lib/module/services/platformAuthService.js +47 -47
  83. package/lib/module/services/storageService.js +31 -31
  84. package/lib/module/services/trainingApiHelpers.js +33 -33
  85. package/lib/module/services/userConnectionsService.js +24 -24
  86. package/lib/module/utils/Portal.js +4 -4
  87. package/lib/module/utils/api.js +24 -24
  88. package/lib/module/utils/auth.js +18 -18
  89. package/lib/module/utils/crypto.js +13 -13
  90. package/lib/module/utils/encryption.js +12 -12
  91. package/lib/module/utils/eventUtils.js +52 -52
  92. package/lib/module/utils/programmaticFlow.js +16 -16
  93. package/lib/module/utils/retryHelper.js +27 -27
  94. package/lib/typescript/components/OnairosButton.d.ts +37 -0
  95. package/lib/typescript/components/OnairosButton.d.ts.map +1 -0
  96. package/lib/typescript/components/OnairosSignInButton.d.ts +2 -1
  97. package/lib/typescript/components/OnairosSignInButton.d.ts.map +1 -1
  98. package/lib/typescript/index.d.ts +3 -4
  99. package/lib/typescript/index.d.ts.map +1 -1
  100. package/package.json +163 -163
  101. package/src/api/index.ts +151 -151
  102. package/src/assets/images/Checkbox.svg +3 -3
  103. package/src/assets/images/EnochE.svg +19 -19
  104. package/src/assets/images/Personalityprofile.svg +3 -3
  105. package/src/assets/images/Personalitytraits.svg +3 -3
  106. package/src/assets/images/Userpreferences.svg +3 -3
  107. package/src/assets/images/arrow.svg +20 -20
  108. package/src/assets/images/basicproficon.svg +43 -43
  109. package/src/assets/images/basicprofile.svg +3 -3
  110. package/src/assets/images/checkmark.svg +4 -4
  111. package/src/assets/images/contentanalysis.svg +3 -3
  112. package/src/assets/images/contenticon.svg +23 -23
  113. package/src/assets/images/personalityicon.svg +18 -18
  114. package/src/assets/images/x-close.svg +3 -3
  115. package/src/components/BodyText.tsx +33 -33
  116. package/src/components/BrandMark.tsx +62 -62
  117. package/src/components/CodeInput.tsx +32 -32
  118. package/src/components/DataRequestScreen.tsx +355 -355
  119. package/src/components/EmailInput.tsx +31 -31
  120. package/src/components/EmailVerificationModal.tsx +363 -363
  121. package/src/components/ExistingUserDataConfirmation.tsx +506 -506
  122. package/src/components/GoogleButton.tsx +55 -55
  123. package/src/components/HeadingGroup.tsx +49 -49
  124. package/src/components/ModalHeader.tsx +125 -125
  125. package/src/components/ModalSheet.tsx +57 -57
  126. package/src/components/Onairos.tsx +422 -422
  127. package/src/components/OnairosButton.tsx +339 -0
  128. package/src/components/OnairosSignInButton.tsx +30 -10
  129. package/src/components/Overlay.tsx +506 -506
  130. package/src/components/PersonaImage.tsx +79 -79
  131. package/src/components/PersonaLoadingScreen.tsx +201 -201
  132. package/src/components/PersonalizationConsentScreen.tsx +410 -410
  133. package/src/components/PinCreationScreen.tsx +492 -492
  134. package/src/components/PinInput.tsx +555 -555
  135. package/src/components/PlatformConnectorsStep.tsx +891 -891
  136. package/src/components/PlatformList.tsx +144 -144
  137. package/src/components/PlatformToggle.tsx +226 -226
  138. package/src/components/PrimaryButton.tsx +213 -213
  139. package/src/components/SignInMatchAnimation.tsx +225 -225
  140. package/src/components/SignInStep.tsx +217 -217
  141. package/src/components/TrainingModal.tsx +1047 -1047
  142. package/src/components/UniversalOnboarding.tsx +2887 -2887
  143. package/src/components/VerificationStep.tsx +198 -198
  144. package/src/components/WelcomeScreen.tsx +473 -473
  145. package/src/components/icons/Basicproficon.tsx +30 -30
  146. package/src/components/icons/Basicprofile.tsx +17 -17
  147. package/src/components/icons/Checkbox.tsx +17 -17
  148. package/src/components/icons/Checkmark.tsx +24 -24
  149. package/src/components/icons/Contentanalysis.tsx +17 -17
  150. package/src/components/icons/Contenticon.tsx +30 -30
  151. package/src/components/icons/EnochE.tsx +39 -39
  152. package/src/components/icons/Personalityicon.tsx +22 -22
  153. package/src/components/icons/Personalityprofile.tsx +17 -17
  154. package/src/components/icons/Personalitytraits.tsx +17 -17
  155. package/src/components/icons/Userpreferences.tsx +17 -17
  156. package/src/components/icons/index.ts +12 -12
  157. package/src/components/onboarding/OAuthWebView.tsx +232 -232
  158. package/src/config/api.ts +25 -25
  159. package/src/context/AuthContext.tsx +393 -393
  160. package/src/hooks/useConnectedAccounts.ts +138 -138
  161. package/src/hooks/useConnections.ts +161 -161
  162. package/src/hooks/useCredentials.ts +174 -174
  163. package/src/hooks/useUserConnections.ts +165 -165
  164. package/src/index.js +14 -0
  165. package/src/index.ts +94 -96
  166. package/src/services/apiClient.ts +336 -336
  167. package/src/services/apiKeyService.ts +919 -919
  168. package/src/services/authService.ts +1008 -1008
  169. package/src/services/biometricPinService.ts +192 -192
  170. package/src/services/connectedAccountsService.ts +289 -289
  171. package/src/services/googleAuthService.ts +279 -279
  172. package/src/services/imageCompressionService.ts +302 -302
  173. package/src/services/jwtStorageService.ts +256 -256
  174. package/src/services/mobileTrainingService.ts +203 -203
  175. package/src/services/pinEncryptionService.ts +75 -75
  176. package/src/services/pinStorageUtils.ts +96 -96
  177. package/src/services/platformAuthService.ts +1346 -1346
  178. package/src/services/storageService.ts +451 -451
  179. package/src/services/trainingApiHelpers.ts +66 -66
  180. package/src/services/userConnectionsService.ts +556 -556
  181. package/src/services/youtubeMigrationService.ts +453 -453
  182. package/src/theme/index.ts +239 -239
  183. package/src/types/ambient.d.ts +28 -28
  184. package/src/types/index.ts +265 -265
  185. package/src/types/node-fix.d.ts +18 -18
  186. package/src/types/node-override.d.ts +23 -23
  187. package/src/types/opacity.d.ts +15 -15
  188. package/src/types/types.d.ts +17 -17
  189. package/src/utils/Portal.tsx +82 -82
  190. package/src/utils/api.js +111 -111
  191. package/src/utils/auth.js +103 -103
  192. package/src/utils/crypto.js +59 -59
  193. package/src/utils/encryption.ts +68 -68
  194. package/src/utils/eventUtils.ts +302 -302
  195. package/src/utils/haptics.ts +58 -58
  196. package/src/utils/imagePreloader.ts +2 -2
  197. package/src/utils/programmaticFlow.ts +112 -112
  198. 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
  };