@onairos/react-native 3.1.16 → 3.1.18

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