feat: Add user registration component and authentication services

- Implemented Register component for user sign-up with form validation and network status handling.
- Created authAPI service for handling user login and registration with online/offline support.
- Developed localStorage service for managing user data and sessions offline.
- Introduced networkService for detecting online/offline status and managing connectivity.
- Defined authentication types for requests and responses.
- Added TypeScript configuration for the project.
This commit is contained in:
2025-12-11 23:33:43 +05:30
parent a7aa9e53a7
commit 801725edf3
75 changed files with 15734 additions and 0 deletions

2
.bundle/config Normal file
View File

@@ -0,0 +1,2 @@
BUNDLE_PATH: "vendor/bundle"
BUNDLE_FORCE_RUBY_PLATFORM: 1

4
.eslintrc.js Normal file
View File

@@ -0,0 +1,4 @@
module.exports = {
root: true,
extends: '@react-native',
};

75
.gitignore vendored Normal file
View File

@@ -0,0 +1,75 @@
# OSX
#
.DS_Store
# Xcode
#
build/
*.pbxuser
!default.pbxuser
*.mode1v3
!default.mode1v3
*.mode2v3
!default.mode2v3
*.perspectivev3
!default.perspectivev3
xcuserdata
*.xccheckout
*.moved-aside
DerivedData
*.hmap
*.ipa
*.xcuserstate
**/.xcode.env.local
# Android/IntelliJ
#
build/
.idea
.gradle
local.properties
*.iml
*.hprof
.cxx/
*.keystore
!debug.keystore
.kotlin/
# node.js
#
node_modules/
npm-debug.log
yarn-error.log
# fastlane
#
# It is recommended to not store the screenshots in the git repo. Instead, use fastlane to re-generate the
# screenshots whenever they are needed.
# For more information about the recommended setup visit:
# https://docs.fastlane.tools/best-practices/source-control/
**/fastlane/report.xml
**/fastlane/Preview.html
**/fastlane/screenshots
**/fastlane/test_output
# Bundle artifact
*.jsbundle
# Ruby / CocoaPods
**/Pods/
/vendor/bundle/
# Temporary files created by Metro to check the health of the file watcher
.metro-health-check*
# testing
/coverage
# Yarn
.yarn/*
!.yarn/patches
!.yarn/plugins
!.yarn/releases
!.yarn/sdks
!.yarn/versions

5
.prettierrc.js Normal file
View File

@@ -0,0 +1,5 @@
module.exports = {
arrowParens: 'avoid',
singleQuote: true,
trailingComma: 'all',
};

1
.watchmanconfig Normal file
View File

@@ -0,0 +1 @@
{}

254
App.tsx Normal file
View File

@@ -0,0 +1,254 @@
import {
StatusBar,
useColorScheme,
View,
StyleSheet,
Text,
TouchableOpacity,
Alert,
} from 'react-native';
import { SafeAreaProvider } from 'react-native-safe-area-context';
import { useState, useEffect } from 'react';
import Login from './src/components/Login';
import Register from './src/components/Register';
import { authAPI } from './src/services/authAPI';
import { networkService } from './src/services/networkService';
type Screen = 'login' | 'register' | 'home';
function App() {
const isDarkMode = useColorScheme() === 'dark';
const [currentScreen, setCurrentScreen] = useState<Screen>('login');
const [isInitialized, setIsInitialized] = useState(false);
const [isOnline, setIsOnline] = useState(true);
const [currentUser, setCurrentUser] = useState<any>(null);
// Initialize app
useEffect(() => {
initializeApp();
// Listen for network changes
const unsubscribe = networkService.addListener((networkState) => {
setIsOnline(networkState.isConnected);
});
return unsubscribe;
}, []);
const initializeApp = async () => {
try {
console.log('Initializing app...');
// Initialize authentication system
await authAPI.initialize();
// Check if user is already logged in
const isLoggedIn = await authAPI.isLoggedIn();
if (isLoggedIn) {
const user = await authAPI.getCurrentUser();
setCurrentUser(user);
setCurrentScreen('home');
}
setIsInitialized(true);
console.log('App initialized successfully');
} catch (error) {
console.error('App initialization error:', error);
setIsInitialized(true); // Continue anyway
}
};
const navigateToLogin = () => {
setCurrentUser(null);
setCurrentScreen('login');
};
const navigateToRegister = () => setCurrentScreen('register');
const navigateToHome = async () => {
const user = await authAPI.getCurrentUser();
setCurrentUser(user);
setCurrentScreen('home');
};
const handleLogout = async () => {
try {
await authAPI.logout();
navigateToLogin();
Alert.alert('Success', 'Logged out successfully');
} catch (error) {
console.error('Logout error:', error);
Alert.alert('Error', 'Failed to logout');
}
};
const showAppStatus = async () => {
try {
const status = await authAPI.getAppStatus();
const statusMessage = `
Network: ${status.network.isOnline ? 'Online' : 'Offline'}
Users: ${status.storage.totalUsers}
Current User: ${status.authentication.currentUser || 'None'}
Logged In: ${status.authentication.isLoggedIn ? 'Yes' : 'No'}
`.trim();
Alert.alert('App Status', statusMessage);
} catch (error) {
Alert.alert('Error', 'Failed to get app status');
}
};
const renderScreen = () => {
switch (currentScreen) {
case 'login':
return (
<Login
onNavigateToRegister={navigateToRegister}
onLoginSuccess={navigateToHome}
/>
);
case 'register':
return (
<Register
onNavigateToLogin={navigateToLogin}
onRegisterSuccess={navigateToHome}
/>
);
case 'home':
return (
<View style={styles.homeContainer}>
<View style={styles.statusBar}>
<Text style={styles.statusText}>
{isOnline ? '🟢 Online' : '🔴 Offline'}
</Text>
</View>
<Text style={styles.welcomeText}>
Welcome, {currentUser?.fullName || 'User'}!
</Text>
<Text style={styles.emailText}>
{currentUser?.email}
</Text>
<View style={styles.buttonContainer}>
<TouchableOpacity style={styles.statusButton} onPress={showAppStatus}>
<Text style={styles.buttonText}>App Status</Text>
</TouchableOpacity>
<TouchableOpacity style={styles.logoutButton} onPress={handleLogout}>
<Text style={styles.buttonText}>Logout</Text>
</TouchableOpacity>
</View>
<Text style={styles.infoText}>
{isOnline
? 'Your data is synced with the cloud'
: 'Working offline - data saved locally'}
</Text>
</View>
);
default:
return null;
}
};
// Show loading screen while initializing
if (!isInitialized) {
return (
<SafeAreaProvider>
<StatusBar barStyle={isDarkMode ? 'light-content' : 'dark-content'} />
<View style={styles.loadingContainer}>
<Text style={styles.loadingText}>Initializing...</Text>
</View>
</SafeAreaProvider>
);
}
return (
<SafeAreaProvider>
<StatusBar barStyle={isDarkMode ? 'light-content' : 'dark-content'} />
{renderScreen()}
</SafeAreaProvider>
);
}
const styles = StyleSheet.create({
loadingContainer: {
flex: 1,
justifyContent: 'center',
alignItems: 'center',
backgroundColor: '#f0f0f0',
},
loadingText: {
fontSize: 18,
color: '#3bb6d8',
fontWeight: '600',
},
homeContainer: {
flex: 1,
justifyContent: 'center',
alignItems: 'center',
backgroundColor: '#f0f0f0',
padding: 20,
},
statusBar: {
position: 'absolute',
top: 60,
right: 20,
backgroundColor: 'white',
paddingHorizontal: 12,
paddingVertical: 6,
borderRadius: 20,
shadowColor: '#000',
shadowOffset: { width: 0, height: 2 },
shadowOpacity: 0.1,
shadowRadius: 4,
elevation: 3,
},
statusText: {
fontSize: 12,
fontWeight: '600',
},
welcomeText: {
fontSize: 28,
fontWeight: 'bold',
color: '#333',
marginBottom: 8,
textAlign: 'center',
},
emailText: {
fontSize: 16,
color: '#666',
marginBottom: 40,
},
buttonContainer: {
flexDirection: 'row',
gap: 15,
marginBottom: 30,
},
statusButton: {
backgroundColor: '#3bb6d8',
paddingHorizontal: 20,
paddingVertical: 12,
borderRadius: 25,
},
logoutButton: {
backgroundColor: '#ff6b6b',
paddingHorizontal: 20,
paddingVertical: 12,
borderRadius: 25,
},
buttonText: {
color: 'white',
fontWeight: '600',
fontSize: 16,
},
infoText: {
fontSize: 14,
color: '#888',
textAlign: 'center',
fontStyle: 'italic',
},
});
export default App;

16
Gemfile Normal file
View File

@@ -0,0 +1,16 @@
source 'https://rubygems.org'
# You may use http://rbenv.org/ or https://rvm.io/ to install and use this version
ruby ">= 2.6.10"
# Exclude problematic versions of cocoapods and activesupport that causes build failures.
gem 'cocoapods', '>= 1.13', '!= 1.15.0', '!= 1.15.1'
gem 'activesupport', '>= 6.1.7.5', '!= 7.1.0'
gem 'xcodeproj', '< 1.26.0'
gem 'concurrent-ruby', '< 1.3.4'
# Ruby 3.4.0 has removed some libraries from the standard library.
gem 'bigdecimal'
gem 'logger'
gem 'benchmark'
gem 'mutex_m'

111
Gemfile.lock Normal file
View File

@@ -0,0 +1,111 @@
GEM
remote: https://rubygems.org/
specs:
CFPropertyList (3.0.8)
activesupport (6.1.7.10)
concurrent-ruby (~> 1.0, >= 1.0.2)
i18n (>= 1.6, < 2)
minitest (>= 5.1)
tzinfo (~> 2.0)
zeitwerk (~> 2.3)
addressable (2.8.7)
public_suffix (>= 2.0.2, < 7.0)
algoliasearch (1.27.5)
httpclient (~> 2.8, >= 2.8.3)
json (>= 1.5.1)
atomos (0.1.3)
benchmark (0.5.0)
bigdecimal (3.3.1)
claide (1.1.0)
cocoapods (1.15.2)
addressable (~> 2.8)
claide (>= 1.0.2, < 2.0)
cocoapods-core (= 1.15.2)
cocoapods-deintegrate (>= 1.0.3, < 2.0)
cocoapods-downloader (>= 2.1, < 3.0)
cocoapods-plugins (>= 1.0.0, < 2.0)
cocoapods-search (>= 1.0.0, < 2.0)
cocoapods-trunk (>= 1.6.0, < 2.0)
cocoapods-try (>= 1.1.0, < 2.0)
colored2 (~> 3.1)
escape (~> 0.0.4)
fourflusher (>= 2.3.0, < 3.0)
gh_inspector (~> 1.0)
molinillo (~> 0.8.0)
nap (~> 1.0)
ruby-macho (>= 2.3.0, < 3.0)
xcodeproj (>= 1.23.0, < 2.0)
cocoapods-core (1.15.2)
activesupport (>= 5.0, < 8)
addressable (~> 2.8)
algoliasearch (~> 1.0)
concurrent-ruby (~> 1.1)
fuzzy_match (~> 2.0.4)
nap (~> 1.0)
netrc (~> 0.11)
public_suffix (~> 4.0)
typhoeus (~> 1.0)
cocoapods-deintegrate (1.0.5)
cocoapods-downloader (2.1)
cocoapods-plugins (1.0.0)
nap
cocoapods-search (1.0.1)
cocoapods-trunk (1.6.0)
nap (>= 0.8, < 2.0)
netrc (~> 0.11)
cocoapods-try (1.2.0)
colored2 (3.1.2)
concurrent-ruby (1.3.3)
escape (0.0.4)
ethon (0.15.0)
ffi (>= 1.15.0)
ffi (1.17.2)
fourflusher (2.3.1)
fuzzy_match (2.0.4)
gh_inspector (1.1.3)
httpclient (2.9.0)
mutex_m
i18n (1.14.7)
concurrent-ruby (~> 1.0)
json (2.7.6)
logger (1.7.0)
minitest (5.25.4)
molinillo (0.8.0)
mutex_m (0.3.0)
nanaimo (0.3.0)
nap (1.1.0)
netrc (0.11.0)
public_suffix (4.0.7)
rexml (3.4.4)
ruby-macho (2.5.1)
typhoeus (1.5.0)
ethon (>= 0.9.0, < 0.16.0)
tzinfo (2.0.6)
concurrent-ruby (~> 1.0)
xcodeproj (1.25.1)
CFPropertyList (>= 2.3.3, < 4.0)
atomos (~> 0.1.3)
claide (>= 1.0.2, < 2.0)
colored2 (~> 3.1)
nanaimo (~> 0.3.0)
rexml (>= 3.3.6, < 4.0)
zeitwerk (2.6.18)
PLATFORMS
ruby
DEPENDENCIES
activesupport (>= 6.1.7.5, != 7.1.0)
benchmark
bigdecimal
cocoapods (>= 1.13, != 1.15.1, != 1.15.0)
concurrent-ruby (< 1.3.4)
logger
mutex_m
xcodeproj (< 1.26.0)
RUBY VERSION
ruby 2.6.10p210
BUNDLED WITH
1.17.2

262
Login.tsx Normal file
View File

@@ -0,0 +1,262 @@
import React from 'react';
import {
Image,
ScrollView,
StatusBar,
StyleSheet,
View,
Text,
TextInput,
TouchableOpacity
} from 'react-native';
import { SafeAreaView } from 'react-native-safe-area-context';
export function Login() {
const [isVisible, setIsVisible] = React.useState(false);
const [isChecked, setIsChecked] = React.useState(false);
return (
<SafeAreaView style={styles.container}>
<StatusBar barStyle={'dark-content'} />
<ScrollView
contentContainerStyle={styles.scrollContainer}
bounces={false}
showsVerticalScrollIndicator={false}
>
<View style={styles.logoContainer}>
<View style={styles.logoWrapper}>
<Image
source={require('./assets/img/logo.png')}
style={styles.logo}
/>
</View>
</View>
<View style={styles.formContainer}>
<Text style={styles.title}>Login</Text>
<Text style={styles.subtitle}>
Welcome back. Enter your credentials to{'\n'}access your account
</Text>
<Text style={styles.label}>Email Address*</Text>
<View style={styles.inputContainer}>
<TextInput
placeholder="Enter Email"
style={styles.input}
placeholderTextColor="#A0A0A0"
/>
</View>
<Text style={styles.label}>Password*</Text>
<View style={styles.passwordContainer}>
<TextInput
placeholder="Password"
style={styles.input}
placeholderTextColor="#A0A0A0"
secureTextEntry={!isVisible}
/>
<TouchableOpacity
onPress={() => setIsVisible(!isVisible)}
style={styles.eyeButton}
>
<Image
source={require('./assets/img/View_light.png')}
style={[
styles.eyeIcon,
{ tintColor: isVisible ? '#FDA913' : '#707070' },
]}
/>
</TouchableOpacity>
</View>
<View style={styles.rowBetween}>
<View style={styles.checkboxRow}>
<TouchableOpacity
onPress={() => setIsChecked(!isChecked)}
style={styles.checkboxBorder}
>
<View
style={[
styles.checkboxFill,
{ backgroundColor: isChecked ? '#02364E' : 'transparent' },
]}
>
<Image
source={require('./assets/img/check.png')}
style={styles.checkIcon}
/>
</View>
</TouchableOpacity>
<Text style={styles.checkLabel}>Keep me signed in</Text>
</View>
<Text style={styles.forgotText}>Forget Password?</Text>
</View>
<TouchableOpacity style={styles.loginButton}>
<Text style={styles.loginText}>Login</Text>
</TouchableOpacity>
</View>
</ScrollView>
</SafeAreaView>
);
}
export const styles = StyleSheet.create({
container: {
flex: 1,
backgroundColor: "#f5f5f5",
justifyContent: "center",
},
scrollContainer: {
flexGrow: 1,
},
logoContainer: {
paddingHorizontal: 15,
paddingTop: 20,
paddingBottom: 20,
},
logoWrapper: {
width: "100%",
height: 50,
},
logo: {
width: "50%",
height: "100%",
resizeMode: "contain",
},
formContainer: {
flex: 1,
paddingHorizontal: 15,
},
title: {
color: "#090A0A",
fontFamily: "Gilroy-SemiBold",
fontSize: 32,
fontWeight: "400",
marginBottom: 10,
},
subtitle: {
color: "#090A0A",
fontFamily: "Gilroy-Regular",
fontSize: 16,
fontWeight: "400",
marginBottom: 30,
lineHeight: 20,
},
label: {
color: "#111111",
fontFamily: "Gilroy-Regular",
fontSize: 14,
marginBottom: 7,
},
inputContainer: {
marginBottom: 20,
height: 50,
borderRadius: 16,
borderWidth: 1,
borderColor: "#CFCFCF",
},
input: {
flex: 1,
paddingHorizontal: 15,
fontSize: 14,
fontFamily: "Gilroy-Regular",
color: "#111111",
},
passwordContainer: {
flexDirection: "row",
marginBottom: 20,
height: 50,
borderRadius: 16,
borderWidth: 1,
borderColor: "#CFCFCF",
},
eyeButton: {
justifyContent: "center",
alignItems: "center",
paddingHorizontal: 15,
},
eyeIcon: {
width: 20,
height: 20,
resizeMode: "contain",
},
rowBetween: {
flexDirection: "row",
justifyContent: "space-between",
alignItems: "center",
marginBottom: 40,
},
checkboxRow: {
flexDirection: "row",
alignItems: "center",
},
checkboxBorder: {
width: 15,
height: 15,
borderRadius: 2,
borderWidth: 1,
borderColor: "#CFCFCF",
justifyContent: "center",
alignItems: "center",
marginRight: 8,
},
checkboxFill: {
width: 15,
height: 15,
borderRadius: 2,
alignItems: "center",
justifyContent: "center",
},
checkIcon: {
width: "70%",
height: "70%",
resizeMode: "contain",
tintColor: "#ffffff",
},
checkLabel: {
fontFamily: "Gilroy-Regular",
fontSize: 14,
color: "#191D23",
},
forgotText: {
textDecorationLine: "underline",
color: "#111111",
fontFamily: "Gilroy-Regular",
},
loginButton: {
height: 50,
borderRadius: 16,
backgroundColor: "#FDA913",
justifyContent: "center",
alignItems: "center",
marginBottom: 20,
},
loginText: {
color: "#FFFFFF",
fontFamily: "Gilroy-SemiBold",
fontSize: 18,
},
});

13
__tests__/App.test.tsx Normal file
View File

@@ -0,0 +1,13 @@
/**
* @format
*/
import React from 'react';
import ReactTestRenderer from 'react-test-renderer';
import App from '../App';
test('renders correctly', async () => {
await ReactTestRenderer.act(() => {
ReactTestRenderer.create(<App />);
});
});

119
android/app/build.gradle Normal file
View File

@@ -0,0 +1,119 @@
apply plugin: "com.android.application"
apply plugin: "org.jetbrains.kotlin.android"
apply plugin: "com.facebook.react"
/**
* This is the configuration block to customize your React Native Android app.
* By default you don't need to apply any configuration, just uncomment the lines you need.
*/
react {
/* Folders */
// The root of your project, i.e. where "package.json" lives. Default is '../..'
// root = file("../../")
// The folder where the react-native NPM package is. Default is ../../node_modules/react-native
// reactNativeDir = file("../../node_modules/react-native")
// The folder where the react-native Codegen package is. Default is ../../node_modules/@react-native/codegen
// codegenDir = file("../../node_modules/@react-native/codegen")
// The cli.js file which is the React Native CLI entrypoint. Default is ../../node_modules/react-native/cli.js
// cliFile = file("../../node_modules/react-native/cli.js")
/* Variants */
// The list of variants to that are debuggable. For those we're going to
// skip the bundling of the JS bundle and the assets. By default is just 'debug'.
// If you add flavors like lite, prod, etc. you'll have to list your debuggableVariants.
// debuggableVariants = ["liteDebug", "prodDebug"]
/* Bundling */
// A list containing the node command and its flags. Default is just 'node'.
// nodeExecutableAndArgs = ["node"]
//
// The command to run when bundling. By default is 'bundle'
// bundleCommand = "ram-bundle"
//
// The path to the CLI configuration file. Default is empty.
// bundleConfig = file(../rn-cli.config.js)
//
// The name of the generated asset file containing your JS bundle
// bundleAssetName = "MyApplication.android.bundle"
//
// The entry file for bundle generation. Default is 'index.android.js' or 'index.js'
// entryFile = file("../js/MyApplication.android.js")
//
// A list of extra flags to pass to the 'bundle' commands.
// See https://github.com/react-native-community/cli/blob/main/docs/commands.md#bundle
// extraPackagerArgs = []
/* Hermes Commands */
// The hermes compiler command to run. By default it is 'hermesc'
// hermesCommand = "$rootDir/my-custom-hermesc/bin/hermesc"
//
// The list of flags to pass to the Hermes compiler. By default is "-O", "-output-source-map"
// hermesFlags = ["-O", "-output-source-map"]
/* Autolinking */
autolinkLibrariesWithApp()
}
/**
* Set this to true to Run Proguard on Release builds to minify the Java bytecode.
*/
def enableProguardInReleaseBuilds = false
/**
* The preferred build flavor of JavaScriptCore (JSC)
*
* For example, to use the international variant, you can use:
* `def jscFlavor = io.github.react-native-community:jsc-android-intl:2026004.+`
*
* The international variant includes ICU i18n library and necessary data
* allowing to use e.g. `Date.toLocaleString` and `String.localeCompare` that
* give correct results when using with locales other than en-US. Note that
* this variant is about 6MiB larger per architecture than default.
*/
def jscFlavor = 'io.github.react-native-community:jsc-android:2026004.+'
android {
ndkVersion rootProject.ext.ndkVersion
buildToolsVersion rootProject.ext.buildToolsVersion
compileSdk rootProject.ext.compileSdkVersion
namespace "com.lynkeduppro"
defaultConfig {
applicationId "com.lynkeduppro"
minSdkVersion rootProject.ext.minSdkVersion
targetSdkVersion rootProject.ext.targetSdkVersion
versionCode 1
versionName "1.0"
}
signingConfigs {
debug {
storeFile file('debug.keystore')
storePassword 'android'
keyAlias 'androiddebugkey'
keyPassword 'android'
}
}
buildTypes {
debug {
signingConfig signingConfigs.debug
}
release {
// Caution! In production, you need to generate your own keystore file.
// see https://reactnative.dev/docs/signed-apk-android.
signingConfig signingConfigs.debug
minifyEnabled enableProguardInReleaseBuilds
proguardFiles getDefaultProguardFile("proguard-android.txt"), "proguard-rules.pro"
}
}
}
dependencies {
// The version of react-native is set by the React Native Gradle Plugin
implementation("com.facebook.react:react-android")
if (hermesEnabled.toBoolean()) {
implementation("com.facebook.react:hermes-android")
} else {
implementation jscFlavor
}
}

BIN
android/app/debug.keystore Normal file

Binary file not shown.

10
android/app/proguard-rules.pro vendored Normal file
View File

@@ -0,0 +1,10 @@
# Add project specific ProGuard rules here.
# By default, the flags in this file are appended to flags specified
# in /usr/local/Cellar/android-sdk/24.3.3/tools/proguard/proguard-android.txt
# You can edit the include path and order by changing the proguardFiles
# directive in build.gradle.
#
# For more details, see
# http://developer.android.com/guide/developing/tools/proguard.html
# Add any project specific keep options here:

View File

@@ -0,0 +1,27 @@
<manifest xmlns:android="http://schemas.android.com/apk/res/android">
<uses-permission android:name="android.permission.INTERNET" />
<application
android:name=".MainApplication"
android:label="@string/app_name"
android:icon="@mipmap/ic_launcher"
android:roundIcon="@mipmap/ic_launcher_round"
android:allowBackup="false"
android:theme="@style/AppTheme"
android:usesCleartextTraffic="${usesCleartextTraffic}"
android:supportsRtl="true">
<activity
android:name=".MainActivity"
android:label="@string/app_name"
android:configChanges="keyboard|keyboardHidden|orientation|screenLayout|screenSize|smallestScreenSize|uiMode"
android:launchMode="singleTask"
android:windowSoftInputMode="adjustResize"
android:exported="true">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
</manifest>

Binary file not shown.

Binary file not shown.

Binary file not shown.

View File

@@ -0,0 +1,22 @@
package com.lynkeduppro
import com.facebook.react.ReactActivity
import com.facebook.react.ReactActivityDelegate
import com.facebook.react.defaults.DefaultNewArchitectureEntryPoint.fabricEnabled
import com.facebook.react.defaults.DefaultReactActivityDelegate
class MainActivity : ReactActivity() {
/**
* Returns the name of the main component registered from JavaScript. This is used to schedule
* rendering of the component.
*/
override fun getMainComponentName(): String = "LynkedUpPro"
/**
* Returns the instance of the [ReactActivityDelegate]. We use [DefaultReactActivityDelegate]
* which allows you to enable New Architecture with a single boolean flags [fabricEnabled]
*/
override fun createReactActivityDelegate(): ReactActivityDelegate =
DefaultReactActivityDelegate(this, mainComponentName, fabricEnabled)
}

View File

@@ -0,0 +1,27 @@
package com.lynkeduppro
import android.app.Application
import com.facebook.react.PackageList
import com.facebook.react.ReactApplication
import com.facebook.react.ReactHost
import com.facebook.react.ReactNativeApplicationEntryPoint.loadReactNative
import com.facebook.react.defaults.DefaultReactHost.getDefaultReactHost
class MainApplication : Application(), ReactApplication {
override val reactHost: ReactHost by lazy {
getDefaultReactHost(
context = applicationContext,
packageList =
PackageList(this).packages.apply {
// Packages that cannot be autolinked yet can be added manually here, for example:
// add(MyReactNativePackage())
},
)
}
override fun onCreate() {
super.onCreate()
loadReactNative(this)
}
}

View File

@@ -0,0 +1,37 @@
<?xml version="1.0" encoding="utf-8"?>
<!-- Copyright (C) 2014 The Android Open Source Project
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
-->
<inset xmlns:android="http://schemas.android.com/apk/res/android"
android:insetLeft="@dimen/abc_edit_text_inset_horizontal_material"
android:insetRight="@dimen/abc_edit_text_inset_horizontal_material"
android:insetTop="@dimen/abc_edit_text_inset_top_material"
android:insetBottom="@dimen/abc_edit_text_inset_bottom_material"
>
<selector>
<!--
This file is a copy of abc_edit_text_material (https://bit.ly/3k8fX7I).
The item below with state_pressed="false" and state_focused="false" causes a NullPointerException.
NullPointerException:tempt to invoke virtual method 'android.graphics.drawable.Drawable android.graphics.drawable.Drawable$ConstantState.newDrawable(android.content.res.Resources)'
<item android:state_pressed="false" android:state_focused="false" android:drawable="@drawable/abc_textfield_default_mtrl_alpha"/>
For more info, see https://bit.ly/3CdLStv (react-native/pull/29452) and https://bit.ly/3nxOMoR.
-->
<item android:state_enabled="false" android:drawable="@drawable/abc_textfield_default_mtrl_alpha"/>
<item android:drawable="@drawable/abc_textfield_activated_mtrl_alpha"/>
</selector>
</inset>

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.0 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.9 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.0 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.8 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.5 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 6.9 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 6.3 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 10 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 9.0 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 15 KiB

View File

@@ -0,0 +1,3 @@
<resources>
<string name="app_name">LynkedUpPro</string>
</resources>

View File

@@ -0,0 +1,9 @@
<resources>
<!-- Base application theme. -->
<style name="AppTheme" parent="Theme.AppCompat.DayNight.NoActionBar">
<!-- Customize your theme here. -->
<item name="android:editTextBackground">@drawable/rn_edit_text_material</item>
</style>
</resources>

21
android/build.gradle Normal file
View File

@@ -0,0 +1,21 @@
buildscript {
ext {
buildToolsVersion = "36.0.0"
minSdkVersion = 24
compileSdkVersion = 36
targetSdkVersion = 36
ndkVersion = "27.1.12297006"
kotlinVersion = "2.1.20"
}
repositories {
google()
mavenCentral()
}
dependencies {
classpath("com.android.tools.build:gradle")
classpath("com.facebook.react:react-native-gradle-plugin")
classpath("org.jetbrains.kotlin:kotlin-gradle-plugin")
}
}
apply plugin: "com.facebook.react.rootproject"

44
android/gradle.properties Normal file
View File

@@ -0,0 +1,44 @@
# Project-wide Gradle settings.
# IDE (e.g. Android Studio) users:
# Gradle settings configured through the IDE *will override*
# any settings specified in this file.
# For more details on how to configure your build environment visit
# http://www.gradle.org/docs/current/userguide/build_environment.html
# Specifies the JVM arguments used for the daemon process.
# The setting is particularly useful for tweaking memory settings.
# Default value: -Xmx512m -XX:MaxMetaspaceSize=256m
org.gradle.jvmargs=-Xmx2048m -XX:MaxMetaspaceSize=512m
# When configured, Gradle will run in incubating parallel mode.
# This option should only be used with decoupled projects. More details, visit
# http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects
# org.gradle.parallel=true
# AndroidX package structure to make it clearer which packages are bundled with the
# Android operating system, and which are packaged with your app's APK
# https://developer.android.com/topic/libraries/support-library/androidx-rn
android.useAndroidX=true
# Use this property to specify which architecture you want to build.
# You can also override it from the CLI using
# ./gradlew <task> -PreactNativeArchitectures=x86_64
reactNativeArchitectures=armeabi-v7a,arm64-v8a,x86,x86_64
# Use this property to enable support to the new architecture.
# This will allow you to use TurboModules and the Fabric render in
# your application. You should enable this flag either if you want
# to write custom TurboModules/Fabric components OR use libraries that
# are providing them.
newArchEnabled=true
# Use this property to enable or disable the Hermes JS engine.
# If set to false, you will be using JSC instead.
hermesEnabled=true
# Use this property to enable edge-to-edge display support.
# This allows your app to draw behind system bars for an immersive UI.
# Note: Only works with ReactActivity and should not be used with custom Activity.
edgeToEdgeEnabled=false

Binary file not shown.

View File

@@ -0,0 +1,7 @@
distributionBase=GRADLE_USER_HOME
distributionPath=wrapper/dists
distributionUrl=https\://services.gradle.org/distributions/gradle-9.0.0-bin.zip
networkTimeout=10000
validateDistributionUrl=true
zipStoreBase=GRADLE_USER_HOME
zipStorePath=wrapper/dists

251
android/gradlew vendored Executable file
View File

@@ -0,0 +1,251 @@
#!/bin/sh
#
# Copyright © 2015 the original authors.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# https://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
# SPDX-License-Identifier: Apache-2.0
#
##############################################################################
#
# Gradle start up script for POSIX generated by Gradle.
#
# Important for running:
#
# (1) You need a POSIX-compliant shell to run this script. If your /bin/sh is
# noncompliant, but you have some other compliant shell such as ksh or
# bash, then to run this script, type that shell name before the whole
# command line, like:
#
# ksh Gradle
#
# Busybox and similar reduced shells will NOT work, because this script
# requires all of these POSIX shell features:
# * functions;
# * expansions «$var», «${var}», «${var:-default}», «${var+SET}»,
# «${var#prefix}», «${var%suffix}», and «$( cmd )»;
# * compound commands having a testable exit status, especially «case»;
# * various built-in commands including «command», «set», and «ulimit».
#
# Important for patching:
#
# (2) This script targets any POSIX shell, so it avoids extensions provided
# by Bash, Ksh, etc; in particular arrays are avoided.
#
# The "traditional" practice of packing multiple parameters into a
# space-separated string is a well documented source of bugs and security
# problems, so this is (mostly) avoided, by progressively accumulating
# options in "$@", and eventually passing that to Java.
#
# Where the inherited environment variables (DEFAULT_JVM_OPTS, JAVA_OPTS,
# and GRADLE_OPTS) rely on word-splitting, this is performed explicitly;
# see the in-line comments for details.
#
# There are tweaks for specific operating systems such as AIX, CygWin,
# Darwin, MinGW, and NonStop.
#
# (3) This script is generated from the Groovy template
# https://github.com/gradle/gradle/blob/HEAD/platforms/jvm/plugins-application/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt
# within the Gradle project.
#
# You can find Gradle at https://github.com/gradle/gradle/.
#
##############################################################################
# Attempt to set APP_HOME
# Resolve links: $0 may be a link
app_path=$0
# Need this for daisy-chained symlinks.
while
APP_HOME=${app_path%"${app_path##*/}"} # leaves a trailing /; empty if no leading path
[ -h "$app_path" ]
do
ls=$( ls -ld "$app_path" )
link=${ls#*' -> '}
case $link in #(
/*) app_path=$link ;; #(
*) app_path=$APP_HOME$link ;;
esac
done
# This is normally unused
# shellcheck disable=SC2034
APP_BASE_NAME=${0##*/}
# Discard cd standard output in case $CDPATH is set (https://github.com/gradle/gradle/issues/25036)
APP_HOME=$( cd -P "${APP_HOME:-./}" > /dev/null && printf '%s\n' "$PWD" ) || exit
# Use the maximum available, or set MAX_FD != -1 to use that value.
MAX_FD=maximum
warn () {
echo "$*"
} >&2
die () {
echo
echo "$*"
echo
exit 1
} >&2
# OS specific support (must be 'true' or 'false').
cygwin=false
msys=false
darwin=false
nonstop=false
case "$( uname )" in #(
CYGWIN* ) cygwin=true ;; #(
Darwin* ) darwin=true ;; #(
MSYS* | MINGW* ) msys=true ;; #(
NONSTOP* ) nonstop=true ;;
esac
CLASSPATH="\\\"\\\""
# Determine the Java command to use to start the JVM.
if [ -n "$JAVA_HOME" ] ; then
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
# IBM's JDK on AIX uses strange locations for the executables
JAVACMD=$JAVA_HOME/jre/sh/java
else
JAVACMD=$JAVA_HOME/bin/java
fi
if [ ! -x "$JAVACMD" ] ; then
die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
Please set the JAVA_HOME variable in your environment to match the
location of your Java installation."
fi
else
JAVACMD=java
if ! command -v java >/dev/null 2>&1
then
die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
Please set the JAVA_HOME variable in your environment to match the
location of your Java installation."
fi
fi
# Increase the maximum file descriptors if we can.
if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then
case $MAX_FD in #(
max*)
# In POSIX sh, ulimit -H is undefined. That's why the result is checked to see if it worked.
# shellcheck disable=SC2039,SC3045
MAX_FD=$( ulimit -H -n ) ||
warn "Could not query maximum file descriptor limit"
esac
case $MAX_FD in #(
'' | soft) :;; #(
*)
# In POSIX sh, ulimit -n is undefined. That's why the result is checked to see if it worked.
# shellcheck disable=SC2039,SC3045
ulimit -n "$MAX_FD" ||
warn "Could not set maximum file descriptor limit to $MAX_FD"
esac
fi
# Collect all arguments for the java command, stacking in reverse order:
# * args from the command line
# * the main class name
# * -classpath
# * -D...appname settings
# * --module-path (only if needed)
# * DEFAULT_JVM_OPTS, JAVA_OPTS, and GRADLE_OPTS environment variables.
# For Cygwin or MSYS, switch paths to Windows format before running java
if "$cygwin" || "$msys" ; then
APP_HOME=$( cygpath --path --mixed "$APP_HOME" )
CLASSPATH=$( cygpath --path --mixed "$CLASSPATH" )
JAVACMD=$( cygpath --unix "$JAVACMD" )
# Now convert the arguments - kludge to limit ourselves to /bin/sh
for arg do
if
case $arg in #(
-*) false ;; # don't mess with options #(
/?*) t=${arg#/} t=/${t%%/*} # looks like a POSIX filepath
[ -e "$t" ] ;; #(
*) false ;;
esac
then
arg=$( cygpath --path --ignore --mixed "$arg" )
fi
# Roll the args list around exactly as many times as the number of
# args, so each arg winds up back in the position where it started, but
# possibly modified.
#
# NB: a `for` loop captures its iteration list before it begins, so
# changing the positional parameters here affects neither the number of
# iterations, nor the values presented in `arg`.
shift # remove old arg
set -- "$@" "$arg" # push replacement arg
done
fi
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"'
# Collect all arguments for the java command:
# * DEFAULT_JVM_OPTS, JAVA_OPTS, and optsEnvironmentVar are not allowed to contain shell fragments,
# and any embedded shellness will be escaped.
# * For example: A user cannot expect ${Hostname} to be expanded, as it is an environment variable and will be
# treated as '${Hostname}' itself on the command line.
set -- \
"-Dorg.gradle.appname=$APP_BASE_NAME" \
-classpath "$CLASSPATH" \
-jar "$APP_HOME/gradle/wrapper/gradle-wrapper.jar" \
"$@"
# Stop when "xargs" is not available.
if ! command -v xargs >/dev/null 2>&1
then
die "xargs is not available"
fi
# Use "xargs" to parse quoted args.
#
# With -n1 it outputs one arg per line, with the quotes and backslashes removed.
#
# In Bash we could simply go:
#
# readarray ARGS < <( xargs -n1 <<<"$var" ) &&
# set -- "${ARGS[@]}" "$@"
#
# but POSIX shell has neither arrays nor command substitution, so instead we
# post-process each arg (as a line of input to sed) to backslash-escape any
# character that might be a shell metacharacter, then use eval to reverse
# that process (while maintaining the separation between arguments), and wrap
# the whole thing up as a single "set" statement.
#
# This will of course break if any of these variables contains a newline or
# an unmatched quote.
#
eval "set -- $(
printf '%s\n' "$DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS" |
xargs -n1 |
sed ' s~[^-[:alnum:]+,./:=@_]~\\&~g; ' |
tr '\n' ' '
)" '"$@"'
exec "$JAVACMD" "$@"

99
android/gradlew.bat vendored Normal file
View File

@@ -0,0 +1,99 @@
@REM Copyright (c) Meta Platforms, Inc. and affiliates.
@REM
@REM This source code is licensed under the MIT license found in the
@REM LICENSE file in the root directory of this source tree.
@rem
@rem Copyright 2015 the original author or authors.
@rem
@rem Licensed under the Apache License, Version 2.0 (the "License");
@rem you may not use this file except in compliance with the License.
@rem You may obtain a copy of the License at
@rem
@rem https://www.apache.org/licenses/LICENSE-2.0
@rem
@rem Unless required by applicable law or agreed to in writing, software
@rem distributed under the License is distributed on an "AS IS" BASIS,
@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
@rem See the License for the specific language governing permissions and
@rem limitations under the License.
@rem
@rem SPDX-License-Identifier: Apache-2.0
@rem
@if "%DEBUG%"=="" @echo off
@rem ##########################################################################
@rem
@rem Gradle startup script for Windows
@rem
@rem ##########################################################################
@rem Set local scope for the variables with windows NT shell
if "%OS%"=="Windows_NT" setlocal
set DIRNAME=%~dp0
if "%DIRNAME%"=="" set DIRNAME=.
@rem This is normally unused
set APP_BASE_NAME=%~n0
set APP_HOME=%DIRNAME%
@rem Resolve any "." and ".." in APP_HOME to make it shorter.
for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi
@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m"
@rem Find java.exe
if defined JAVA_HOME goto findJavaFromJavaHome
set JAVA_EXE=java.exe
%JAVA_EXE% -version >NUL 2>&1
if %ERRORLEVEL% equ 0 goto execute
echo. 1>&2
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 1>&2
echo. 1>&2
echo Please set the JAVA_HOME variable in your environment to match the 1>&2
echo location of your Java installation. 1>&2
goto fail
:findJavaFromJavaHome
set JAVA_HOME=%JAVA_HOME:"=%
set JAVA_EXE=%JAVA_HOME%/bin/java.exe
if exist "%JAVA_EXE%" goto execute
echo. 1>&2
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% 1>&2
echo. 1>&2
echo Please set the JAVA_HOME variable in your environment to match the 1>&2
echo location of your Java installation. 1>&2
goto fail
:execute
@rem Setup the command line
set CLASSPATH=
@rem Execute Gradle
"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" -jar "%APP_HOME%\gradle\wrapper\gradle-wrapper.jar" %*
:end
@rem End local scope for the variables with windows NT shell
if %ERRORLEVEL% equ 0 goto mainEnd
:fail
rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
rem the _cmd.exe /c_ return code!
set EXIT_CODE=%ERRORLEVEL%
if %EXIT_CODE% equ 0 set EXIT_CODE=1
if not ""=="%GRADLE_EXIT_CONSOLE%" exit %EXIT_CODE%
exit /b %EXIT_CODE%
:mainEnd
if "%OS%"=="Windows_NT" endlocal
:omega

View File

@@ -0,0 +1,17 @@
{
"migIndex": 1,
"data": [
{
"path": "assets/fonts/Gilroy-Bold.ttf",
"sha1": "7fe4d8cf5325dd47363d0f80fefb40ac7df5d5a6"
},
{
"path": "assets/fonts/Gilroy-Regular.ttf",
"sha1": "a737c19356e50cb9a50eb64ca7446c80dd3aa3e3"
},
{
"path": "assets/fonts/Gilroy-SemiBold.ttf",
"sha1": "8663a356eb9bc23b77696208f74376daf59687ea"
}
]
}

6
android/settings.gradle Normal file
View File

@@ -0,0 +1,6 @@
pluginManagement { includeBuild("../node_modules/@react-native/gradle-plugin") }
plugins { id("com.facebook.react.settings") }
extensions.configure(com.facebook.react.ReactSettingsExtension){ ex -> ex.autolinkLibrariesFromCommand() }
rootProject.name = 'LynkedUpPro'
include ':app'
includeBuild('../node_modules/@react-native/gradle-plugin')

4
app.json Normal file
View File

@@ -0,0 +1,4 @@
{
"name": "LynkedUpPro",
"displayName": "LynkedUpPro"
}

Binary file not shown.

Binary file not shown.

Binary file not shown.

BIN
assets/img/View_light.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 10 KiB

BIN
assets/img/check.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.4 KiB

BIN
assets/img/logo.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 8.9 KiB

3
babel.config.js Normal file
View File

@@ -0,0 +1,3 @@
module.exports = {
presets: ['module:@react-native/babel-preset'],
};

9
index.js Normal file
View File

@@ -0,0 +1,9 @@
/**
* @format
*/
import { AppRegistry, LogBox } from 'react-native';
import App from './App';
import { name as appName } from './app.json';
LogBox.ignoreAllLogs();
AppRegistry.registerComponent(appName, () => App);

11
ios/.xcode.env Normal file
View File

@@ -0,0 +1,11 @@
# This `.xcode.env` file is versioned and is used to source the environment
# used when running script phases inside Xcode.
# To customize your local environment, you can create an `.xcode.env.local`
# file that is not versioned.
# NODE_BINARY variable contains the PATH to the node executable.
#
# Customize the NODE_BINARY variable here.
# For example, to use nvm with brew, add the following line
# . "$(brew --prefix nvm)/nvm.sh" --no-use
export NODE_BINARY=$(command -v node)

View File

@@ -0,0 +1,492 @@
// !$*UTF8*$!
{
archiveVersion = 1;
classes = {
};
objectVersion = 54;
objects = {
/* Begin PBXBuildFile section */
0C80B921A6F3F58F76C31292 /* libPods-LynkedUpPro.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 5DCACB8F33CDC322A6C60F78 /* libPods-LynkedUpPro.a */; };
13B07FBF1A68108700A75B9A /* Images.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 13B07FB51A68108700A75B9A /* Images.xcassets */; };
761780ED2CA45674006654EE /* AppDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = 761780EC2CA45674006654EE /* AppDelegate.swift */; };
81AB9BB82411601600AC10FF /* LaunchScreen.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 81AB9BB72411601600AC10FF /* LaunchScreen.storyboard */; };
028DCFBD5C2D4A61AC400444 /* Gilroy-Bold.ttf in Resources */ = {isa = PBXBuildFile; fileRef = 3279BDA3B7CD4FD387291FE9 /* Gilroy-Bold.ttf */; };
F2C66068A3EB498DB8DE519F /* Gilroy-Regular.ttf in Resources */ = {isa = PBXBuildFile; fileRef = 81A4EC19CD3949FBA95B1C05 /* Gilroy-Regular.ttf */; };
7E0F78E2BBB347D184948BFA /* Gilroy-SemiBold.ttf in Resources */ = {isa = PBXBuildFile; fileRef = 00E8E7943DA44F86B055871C /* Gilroy-SemiBold.ttf */; };
/* End PBXBuildFile section */
/* Begin PBXFileReference section */
13B07F961A680F5B00A75B9A /* LynkedUpPro.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = LynkedUpPro.app; sourceTree = BUILT_PRODUCTS_DIR; };
13B07FB51A68108700A75B9A /* Images.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; name = Images.xcassets; path = LynkedUpPro/Images.xcassets; sourceTree = "<group>"; };
13B07FB61A68108700A75B9A /* Info.plist */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.xml; name = Info.plist; path = LynkedUpPro/Info.plist; sourceTree = "<group>"; };
13B07FB81A68108700A75B9A /* PrivacyInfo.xcprivacy */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.xml; name = PrivacyInfo.xcprivacy; path = LynkedUpPro/PrivacyInfo.xcprivacy; sourceTree = "<group>"; };
3B4392A12AC88292D35C810B /* Pods-LynkedUpPro.debug.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-LynkedUpPro.debug.xcconfig"; path = "Target Support Files/Pods-LynkedUpPro/Pods-LynkedUpPro.debug.xcconfig"; sourceTree = "<group>"; };
5709B34CF0A7D63546082F79 /* Pods-LynkedUpPro.release.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-LynkedUpPro.release.xcconfig"; path = "Target Support Files/Pods-LynkedUpPro/Pods-LynkedUpPro.release.xcconfig"; sourceTree = "<group>"; };
5DCACB8F33CDC322A6C60F78 /* libPods-LynkedUpPro.a */ = {isa = PBXFileReference; explicitFileType = archive.ar; includeInIndex = 0; path = "libPods-LynkedUpPro.a"; sourceTree = BUILT_PRODUCTS_DIR; };
761780EC2CA45674006654EE /* AppDelegate.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; name = AppDelegate.swift; path = LynkedUpPro/AppDelegate.swift; sourceTree = "<group>"; };
81AB9BB72411601600AC10FF /* LaunchScreen.storyboard */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = file.storyboard; name = LaunchScreen.storyboard; path = LynkedUpPro/LaunchScreen.storyboard; sourceTree = "<group>"; };
ED297162215061F000B7C4FE /* JavaScriptCore.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = JavaScriptCore.framework; path = System/Library/Frameworks/JavaScriptCore.framework; sourceTree = SDKROOT; };
3279BDA3B7CD4FD387291FE9 /* Gilroy-Bold.ttf */ = {isa = PBXFileReference; name = "Gilroy-Bold.ttf"; path = "../assets/fonts/Gilroy-Bold.ttf"; sourceTree = "<group>"; fileEncoding = undefined; lastKnownFileType = unknown; explicitFileType = undefined; includeInIndex = 0; };
81A4EC19CD3949FBA95B1C05 /* Gilroy-Regular.ttf */ = {isa = PBXFileReference; name = "Gilroy-Regular.ttf"; path = "../assets/fonts/Gilroy-Regular.ttf"; sourceTree = "<group>"; fileEncoding = undefined; lastKnownFileType = unknown; explicitFileType = undefined; includeInIndex = 0; };
00E8E7943DA44F86B055871C /* Gilroy-SemiBold.ttf */ = {isa = PBXFileReference; name = "Gilroy-SemiBold.ttf"; path = "../assets/fonts/Gilroy-SemiBold.ttf"; sourceTree = "<group>"; fileEncoding = undefined; lastKnownFileType = unknown; explicitFileType = undefined; includeInIndex = 0; };
/* End PBXFileReference section */
/* Begin PBXFrameworksBuildPhase section */
13B07F8C1A680F5B00A75B9A /* Frameworks */ = {
isa = PBXFrameworksBuildPhase;
buildActionMask = 2147483647;
files = (
0C80B921A6F3F58F76C31292 /* libPods-LynkedUpPro.a in Frameworks */,
);
runOnlyForDeploymentPostprocessing = 0;
};
/* End PBXFrameworksBuildPhase section */
/* Begin PBXGroup section */
13B07FAE1A68108700A75B9A /* LynkedUpPro */ = {
isa = PBXGroup;
children = (
13B07FB51A68108700A75B9A /* Images.xcassets */,
761780EC2CA45674006654EE /* AppDelegate.swift */,
13B07FB61A68108700A75B9A /* Info.plist */,
81AB9BB72411601600AC10FF /* LaunchScreen.storyboard */,
13B07FB81A68108700A75B9A /* PrivacyInfo.xcprivacy */,
);
name = LynkedUpPro;
sourceTree = "<group>";
};
2D16E6871FA4F8E400B85C8A /* Frameworks */ = {
isa = PBXGroup;
children = (
ED297162215061F000B7C4FE /* JavaScriptCore.framework */,
5DCACB8F33CDC322A6C60F78 /* libPods-LynkedUpPro.a */,
);
name = Frameworks;
sourceTree = "<group>";
};
832341AE1AAA6A7D00B99B32 /* Libraries */ = {
isa = PBXGroup;
children = (
);
name = Libraries;
sourceTree = "<group>";
};
83CBB9F61A601CBA00E9B192 = {
isa = PBXGroup;
children = (
13B07FAE1A68108700A75B9A /* LynkedUpPro */,
832341AE1AAA6A7D00B99B32 /* Libraries */,
83CBBA001A601CBA00E9B192 /* Products */,
2D16E6871FA4F8E400B85C8A /* Frameworks */,
BBD78D7AC51CEA395F1C20DB /* Pods */,
E0759812679E4780BA58D612 /* Resources */,
);
indentWidth = 2;
sourceTree = "<group>";
tabWidth = 2;
usesTabs = 0;
};
83CBBA001A601CBA00E9B192 /* Products */ = {
isa = PBXGroup;
children = (
13B07F961A680F5B00A75B9A /* LynkedUpPro.app */,
);
name = Products;
sourceTree = "<group>";
};
BBD78D7AC51CEA395F1C20DB /* Pods */ = {
isa = PBXGroup;
children = (
3B4392A12AC88292D35C810B /* Pods-LynkedUpPro.debug.xcconfig */,
5709B34CF0A7D63546082F79 /* Pods-LynkedUpPro.release.xcconfig */,
);
path = Pods;
sourceTree = "<group>";
};
E0759812679E4780BA58D612 /* Resources */ = {
isa = "PBXGroup";
children = (
3279BDA3B7CD4FD387291FE9 /* Gilroy-Bold.ttf */,
81A4EC19CD3949FBA95B1C05 /* Gilroy-Regular.ttf */,
00E8E7943DA44F86B055871C /* Gilroy-SemiBold.ttf */,
);
name = Resources;
sourceTree = "<group>";
path = "";
};
/* End PBXGroup section */
/* Begin PBXNativeTarget section */
13B07F861A680F5B00A75B9A /* LynkedUpPro */ = {
isa = PBXNativeTarget;
buildConfigurationList = 13B07F931A680F5B00A75B9A /* Build configuration list for PBXNativeTarget "LynkedUpPro" */;
buildPhases = (
C38B50BA6285516D6DCD4F65 /* [CP] Check Pods Manifest.lock */,
13B07F871A680F5B00A75B9A /* Sources */,
13B07F8C1A680F5B00A75B9A /* Frameworks */,
13B07F8E1A680F5B00A75B9A /* Resources */,
00DD1BFF1BD5951E006B06BC /* Bundle React Native code and images */,
00EEFC60759A1932668264C0 /* [CP] Embed Pods Frameworks */,
E235C05ADACE081382539298 /* [CP] Copy Pods Resources */,
);
buildRules = (
);
dependencies = (
);
name = LynkedUpPro;
productName = LynkedUpPro;
productReference = 13B07F961A680F5B00A75B9A /* LynkedUpPro.app */;
productType = "com.apple.product-type.application";
};
/* End PBXNativeTarget section */
/* Begin PBXProject section */
83CBB9F71A601CBA00E9B192 /* Project object */ = {
isa = PBXProject;
attributes = {
LastUpgradeCheck = 1210;
TargetAttributes = {
13B07F861A680F5B00A75B9A = {
LastSwiftMigration = 1120;
};
};
};
buildConfigurationList = 83CBB9FA1A601CBA00E9B192 /* Build configuration list for PBXProject "LynkedUpPro" */;
compatibilityVersion = "Xcode 12.0";
developmentRegion = en;
hasScannedForEncodings = 0;
knownRegions = (
en,
Base,
);
mainGroup = 83CBB9F61A601CBA00E9B192;
productRefGroup = 83CBBA001A601CBA00E9B192 /* Products */;
projectDirPath = "";
projectRoot = "";
targets = (
13B07F861A680F5B00A75B9A /* LynkedUpPro */,
);
};
/* End PBXProject section */
/* Begin PBXResourcesBuildPhase section */
13B07F8E1A680F5B00A75B9A /* Resources */ = {
isa = PBXResourcesBuildPhase;
buildActionMask = 2147483647;
files = (
81AB9BB82411601600AC10FF /* LaunchScreen.storyboard in Resources */,
13B07FBF1A68108700A75B9A /* Images.xcassets in Resources */,
028DCFBD5C2D4A61AC400444 /* Gilroy-Bold.ttf in Resources */,
F2C66068A3EB498DB8DE519F /* Gilroy-Regular.ttf in Resources */,
7E0F78E2BBB347D184948BFA /* Gilroy-SemiBold.ttf in Resources */,
);
runOnlyForDeploymentPostprocessing = 0;
};
/* End PBXResourcesBuildPhase section */
/* Begin PBXShellScriptBuildPhase section */
00DD1BFF1BD5951E006B06BC /* Bundle React Native code and images */ = {
isa = PBXShellScriptBuildPhase;
buildActionMask = 2147483647;
files = (
);
inputPaths = (
"$(SRCROOT)/.xcode.env.local",
"$(SRCROOT)/.xcode.env",
);
name = "Bundle React Native code and images";
outputPaths = (
);
runOnlyForDeploymentPostprocessing = 0;
shellPath = /bin/sh;
shellScript = "set -e\n\nWITH_ENVIRONMENT=\"$REACT_NATIVE_PATH/scripts/xcode/with-environment.sh\"\nREACT_NATIVE_XCODE=\"$REACT_NATIVE_PATH/scripts/react-native-xcode.sh\"\n\n/bin/sh -c \"$WITH_ENVIRONMENT $REACT_NATIVE_XCODE\"\n";
};
00EEFC60759A1932668264C0 /* [CP] Embed Pods Frameworks */ = {
isa = PBXShellScriptBuildPhase;
buildActionMask = 2147483647;
files = (
);
inputFileListPaths = (
"${PODS_ROOT}/Target Support Files/Pods-LynkedUpPro/Pods-LynkedUpPro-frameworks-${CONFIGURATION}-input-files.xcfilelist",
);
name = "[CP] Embed Pods Frameworks";
outputFileListPaths = (
"${PODS_ROOT}/Target Support Files/Pods-LynkedUpPro/Pods-LynkedUpPro-frameworks-${CONFIGURATION}-output-files.xcfilelist",
);
runOnlyForDeploymentPostprocessing = 0;
shellPath = /bin/sh;
shellScript = "\"${PODS_ROOT}/Target Support Files/Pods-LynkedUpPro/Pods-LynkedUpPro-frameworks.sh\"\n";
showEnvVarsInLog = 0;
};
C38B50BA6285516D6DCD4F65 /* [CP] Check Pods Manifest.lock */ = {
isa = PBXShellScriptBuildPhase;
buildActionMask = 2147483647;
files = (
);
inputFileListPaths = (
);
inputPaths = (
"${PODS_PODFILE_DIR_PATH}/Podfile.lock",
"${PODS_ROOT}/Manifest.lock",
);
name = "[CP] Check Pods Manifest.lock";
outputFileListPaths = (
);
outputPaths = (
"$(DERIVED_FILE_DIR)/Pods-LynkedUpPro-checkManifestLockResult.txt",
);
runOnlyForDeploymentPostprocessing = 0;
shellPath = /bin/sh;
shellScript = "diff \"${PODS_PODFILE_DIR_PATH}/Podfile.lock\" \"${PODS_ROOT}/Manifest.lock\" > /dev/null\nif [ $? != 0 ] ; then\n # print error to STDERR\n echo \"error: The sandbox is not in sync with the Podfile.lock. Run 'pod install' or update your CocoaPods installation.\" >&2\n exit 1\nfi\n# This output is used by Xcode 'outputs' to avoid re-running this script phase.\necho \"SUCCESS\" > \"${SCRIPT_OUTPUT_FILE_0}\"\n";
showEnvVarsInLog = 0;
};
E235C05ADACE081382539298 /* [CP] Copy Pods Resources */ = {
isa = PBXShellScriptBuildPhase;
buildActionMask = 2147483647;
files = (
);
inputFileListPaths = (
"${PODS_ROOT}/Target Support Files/Pods-LynkedUpPro/Pods-LynkedUpPro-resources-${CONFIGURATION}-input-files.xcfilelist",
);
name = "[CP] Copy Pods Resources";
outputFileListPaths = (
"${PODS_ROOT}/Target Support Files/Pods-LynkedUpPro/Pods-LynkedUpPro-resources-${CONFIGURATION}-output-files.xcfilelist",
);
runOnlyForDeploymentPostprocessing = 0;
shellPath = /bin/sh;
shellScript = "\"${PODS_ROOT}/Target Support Files/Pods-LynkedUpPro/Pods-LynkedUpPro-resources.sh\"\n";
showEnvVarsInLog = 0;
};
/* End PBXShellScriptBuildPhase section */
/* Begin PBXSourcesBuildPhase section */
13B07F871A680F5B00A75B9A /* Sources */ = {
isa = PBXSourcesBuildPhase;
buildActionMask = 2147483647;
files = (
761780ED2CA45674006654EE /* AppDelegate.swift in Sources */,
);
runOnlyForDeploymentPostprocessing = 0;
};
/* End PBXSourcesBuildPhase section */
/* Begin XCBuildConfiguration section */
13B07F941A680F5B00A75B9A /* Debug */ = {
isa = XCBuildConfiguration;
baseConfigurationReference = 3B4392A12AC88292D35C810B /* Pods-LynkedUpPro.debug.xcconfig */;
buildSettings = {
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
CLANG_ENABLE_MODULES = YES;
CURRENT_PROJECT_VERSION = 1;
ENABLE_BITCODE = NO;
INFOPLIST_FILE = LynkedUpPro/Info.plist;
IPHONEOS_DEPLOYMENT_TARGET = 15.1;
LD_RUNPATH_SEARCH_PATHS = (
"$(inherited)",
"@executable_path/Frameworks",
);
MARKETING_VERSION = 1.0;
OTHER_LDFLAGS = (
"$(inherited)",
"-ObjC",
"-lc++",
);
PRODUCT_BUNDLE_IDENTIFIER = "org.reactjs.native.example.$(PRODUCT_NAME:rfc1034identifier)";
PRODUCT_NAME = LynkedUpPro;
SWIFT_OPTIMIZATION_LEVEL = "-Onone";
SWIFT_VERSION = 5.0;
VERSIONING_SYSTEM = "apple-generic";
};
name = Debug;
};
13B07F951A680F5B00A75B9A /* Release */ = {
isa = XCBuildConfiguration;
baseConfigurationReference = 5709B34CF0A7D63546082F79 /* Pods-LynkedUpPro.release.xcconfig */;
buildSettings = {
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
CLANG_ENABLE_MODULES = YES;
CURRENT_PROJECT_VERSION = 1;
INFOPLIST_FILE = LynkedUpPro/Info.plist;
IPHONEOS_DEPLOYMENT_TARGET = 15.1;
LD_RUNPATH_SEARCH_PATHS = (
"$(inherited)",
"@executable_path/Frameworks",
);
MARKETING_VERSION = 1.0;
OTHER_LDFLAGS = (
"$(inherited)",
"-ObjC",
"-lc++",
);
PRODUCT_BUNDLE_IDENTIFIER = "org.reactjs.native.example.$(PRODUCT_NAME:rfc1034identifier)";
PRODUCT_NAME = LynkedUpPro;
SWIFT_VERSION = 5.0;
VERSIONING_SYSTEM = "apple-generic";
};
name = Release;
};
83CBBA201A601CBA00E9B192 /* Debug */ = {
isa = XCBuildConfiguration;
buildSettings = {
ALWAYS_SEARCH_USER_PATHS = NO;
CLANG_ANALYZER_LOCALIZABILITY_NONLOCALIZED = YES;
CLANG_CXX_LANGUAGE_STANDARD = "c++20";
CLANG_CXX_LIBRARY = "libc++";
CLANG_ENABLE_MODULES = YES;
CLANG_ENABLE_OBJC_ARC = YES;
CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES;
CLANG_WARN_BOOL_CONVERSION = YES;
CLANG_WARN_COMMA = YES;
CLANG_WARN_CONSTANT_CONVERSION = YES;
CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES;
CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;
CLANG_WARN_EMPTY_BODY = YES;
CLANG_WARN_ENUM_CONVERSION = YES;
CLANG_WARN_INFINITE_RECURSION = YES;
CLANG_WARN_INT_CONVERSION = YES;
CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES;
CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES;
CLANG_WARN_OBJC_LITERAL_CONVERSION = YES;
CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;
CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES;
CLANG_WARN_RANGE_LOOP_ANALYSIS = YES;
CLANG_WARN_STRICT_PROTOTYPES = YES;
CLANG_WARN_SUSPICIOUS_MOVE = YES;
CLANG_WARN_UNREACHABLE_CODE = YES;
CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
"CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer";
COPY_PHASE_STRIP = NO;
ENABLE_STRICT_OBJC_MSGSEND = YES;
ENABLE_TESTABILITY = YES;
"EXCLUDED_ARCHS[sdk=iphonesimulator*]" = "";
GCC_C_LANGUAGE_STANDARD = gnu99;
GCC_DYNAMIC_NO_PIC = NO;
GCC_NO_COMMON_BLOCKS = YES;
GCC_OPTIMIZATION_LEVEL = 0;
GCC_PREPROCESSOR_DEFINITIONS = (
"DEBUG=1",
"$(inherited)",
);
GCC_SYMBOLS_PRIVATE_EXTERN = NO;
GCC_WARN_64_TO_32_BIT_CONVERSION = YES;
GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;
GCC_WARN_UNDECLARED_SELECTOR = YES;
GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;
GCC_WARN_UNUSED_FUNCTION = YES;
GCC_WARN_UNUSED_VARIABLE = YES;
IPHONEOS_DEPLOYMENT_TARGET = 15.1;
LD_RUNPATH_SEARCH_PATHS = (
/usr/lib/swift,
"$(inherited)",
);
LIBRARY_SEARCH_PATHS = (
"\"$(SDKROOT)/usr/lib/swift\"",
"\"$(TOOLCHAIN_DIR)/usr/lib/swift/$(PLATFORM_NAME)\"",
"\"$(inherited)\"",
);
MTL_ENABLE_DEBUG_INFO = YES;
ONLY_ACTIVE_ARCH = YES;
OTHER_CPLUSPLUSFLAGS = (
"$(OTHER_CFLAGS)",
"-DFOLLY_NO_CONFIG",
"-DFOLLY_MOBILE=1",
"-DFOLLY_USE_LIBCPP=1",
"-DFOLLY_CFG_NO_COROUTINES=1",
"-DFOLLY_HAVE_CLOCK_GETTIME=1",
);
SDKROOT = iphoneos;
};
name = Debug;
};
83CBBA211A601CBA00E9B192 /* Release */ = {
isa = XCBuildConfiguration;
buildSettings = {
ALWAYS_SEARCH_USER_PATHS = NO;
CLANG_ANALYZER_LOCALIZABILITY_NONLOCALIZED = YES;
CLANG_CXX_LANGUAGE_STANDARD = "c++20";
CLANG_CXX_LIBRARY = "libc++";
CLANG_ENABLE_MODULES = YES;
CLANG_ENABLE_OBJC_ARC = YES;
CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES;
CLANG_WARN_BOOL_CONVERSION = YES;
CLANG_WARN_COMMA = YES;
CLANG_WARN_CONSTANT_CONVERSION = YES;
CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES;
CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;
CLANG_WARN_EMPTY_BODY = YES;
CLANG_WARN_ENUM_CONVERSION = YES;
CLANG_WARN_INFINITE_RECURSION = YES;
CLANG_WARN_INT_CONVERSION = YES;
CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES;
CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES;
CLANG_WARN_OBJC_LITERAL_CONVERSION = YES;
CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;
CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES;
CLANG_WARN_RANGE_LOOP_ANALYSIS = YES;
CLANG_WARN_STRICT_PROTOTYPES = YES;
CLANG_WARN_SUSPICIOUS_MOVE = YES;
CLANG_WARN_UNREACHABLE_CODE = YES;
CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
"CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer";
COPY_PHASE_STRIP = YES;
ENABLE_NS_ASSERTIONS = NO;
ENABLE_STRICT_OBJC_MSGSEND = YES;
"EXCLUDED_ARCHS[sdk=iphonesimulator*]" = "";
GCC_C_LANGUAGE_STANDARD = gnu99;
GCC_NO_COMMON_BLOCKS = YES;
GCC_WARN_64_TO_32_BIT_CONVERSION = YES;
GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;
GCC_WARN_UNDECLARED_SELECTOR = YES;
GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;
GCC_WARN_UNUSED_FUNCTION = YES;
GCC_WARN_UNUSED_VARIABLE = YES;
IPHONEOS_DEPLOYMENT_TARGET = 15.1;
LD_RUNPATH_SEARCH_PATHS = (
/usr/lib/swift,
"$(inherited)",
);
LIBRARY_SEARCH_PATHS = (
"\"$(SDKROOT)/usr/lib/swift\"",
"\"$(TOOLCHAIN_DIR)/usr/lib/swift/$(PLATFORM_NAME)\"",
"\"$(inherited)\"",
);
MTL_ENABLE_DEBUG_INFO = NO;
OTHER_CPLUSPLUSFLAGS = (
"$(OTHER_CFLAGS)",
"-DFOLLY_NO_CONFIG",
"-DFOLLY_MOBILE=1",
"-DFOLLY_USE_LIBCPP=1",
"-DFOLLY_CFG_NO_COROUTINES=1",
"-DFOLLY_HAVE_CLOCK_GETTIME=1",
);
SDKROOT = iphoneos;
VALIDATE_PRODUCT = YES;
};
name = Release;
};
/* End XCBuildConfiguration section */
/* Begin XCConfigurationList section */
13B07F931A680F5B00A75B9A /* Build configuration list for PBXNativeTarget "LynkedUpPro" */ = {
isa = XCConfigurationList;
buildConfigurations = (
13B07F941A680F5B00A75B9A /* Debug */,
13B07F951A680F5B00A75B9A /* Release */,
);
defaultConfigurationIsVisible = 0;
defaultConfigurationName = Release;
};
83CBB9FA1A601CBA00E9B192 /* Build configuration list for PBXProject "LynkedUpPro" */ = {
isa = XCConfigurationList;
buildConfigurations = (
83CBBA201A601CBA00E9B192 /* Debug */,
83CBBA211A601CBA00E9B192 /* Release */,
);
defaultConfigurationIsVisible = 0;
defaultConfigurationName = Release;
};
/* End XCConfigurationList section */
};
rootObject = 83CBB9F71A601CBA00E9B192 /* Project object */;
}

View File

@@ -0,0 +1,88 @@
<?xml version="1.0" encoding="UTF-8"?>
<Scheme
LastUpgradeVersion = "1210"
version = "1.3">
<BuildAction
parallelizeBuildables = "YES"
buildImplicitDependencies = "YES">
<BuildActionEntries>
<BuildActionEntry
buildForTesting = "YES"
buildForRunning = "YES"
buildForProfiling = "YES"
buildForArchiving = "YES"
buildForAnalyzing = "YES">
<BuildableReference
BuildableIdentifier = "primary"
BlueprintIdentifier = "13B07F861A680F5B00A75B9A"
BuildableName = "LynkedUpPro.app"
BlueprintName = "LynkedUpPro"
ReferencedContainer = "container:LynkedUpPro.xcodeproj">
</BuildableReference>
</BuildActionEntry>
</BuildActionEntries>
</BuildAction>
<TestAction
buildConfiguration = "Debug"
selectedDebuggerIdentifier = "Xcode.DebuggerFoundation.Debugger.LLDB"
selectedLauncherIdentifier = "Xcode.DebuggerFoundation.Launcher.LLDB"
shouldUseLaunchSchemeArgsEnv = "YES">
<Testables>
<TestableReference
skipped = "NO">
<BuildableReference
BuildableIdentifier = "primary"
BlueprintIdentifier = "00E356ED1AD99517003FC87E"
BuildableName = "LynkedUpProTests.xctest"
BlueprintName = "LynkedUpProTests"
ReferencedContainer = "container:LynkedUpPro.xcodeproj">
</BuildableReference>
</TestableReference>
</Testables>
</TestAction>
<LaunchAction
buildConfiguration = "Debug"
selectedDebuggerIdentifier = "Xcode.DebuggerFoundation.Debugger.LLDB"
selectedLauncherIdentifier = "Xcode.DebuggerFoundation.Launcher.LLDB"
launchStyle = "0"
useCustomWorkingDirectory = "NO"
ignoresPersistentStateOnLaunch = "NO"
debugDocumentVersioning = "YES"
debugServiceExtension = "internal"
allowLocationSimulation = "YES">
<BuildableProductRunnable
runnableDebuggingMode = "0">
<BuildableReference
BuildableIdentifier = "primary"
BlueprintIdentifier = "13B07F861A680F5B00A75B9A"
BuildableName = "LynkedUpPro.app"
BlueprintName = "LynkedUpPro"
ReferencedContainer = "container:LynkedUpPro.xcodeproj">
</BuildableReference>
</BuildableProductRunnable>
</LaunchAction>
<ProfileAction
buildConfiguration = "Release"
shouldUseLaunchSchemeArgsEnv = "YES"
savedToolIdentifier = ""
useCustomWorkingDirectory = "NO"
debugDocumentVersioning = "YES">
<BuildableProductRunnable
runnableDebuggingMode = "0">
<BuildableReference
BuildableIdentifier = "primary"
BlueprintIdentifier = "13B07F861A680F5B00A75B9A"
BuildableName = "LynkedUpPro.app"
BlueprintName = "LynkedUpPro"
ReferencedContainer = "container:LynkedUpPro.xcodeproj">
</BuildableReference>
</BuildableProductRunnable>
</ProfileAction>
<AnalyzeAction
buildConfiguration = "Debug">
</AnalyzeAction>
<ArchiveAction
buildConfiguration = "Release"
revealArchiveInOrganizer = "YES">
</ArchiveAction>
</Scheme>

View File

@@ -0,0 +1,48 @@
import UIKit
import React
import React_RCTAppDelegate
import ReactAppDependencyProvider
@main
class AppDelegate: UIResponder, UIApplicationDelegate {
var window: UIWindow?
var reactNativeDelegate: ReactNativeDelegate?
var reactNativeFactory: RCTReactNativeFactory?
func application(
_ application: UIApplication,
didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]? = nil
) -> Bool {
let delegate = ReactNativeDelegate()
let factory = RCTReactNativeFactory(delegate: delegate)
delegate.dependencyProvider = RCTAppDependencyProvider()
reactNativeDelegate = delegate
reactNativeFactory = factory
window = UIWindow(frame: UIScreen.main.bounds)
factory.startReactNative(
withModuleName: "LynkedUpPro",
in: window,
launchOptions: launchOptions
)
return true
}
}
class ReactNativeDelegate: RCTDefaultReactNativeFactoryDelegate {
override func sourceURL(for bridge: RCTBridge) -> URL? {
self.bundleURL()
}
override func bundleURL() -> URL? {
#if DEBUG
RCTBundleURLProvider.sharedSettings().jsBundleURL(forBundleRoot: "index")
#else
Bundle.main.url(forResource: "main", withExtension: "jsbundle")
#endif
}
}

View File

@@ -0,0 +1,53 @@
{
"images" : [
{
"idiom" : "iphone",
"scale" : "2x",
"size" : "20x20"
},
{
"idiom" : "iphone",
"scale" : "3x",
"size" : "20x20"
},
{
"idiom" : "iphone",
"scale" : "2x",
"size" : "29x29"
},
{
"idiom" : "iphone",
"scale" : "3x",
"size" : "29x29"
},
{
"idiom" : "iphone",
"scale" : "2x",
"size" : "40x40"
},
{
"idiom" : "iphone",
"scale" : "3x",
"size" : "40x40"
},
{
"idiom" : "iphone",
"scale" : "2x",
"size" : "60x60"
},
{
"idiom" : "iphone",
"scale" : "3x",
"size" : "60x60"
},
{
"idiom" : "ios-marketing",
"scale" : "1x",
"size" : "1024x1024"
}
],
"info" : {
"author" : "xcode",
"version" : 1
}
}

View File

@@ -0,0 +1,6 @@
{
"info" : {
"version" : 1,
"author" : "xcode"
}
}

View File

@@ -0,0 +1,59 @@
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<key>CADisableMinimumFrameDurationOnPhone</key>
<true/>
<key>CFBundleDevelopmentRegion</key>
<string>en</string>
<key>CFBundleDisplayName</key>
<string>LynkedUpPro</string>
<key>CFBundleExecutable</key>
<string>$(EXECUTABLE_NAME)</string>
<key>CFBundleIdentifier</key>
<string>$(PRODUCT_BUNDLE_IDENTIFIER)</string>
<key>CFBundleInfoDictionaryVersion</key>
<string>6.0</string>
<key>CFBundleName</key>
<string>$(PRODUCT_NAME)</string>
<key>CFBundlePackageType</key>
<string>APPL</string>
<key>CFBundleShortVersionString</key>
<string>$(MARKETING_VERSION)</string>
<key>CFBundleSignature</key>
<string>????</string>
<key>CFBundleVersion</key>
<string>$(CURRENT_PROJECT_VERSION)</string>
<key>LSRequiresIPhoneOS</key>
<true/>
<key>NSAppTransportSecurity</key>
<dict>
<key>NSAllowsArbitraryLoads</key>
<false/>
<key>NSAllowsLocalNetworking</key>
<true/>
</dict>
<key>NSLocationWhenInUseUsageDescription</key>
<string/>
<key>UILaunchStoryboardName</key>
<string>LaunchScreen</string>
<key>UIRequiredDeviceCapabilities</key>
<array>
<string>arm64</string>
</array>
<key>UISupportedInterfaceOrientations</key>
<array>
<string>UIInterfaceOrientationPortrait</string>
<string>UIInterfaceOrientationLandscapeLeft</string>
<string>UIInterfaceOrientationLandscapeRight</string>
</array>
<key>UIViewControllerBasedStatusBarAppearance</key>
<false/>
<key>UIAppFonts</key>
<array>
<string>Gilroy-Bold.ttf</string>
<string>Gilroy-Regular.ttf</string>
<string>Gilroy-SemiBold.ttf</string>
</array>
</dict>
</plist>

View File

@@ -0,0 +1,47 @@
<?xml version="1.0" encoding="UTF-8"?>
<document type="com.apple.InterfaceBuilder3.CocoaTouch.Storyboard.XIB" version="3.0" toolsVersion="15702" targetRuntime="iOS.CocoaTouch" propertyAccessControl="none" useAutolayout="YES" launchScreen="YES" useTraitCollections="YES" useSafeAreas="YES" colorMatched="YES" initialViewController="01J-lp-oVM">
<device id="retina4_7" orientation="portrait" appearance="light"/>
<dependencies>
<deployment identifier="iOS"/>
<plugIn identifier="com.apple.InterfaceBuilder.IBCocoaTouchPlugin" version="15704"/>
<capability name="Safe area layout guides" minToolsVersion="9.0"/>
<capability name="documents saved in the Xcode 8 format" minToolsVersion="8.0"/>
</dependencies>
<scenes>
<!--View Controller-->
<scene sceneID="EHf-IW-A2E">
<objects>
<viewController id="01J-lp-oVM" sceneMemberID="viewController">
<view key="view" contentMode="scaleToFill" id="Ze5-6b-2t3">
<rect key="frame" x="0.0" y="0.0" width="375" height="667"/>
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
<subviews>
<label opaque="NO" clipsSubviews="YES" userInteractionEnabled="NO" contentMode="left" horizontalHuggingPriority="251" verticalHuggingPriority="251" text="LynkedUpPro" textAlignment="center" lineBreakMode="middleTruncation" baselineAdjustment="alignBaselines" minimumFontSize="18" translatesAutoresizingMaskIntoConstraints="NO" id="GJd-Yh-RWb">
<rect key="frame" x="0.0" y="202" width="375" height="43"/>
<fontDescription key="fontDescription" type="boldSystem" pointSize="36"/>
<nil key="highlightedColor"/>
</label>
<label opaque="NO" clipsSubviews="YES" userInteractionEnabled="NO" contentMode="left" horizontalHuggingPriority="251" verticalHuggingPriority="251" text="Powered by React Native" textAlignment="center" lineBreakMode="tailTruncation" baselineAdjustment="alignBaselines" minimumFontSize="9" translatesAutoresizingMaskIntoConstraints="NO" id="MN2-I3-ftu">
<rect key="frame" x="0.0" y="626" width="375" height="21"/>
<fontDescription key="fontDescription" type="system" pointSize="17"/>
<nil key="highlightedColor"/>
</label>
</subviews>
<color key="backgroundColor" systemColor="systemBackgroundColor" cocoaTouchSystemColor="whiteColor"/>
<constraints>
<constraint firstItem="Bcu-3y-fUS" firstAttribute="bottom" secondItem="MN2-I3-ftu" secondAttribute="bottom" constant="20" id="OZV-Vh-mqD"/>
<constraint firstItem="Bcu-3y-fUS" firstAttribute="centerX" secondItem="GJd-Yh-RWb" secondAttribute="centerX" id="Q3B-4B-g5h"/>
<constraint firstItem="MN2-I3-ftu" firstAttribute="centerX" secondItem="Bcu-3y-fUS" secondAttribute="centerX" id="akx-eg-2ui"/>
<constraint firstItem="MN2-I3-ftu" firstAttribute="leading" secondItem="Bcu-3y-fUS" secondAttribute="leading" id="i1E-0Y-4RG"/>
<constraint firstItem="GJd-Yh-RWb" firstAttribute="centerY" secondItem="Ze5-6b-2t3" secondAttribute="bottom" multiplier="1/3" constant="1" id="moa-c2-u7t"/>
<constraint firstItem="GJd-Yh-RWb" firstAttribute="leading" secondItem="Bcu-3y-fUS" secondAttribute="leading" symbolic="YES" id="x7j-FC-K8j"/>
</constraints>
<viewLayoutGuide key="safeArea" id="Bcu-3y-fUS"/>
</view>
</viewController>
<placeholder placeholderIdentifier="IBFirstResponder" id="iYj-Kq-Ea1" userLabel="First Responder" sceneMemberID="firstResponder"/>
</objects>
<point key="canvasLocation" x="52.173913043478265" y="375"/>
</scene>
</scenes>
</document>

View File

@@ -0,0 +1,37 @@
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<key>NSPrivacyAccessedAPITypes</key>
<array>
<dict>
<key>NSPrivacyAccessedAPIType</key>
<string>NSPrivacyAccessedAPICategoryFileTimestamp</string>
<key>NSPrivacyAccessedAPITypeReasons</key>
<array>
<string>C617.1</string>
</array>
</dict>
<dict>
<key>NSPrivacyAccessedAPIType</key>
<string>NSPrivacyAccessedAPICategoryUserDefaults</string>
<key>NSPrivacyAccessedAPITypeReasons</key>
<array>
<string>CA92.1</string>
</array>
</dict>
<dict>
<key>NSPrivacyAccessedAPIType</key>
<string>NSPrivacyAccessedAPICategorySystemBootTime</string>
<key>NSPrivacyAccessedAPITypeReasons</key>
<array>
<string>35F9.1</string>
</array>
</dict>
</array>
<key>NSPrivacyCollectedDataTypes</key>
<array/>
<key>NSPrivacyTracking</key>
<false/>
</dict>
</plist>

34
ios/Podfile Normal file
View File

@@ -0,0 +1,34 @@
# Resolve react_native_pods.rb with node to allow for hoisting
require Pod::Executable.execute_command('node', ['-p',
'require.resolve(
"react-native/scripts/react_native_pods.rb",
{paths: [process.argv[1]]},
)', __dir__]).strip
platform :ios, min_ios_version_supported
prepare_react_native_project!
linkage = ENV['USE_FRAMEWORKS']
if linkage != nil
Pod::UI.puts "Configuring Pod with #{linkage}ally linked Frameworks".green
use_frameworks! :linkage => linkage.to_sym
end
target 'LynkedUpPro' do
config = use_native_modules!
use_react_native!(
:path => config[:reactNativePath],
# An absolute path to your application root.
:app_path => "#{Pod::Config.instance.installation_root}/.."
)
post_install do |installer|
react_native_post_install(
installer,
config[:reactNativePath],
:mac_catalyst_enabled => false,
# :ccache_enabled => true
)
end
end

View File

@@ -0,0 +1,17 @@
{
"migIndex": 1,
"data": [
{
"path": "assets/fonts/Gilroy-Bold.ttf",
"sha1": "7fe4d8cf5325dd47363d0f80fefb40ac7df5d5a6"
},
{
"path": "assets/fonts/Gilroy-Regular.ttf",
"sha1": "a737c19356e50cb9a50eb64ca7446c80dd3aa3e3"
},
{
"path": "assets/fonts/Gilroy-SemiBold.ttf",
"sha1": "8663a356eb9bc23b77696208f74376daf59687ea"
}
]
}

3
jest.config.js Normal file
View File

@@ -0,0 +1,3 @@
module.exports = {
preset: 'react-native',
};

11
metro.config.js Normal file
View File

@@ -0,0 +1,11 @@
const { getDefaultConfig, mergeConfig } = require('@react-native/metro-config');
/**
* Metro configuration
* https://reactnative.dev/docs/metro
*
* @type {import('@react-native/metro-config').MetroConfig}
*/
const config = {};
module.exports = mergeConfig(getDefaultConfig(__dirname), config);

11711
package-lock.json generated Normal file

File diff suppressed because it is too large Load Diff

44
package.json Normal file
View File

@@ -0,0 +1,44 @@
{
"name": "LynkedUpPro",
"version": "0.0.1",
"private": true,
"scripts": {
"android": "react-native run-android",
"ios": "react-native run-ios",
"lint": "eslint .",
"start": "react-native start",
"test": "jest"
},
"dependencies": {
"@react-native-async-storage/async-storage": "^1.24.0",
"@react-native-community/netinfo": "^11.3.1",
"@react-native/new-app-screen": "0.82.1",
"lynkeduppro-login-sdk": "^0.1.9",
"react": "19.1.1",
"react-native": "0.82.1",
"react-native-safe-area-context": "^5.5.2"
},
"devDependencies": {
"@babel/core": "^7.25.2",
"@babel/preset-env": "^7.25.3",
"@babel/runtime": "^7.25.0",
"@react-native-community/cli": "20.0.0",
"@react-native-community/cli-platform-android": "20.0.0",
"@react-native-community/cli-platform-ios": "20.0.0",
"@react-native/babel-preset": "0.82.1",
"@react-native/eslint-config": "0.82.1",
"@react-native/metro-config": "0.82.1",
"@react-native/typescript-config": "0.82.1",
"@types/jest": "^29.5.13",
"@types/react": "^19.1.1",
"@types/react-test-renderer": "^19.1.0",
"eslint": "^8.19.0",
"jest": "^29.6.3",
"prettier": "2.8.8",
"react-test-renderer": "19.1.1",
"typescript": "^5.8.3"
},
"engines": {
"node": ">=20"
}
}

7
react-native.config.js Normal file
View File

@@ -0,0 +1,7 @@
module.exports = {
project: {
ios: {},
android: {},
},
assets: ['./assets/fonts'], // path to your font folder
};

BIN
src/assets/img/eye.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 6.6 KiB

BIN
src/assets/img/images.jpeg Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.9 KiB

BIN
src/assets/img/images.jpg Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 13 KiB

305
src/components/Login.tsx Normal file
View File

@@ -0,0 +1,305 @@
import React, { useState } from 'react';
import {
Image,
ImageBackground,
Keyboard,
KeyboardAvoidingView,
Platform,
StyleSheet,
Text,
TextInput,
TouchableOpacity,
View,
Alert,
ActivityIndicator,
} from 'react-native';
import { SafeAreaView } from 'react-native-safe-area-context';
import { authAPI } from '../services/authAPI';
import { ApiError } from '../types/auth';
import { networkService } from '../services/networkService';
interface LoginProps {
onNavigateToRegister: () => void;
onLoginSuccess: () => void;
}
const Login: React.FC<LoginProps> = ({ onNavigateToRegister, onLoginSuccess }) => {
const [secureTextEntry, setSecureTextEntry] = useState(true);
const [email, setEmail] = useState('');
const [password, setPassword] = useState('');
const [isLoading, setIsLoading] = useState(false);
const [isOnline, setIsOnline] = useState(true);
// Listen for network changes
React.useEffect(() => {
setIsOnline(networkService.isOnline());
const unsubscribe = networkService.addListener((networkState) => {
setIsOnline(networkState.isConnected);
});
return unsubscribe;
}, []);
const handleLogin = async () => {
if (!email || !password) {
Alert.alert('Error', 'Please fill in all fields');
return;
}
// Basic email validation
const emailRegex = /^[^\s@]+@[^\s@]+\.[^\s@]+$/;
if (!emailRegex.test(email)) {
Alert.alert('Error', 'Please enter a valid email address');
return;
}
setIsLoading(true);
try {
const response = await authAPI.login({ email, password });
Alert.alert('Success', response.message, [
{
text: 'OK',
onPress: () => {
console.log('User logged in:', response.user);
console.log('Token:', response.token);
onLoginSuccess();
}
}
]);
} catch (error: any) {
const apiError = error as ApiError;
Alert.alert(
'Login Failed',
apiError.error || apiError.message || 'An unexpected error occurred'
);
console.error('Login error:', error);
} finally {
setIsLoading(false);
}
};
const handleForgotPassword = () => {
Alert.alert('Forgot Password', 'Password reset functionality would be implemented here');
};
const handleUseTestCredentials = () => {
const testCreds = authAPI.getTestCredentials();
setEmail(testCreds.email);
setPassword(testCreds.password);
Alert.alert('Test Credentials', `Using ReqRes API test credentials:\n${testCreds.email} / ${testCreds.password}`);
};
return (
<ImageBackground
source={require('../assets/img/images.jpg')}
style={styles.container}
resizeMode="cover"
>
<KeyboardAvoidingView
behavior={Platform.OS === 'ios' ? 'padding' : 'height'}
style={styles.containerMain}
>
<SafeAreaView style={styles.safeArea}>
<View style={styles.containerMain}>
<Text style={styles.text}>Login</Text>
{/* Status Info */}
<View style={styles.testInfo}>
<Text style={styles.testInfoText}>
{isOnline ? '🟢 Online Mode' : '🔴 Offline Mode'}
</Text>
<TouchableOpacity onPress={handleUseTestCredentials} style={styles.testCredentialsButton}>
<Text style={styles.testCredentialsText}>Use Test Credentials</Text>
</TouchableOpacity>
</View>
<View style={styles.inputContainer}>
<TextInput
style={styles.input}
placeholder="Email"
value={email}
onChangeText={setEmail}
keyboardType="email-address"
autoCapitalize="none"
autoCorrect={false}
autoComplete="email"
returnKeyType="next"
onSubmitEditing={() => Keyboard.dismiss()}
placeholderTextColor="black"
/>
</View>
<View style={styles.inputContainer}>
<TextInput
style={styles.input}
placeholder="Password"
value={password}
onChangeText={setPassword}
secureTextEntry={secureTextEntry}
returnKeyType="done"
onSubmitEditing={handleLogin}
placeholderTextColor="black"
/>
<TouchableOpacity
style={styles.eyeIcon}
onPress={() => setSecureTextEntry(!secureTextEntry)}
>
<Image
source={require('../assets/img/eye.png')}
style={[
styles.Icon,
!secureTextEntry ? styles.eyeIconActive : styles.eyeIconInactive
]}
/>
</TouchableOpacity>
</View>
{/* <TouchableOpacity style={styles.forgotPassword} onPress={handleForgotPassword}>
<Text style={styles.forgotPasswordText}>Forgot Password?</Text>
</TouchableOpacity> */}
<TouchableOpacity
style={[styles.button, isLoading && styles.buttonDisabled]}
onPress={handleLogin}
disabled={isLoading}
>
{isLoading ? (
<ActivityIndicator color="white" size="small" />
) : (
<Text style={styles.buttonText}>Login</Text>
)}
</TouchableOpacity>
<TouchableOpacity style={styles.buttonSignup} onPress={onNavigateToRegister}>
<Text style={styles.buttonSignupText}>Not a member? Sign Up</Text>
</TouchableOpacity>
</View>
</SafeAreaView>
</KeyboardAvoidingView>
</ImageBackground>
);
};
const styles = StyleSheet.create({
container: {
flex: 1,
},
safeArea: {
flex: 1,
},
containerMain: {
flex: 1,
justifyContent: 'center',
marginHorizontal: 10,
},
text: {
textAlign: 'center',
marginBottom: 20,
color: 'black',
fontSize: 30,
fontWeight: 'bold',
},
testInfo: {
alignItems: 'center',
marginBottom: 15,
padding: 10,
backgroundColor: '#ffffff10',
borderRadius: 10,
},
testInfoText: {
color: '#3bb6d8',
fontSize: 12,
fontWeight: '600',
marginBottom: 5,
},
testCredentialsButton: {
paddingHorizontal: 12,
paddingVertical: 4,
backgroundColor: '#3bb6d8',
borderRadius: 15,
},
testCredentialsText: {
color: 'white',
fontSize: 10,
fontWeight: '500',
},
inputContainer: {
flexDirection: 'row',
marginBottom: 10,
height: 40,
borderWidth: 0.5,
borderColor: 'white',
borderRadius: 100,
backgroundColor: '#ffffff20',
paddingHorizontal: 10,
alignItems: 'center',
},
input: {
flex: 1,
width: '100%',
color: 'black',
},
eyeIcon: {
width: 30,
height: 30,
justifyContent: 'center',
alignItems: 'center',
},
Icon: {
width: '80%',
height: '80%',
},
eyeIconActive: {
tintColor: '#3bb6d8',
},
eyeIconInactive: {
tintColor: '#ffffff',
},
forgotPassword: {
alignSelf: 'flex-end',
marginBottom: 10,
},
forgotPasswordText: {
color: '#3bb6d8',
fontSize: 14,
textDecorationLine: 'underline',
},
button: {
marginTop: 20,
backgroundColor: '#3bb6d8',
padding: 8,
borderRadius: 100,
borderWidth: 0.5,
borderColor: 'white',
minHeight: 40,
justifyContent: 'center',
},
buttonDisabled: {
backgroundColor: '#a0a0a0',
borderColor: '#cccccc',
},
buttonText: {
color: 'white',
textAlign: 'center',
fontSize: 16,
fontWeight: 'bold',
},
buttonSignup: {
marginTop: 10,
backgroundColor: '#ffffff80',
padding: 8,
borderRadius: 100,
borderWidth: 0.5,
borderColor: '#3bb6d8',
},
buttonSignupText: {
color: 'black',
textAlign: 'center',
fontSize: 16,
},
});
export default Login;

386
src/components/Register.tsx Normal file
View File

@@ -0,0 +1,386 @@
import React, { useState } from 'react';
import {
Image,
ImageBackground,
KeyboardAvoidingView,
Platform,
StyleSheet,
Text,
TextInput,
TouchableOpacity,
View,
Alert,
ScrollView,
ActivityIndicator,
} from 'react-native';
import { SafeAreaView } from 'react-native-safe-area-context';
import { authAPI } from '../services/authAPI';
import { ApiError } from '../types/auth';
import { networkService } from '../services/networkService';
interface RegisterProps {
onNavigateToLogin: () => void;
onRegisterSuccess: () => void;
}
const Register: React.FC<RegisterProps> = ({ onNavigateToLogin, onRegisterSuccess }) => {
const [secureTextEntry, setSecureTextEntry] = useState(true);
const [confirmSecureTextEntry, setConfirmSecureTextEntry] = useState(true);
const [isLoading, setIsLoading] = useState(false);
const [isOnline, setIsOnline] = useState(true);
// Listen for network changes
React.useEffect(() => {
setIsOnline(networkService.isOnline());
const unsubscribe = networkService.addListener((networkState) => {
setIsOnline(networkState.isConnected);
});
return unsubscribe;
}, []);
const [formData, setFormData] = useState({
fullName: '',
email: '',
phone: '',
password: '',
confirmPassword: '',
});
const handleInputChange = (field: string, value: string) => {
setFormData(prev => ({
...prev,
[field]: value
}));
};
const validateForm = () => {
const { fullName, email, phone, password, confirmPassword } = formData;
if (!fullName || !email || !phone || !password || !confirmPassword) {
Alert.alert('Error', 'Please fill in all fields');
return false;
}
if (password !== confirmPassword) {
Alert.alert('Error', 'Passwords do not match');
return false;
}
if (password.length < 6) {
Alert.alert('Error', 'Password must be at least 6 characters long');
return false;
}
const emailRegex = /^[^\s@]+@[^\s@]+\.[^\s@]+$/;
if (!emailRegex.test(email)) {
Alert.alert('Error', 'Please enter a valid email address');
return false;
}
return true;
};
const handleRegister = async () => {
if (!validateForm()) return;
setIsLoading(true);
try {
const response = await authAPI.register({
fullName: formData.fullName,
email: formData.email,
phone: formData.phone,
password: formData.password,
});
Alert.alert('Success', response.message, [
{
text: 'OK',
onPress: () => {
console.log('User registered:', response.user);
console.log('Token:', response.token);
onRegisterSuccess();
}
}
]);
} catch (error: any) {
const apiError = error as ApiError;
Alert.alert(
'Registration Failed',
apiError.error || apiError.message || 'An unexpected error occurred'
);
console.error('Registration error:', error);
} finally {
setIsLoading(false);
}
};
return (
<ImageBackground
source={require('../assets/img/images.jpg')}
style={styles.container}
resizeMode="cover"
>
<KeyboardAvoidingView
behavior={Platform.OS === 'ios' ? 'padding' : 'height'}
style={styles.containerMain}
>
<SafeAreaView style={styles.safeArea}>
<ScrollView
contentContainerStyle={styles.scrollContainer}
showsVerticalScrollIndicator={false}
keyboardShouldPersistTaps="handled"
>
<View style={styles.containerMain}>
<Text style={styles.text}>Sign Up</Text>
{/* Status Info */}
<View style={styles.apiInfo}>
<Text style={styles.apiInfoText}>
{isOnline ? '🟢 Online Mode' : '🔴 Offline Mode'}
</Text>
</View>
<View style={styles.inputContainer}>
<TextInput
style={styles.input}
placeholder="Full Name"
value={formData.fullName}
onChangeText={(value) => handleInputChange('fullName', value)}
autoCapitalize="words"
autoCorrect={false}
returnKeyType="next"
placeholderTextColor="black"
/>
</View>
<View style={styles.inputContainer}>
<TextInput
style={styles.input}
placeholder="Email"
value={formData.email}
onChangeText={(value) => handleInputChange('email', value)}
keyboardType="email-address"
autoCapitalize="none"
autoCorrect={false}
autoComplete="email"
returnKeyType="next"
placeholderTextColor="black"
/>
</View>
<View style={styles.inputContainer}>
<TextInput
style={styles.input}
placeholder="Phone Number"
value={formData.phone}
onChangeText={(value) => handleInputChange('phone', value)}
keyboardType="phone-pad"
autoComplete="tel"
returnKeyType="next"
placeholderTextColor="black"
/>
</View>
<View style={styles.inputContainer}>
<TextInput
style={styles.input}
placeholder="Password"
value={formData.password}
onChangeText={(value) => handleInputChange('password', value)}
secureTextEntry={secureTextEntry}
returnKeyType="next"
placeholderTextColor="black"
/>
<TouchableOpacity
style={styles.eyeIcon}
onPress={() => setSecureTextEntry(!secureTextEntry)}
>
<Image
source={require('../assets/img/eye.png')}
style={[
styles.Icon,
!secureTextEntry ? styles.eyeIconActive : styles.eyeIconInactive
]}
/>
</TouchableOpacity>
</View>
<View style={styles.inputContainer}>
<TextInput
style={styles.input}
placeholder="Confirm Password"
value={formData.confirmPassword}
onChangeText={(value) => handleInputChange('confirmPassword', value)}
secureTextEntry={confirmSecureTextEntry}
returnKeyType="done"
onSubmitEditing={handleRegister}
placeholderTextColor="black"
/>
<TouchableOpacity
style={styles.eyeIcon}
onPress={() => setConfirmSecureTextEntry(!confirmSecureTextEntry)}
>
<Image
source={require('../assets/img/eye.png')}
style={[
styles.Icon,
!confirmSecureTextEntry ? styles.eyeIconActive : styles.eyeIconInactive
]}
/>
</TouchableOpacity>
</View>
<View style={styles.termsContainer}>
<Text style={styles.termsText}>
By signing up, you agree to our{' '}
<Text style={styles.linkText}>Terms of Service</Text>
{' '}and{' '}
<Text style={styles.linkText}>Privacy Policy</Text>
</Text>
</View>
<TouchableOpacity
style={[styles.button, isLoading && styles.buttonDisabled]}
onPress={handleRegister}
disabled={isLoading}
>
{isLoading ? (
<ActivityIndicator color="white" size="small" />
) : (
<Text style={styles.buttonText}>Sign Up</Text>
)}
</TouchableOpacity>
<TouchableOpacity style={styles.buttonSignup} onPress={onNavigateToLogin}>
<Text style={styles.buttonSignupText}>Already have an account? Login</Text>
</TouchableOpacity>
</View>
</ScrollView>
</SafeAreaView>
</KeyboardAvoidingView>
</ImageBackground>
);
};
const styles = StyleSheet.create({
container: {
flex: 1,
},
safeArea: {
flex: 1,
},
scrollContainer: {
flexGrow: 1,
justifyContent: 'center',
},
containerMain: {
flex: 1,
justifyContent: 'center',
marginHorizontal: 10,
paddingVertical: 20,
},
text: {
textAlign: 'center',
marginBottom: 20,
color: 'black',
fontSize: 30,
fontWeight: 'bold',
},
inputContainer: {
flexDirection: 'row',
marginBottom: 10,
height: 40,
borderWidth: 0.5,
borderColor: 'white',
borderRadius: 100,
backgroundColor: '#ffffff20',
paddingHorizontal: 10,
alignItems: 'center',
},
input: {
flex: 1,
width: '100%',
color: 'black',
},
eyeIcon: {
width: 30,
height: 30,
justifyContent: 'center',
alignItems: 'center',
},
Icon: {
width: '80%',
height: '80%',
},
eyeIconActive: {
tintColor: '#3bb6d8',
},
eyeIconInactive: {
tintColor: '#ffffff',
},
termsContainer: {
marginVertical: 15,
paddingHorizontal: 10,
},
termsText: {
color: 'black',
fontSize: 12,
textAlign: 'center',
lineHeight: 18,
},
linkText: {
color: '#3bb6d8',
textDecorationLine: 'underline',
fontWeight: '500',
},
apiInfo: {
alignItems: 'center',
marginBottom: 15,
padding: 8,
backgroundColor: '#ffffff10',
borderRadius: 10,
},
apiInfoText: {
color: '#3bb6d8',
fontSize: 12,
fontWeight: '600',
},
button: {
marginTop: 10,
backgroundColor: '#3bb6d8',
padding: 8,
borderRadius: 100,
borderWidth: 0.5,
borderColor: 'white',
minHeight: 40,
justifyContent: 'center',
},
buttonDisabled: {
backgroundColor: '#a0a0a0',
borderColor: '#cccccc',
},
buttonText: {
color: 'white',
textAlign: 'center',
fontSize: 16,
fontWeight: 'bold',
},
buttonSignup: {
marginTop: 10,
backgroundColor: '#ffffff80',
padding: 8,
borderRadius: 100,
borderWidth: 0.5,
borderColor: '#3bb6d8',
},
buttonSignupText: {
color: 'black',
textAlign: 'center',
fontSize: 16,
},
});
export default Register;

446
src/services/authAPI.ts Normal file
View File

@@ -0,0 +1,446 @@
// Simple Open Authentication API using ReqRes
// ReqRes is a free, open API for testing - no setup required!
// Website: https://reqres.in
import { LoginRequest, RegisterRequest, AuthResponse, ApiError } from '../types/auth';
import { localStorageService } from './localStorage';
import { networkService } from './networkService';
// ReqRes API Configuration (Free testing API)
const API_BASE_URL = 'https://reqres.in/api';
// API request helper
async function apiRequest(endpoint: string, options: RequestInit = {}): Promise<any> {
const url = `${API_BASE_URL}${endpoint}`;
try {
console.log('Making API request to:', url);
console.log('Request options:', options);
const response = await fetch(url, {
headers: {
'Content-Type': 'application/json',
...options.headers,
},
...options,
});
console.log('Response status:', response.status);
console.log('Response ok:', response.ok);
let data;
try {
const responseText = await response.text();
console.log('Response text:', responseText);
if (responseText && responseText.trim()) {
// Try to parse as JSON
try {
data = JSON.parse(responseText);
} catch {
console.error('JSON parsing failed, response was:', responseText);
// If JSON parsing fails, treat as a server error and use fallback
throw {
success: false,
message: 'Server returned invalid response',
error: 'API_PARSE_ERROR',
isParseError: true,
} as ApiError & { isParseError: boolean };
}
} else {
console.log('Empty response, treating as error');
throw {
success: false,
message: 'Server returned empty response',
error: 'API_EMPTY_RESPONSE',
isParseError: true,
} as ApiError & { isParseError: boolean };
}
} catch (error: any) {
if (error.isParseError) {
throw error;
}
console.error('Response reading error:', error);
throw {
success: false,
message: 'Failed to read server response',
error: 'API_READ_ERROR',
isParseError: true,
} as ApiError & { isParseError: boolean };
}
if (!response.ok) {
console.error('API error response:', data);
throw {
success: false,
message: data.error || data.message || 'Request failed',
error: `HTTP ${response.status}: ${data.error || 'Unknown error'}`,
} as ApiError;
}
console.log('API success response:', data);
return data;
} catch (error: any) {
console.error('API request error:', error);
if (error.success === false) {
throw error;
}
// Handle network errors
if (error.name === 'TypeError' && error.message.includes('fetch')) {
throw {
success: false,
message: 'Network connection failed',
error: 'Please check your internet connection',
} as ApiError;
}
throw {
success: false,
message: 'Network error',
error: error.message || 'Unable to connect to server',
} as ApiError;
}
}
// Authentication API
export const authAPI = {
// Initialize local storage with default users
async initialize(): Promise<void> {
await localStorageService.initializeDefaultUsers();
},
// Login with online/offline support
async login(credentials: LoginRequest): Promise<AuthResponse> {
console.log('Attempting login with credentials:', { email: credentials.email });
console.log('Network status:', networkService.getNetworkInfo());
// Always try local authentication first for better performance
const localResult = await this.localLogin(credentials);
if (localResult.success) {
// If we're online, also try to sync with API in background
if (networkService.isOnline()) {
this.backgroundSync(credentials).catch(error => {
console.log('Background sync failed:', error);
});
}
return localResult;
}
// If local login failed and we're online, try API
if (networkService.isOnline()) {
try {
console.log('Local login failed, trying API login');
const response = await apiRequest('/login', {
method: 'POST',
body: JSON.stringify({
email: credentials.email,
password: credentials.password,
}),
});
if (!response.token) {
throw {
success: false,
message: 'Login failed',
error: 'No authentication token received',
} as ApiError;
}
// Save user locally for future offline use
try {
await localStorageService.saveUser({
fullName: 'API User',
email: credentials.email,
password: credentials.password,
});
} catch (saveError) {
console.log('User already exists locally:', saveError);
}
// Create local session
const user = await localStorageService.findUserByEmail(credentials.email);
if (user) {
const token = await localStorageService.createSession(user);
return {
success: true,
message: 'Login successful! (Online)',
user: {
id: user.id,
fullName: user.fullName,
email: user.email,
phone: user.phone,
},
token,
};
}
// Fallback response
return {
success: true,
message: 'Login successful!',
user: {
id: '1',
fullName: 'Test User',
email: credentials.email,
},
token: response.token,
};
} catch (error: any) {
console.error('API login error:', error);
// If API fails, fall back to local authentication
return await this.localLogin(credentials, true);
}
}
// Offline and local login failed
throw {
success: false,
message: 'Login failed',
error: 'Invalid credentials. Please check your email and password.',
} as ApiError;
},
// Register with online/offline support
async register(userData: RegisterRequest): Promise<AuthResponse> {
try {
console.log('Attempting registration with data:', { email: userData.email });
console.log('Network status:', networkService.getNetworkInfo());
// Always save user locally first
const localUser = await localStorageService.saveUser({
fullName: userData.fullName,
email: userData.email,
phone: userData.phone,
password: userData.password,
});
// Create local session
const token = await localStorageService.createSession(localUser);
// If online, try to sync with API in background
if (networkService.isOnline()) {
this.backgroundRegisterSync(userData).catch(error => {
console.log('Background registration sync failed:', error);
});
}
const message = networkService.isOnline()
? 'Registration successful!'
: 'Registration successful! (Offline mode)';
return {
success: true,
message,
user: {
id: localUser.id,
fullName: localUser.fullName,
email: localUser.email,
phone: localUser.phone,
},
token,
};
} catch (error: any) {
console.error('Registration error:', error);
if (error.message && error.message.includes('User already exists')) {
throw {
success: false,
message: 'Registration failed',
error: 'An account with this email already exists',
} as ApiError;
}
throw {
success: false,
message: 'Registration failed',
error: error.message || 'An unexpected error occurred',
} as ApiError;
}
},
// Background registration sync with API
async backgroundRegisterSync(userData: RegisterRequest): Promise<void> {
try {
console.log('Background registration sync with API');
await apiRequest('/register', {
method: 'POST',
body: JSON.stringify({
email: userData.email,
password: userData.password,
}),
});
console.log('Background registration sync successful');
} catch (error) {
console.log('Background registration sync failed (non-critical):', error);
}
},
// Get test credentials for easy testing
getTestCredentials() {
return {
email: 'eve.holt@reqres.in',
password: 'cityslicka',
};
},
// Check if credentials are test credentials
isTestCredentials(email: string, password: string) {
const testCreds = this.getTestCredentials();
return email === testCreds.email && password === testCreds.password;
},
// Test API connectivity
async testAPI(): Promise<boolean> {
try {
console.log('Testing API connectivity...');
const response = await fetch(`${API_BASE_URL}/users/1`, {
method: 'GET',
headers: {
'Content-Type': 'application/json',
},
});
const isWorking = response.ok;
console.log('API test result:', isWorking ? 'Working' : 'Failed');
return isWorking;
} catch (error) {
console.log('API test failed:', error);
return false;
}
},
// Get current user (from local storage)
async getCurrentUser() {
return await localStorageService.getCurrentUser();
},
// Check if user is logged in
async isLoggedIn(): Promise<boolean> {
const token = await localStorageService.getCurrentToken();
if (!token) return false;
return await localStorageService.isValidSession(token);
},
// Logout user
async logout(): Promise<void> {
await localStorageService.logout();
},
// Get app status
async getAppStatus() {
const storageInfo = await localStorageService.getStorageInfo();
const networkState = networkService.getNetworkState();
const isLoggedIn = await this.isLoggedIn();
return {
network: {
isOnline: networkService.isOnline(),
...networkState,
},
storage: storageInfo,
authentication: {
isLoggedIn,
currentUser: storageInfo.currentUser,
},
};
},
// Clear all local data (for debugging)
async clearAllData(): Promise<void> {
await localStorageService.clearAllData();
},
// Local authentication (works offline)
async localLogin(credentials: LoginRequest, showOfflineMessage = false): Promise<AuthResponse> {
try {
console.log('Attempting local login');
// Validate user locally
const user = await localStorageService.validateUser(credentials.email, credentials.password);
if (user) {
// Create local session
const token = await localStorageService.createSession(user);
const message = showOfflineMessage
? 'Login successful! (Offline mode)'
: 'Login successful! (Local authentication)';
return {
success: true,
message,
user: {
id: user.id,
fullName: user.fullName,
email: user.email,
phone: user.phone,
},
token,
};
}
throw {
success: false,
message: 'Invalid credentials',
error: 'User not found or invalid password',
} as ApiError;
} catch (error) {
console.error('Local login error:', error);
throw error;
}
},
// Background sync with API (non-blocking)
async backgroundSync(credentials: LoginRequest): Promise<void> {
try {
console.log('Background sync with API');
await apiRequest('/login', {
method: 'POST',
body: JSON.stringify({
email: credentials.email,
password: credentials.password,
}),
});
console.log('Background sync successful');
} catch (error) {
console.log('Background sync failed (non-critical):', error);
}
},
// Fallback login for when API is not working (legacy)
async fallbackLogin(credentials: LoginRequest): Promise<AuthResponse> {
return await this.localLogin(credentials, true);
},
// Fallback register for when API is not working
async fallbackRegister(userData: RegisterRequest): Promise<AuthResponse> {
console.log('Using fallback registration');
// Simulate API delay
await new Promise<void>(resolve => setTimeout(resolve, 1000));
// Simple validation
if (!userData.email || !userData.password) {
throw {
success: false,
message: 'Registration failed',
error: 'Email and password are required',
} as ApiError;
}
return {
success: true,
message: 'Registration successful! (Using local authentication)',
user: {
id: '2',
fullName: userData.fullName,
email: userData.email,
phone: userData.phone,
},
token: 'fallback-register-token-67890',
};
},
};

View File

@@ -0,0 +1,296 @@
// Local Storage Service for Offline Data Management
// Works without internet connection
import AsyncStorage from '@react-native-async-storage/async-storage';
import { User } from '../types/auth';
// Storage keys
const STORAGE_KEYS = {
USERS: '@local_users',
CURRENT_USER: '@current_user',
AUTH_TOKEN: '@auth_token',
USER_SESSIONS: '@user_sessions',
APP_SETTINGS: '@app_settings',
};
// Local user data structure
interface LocalUser {
id: string;
fullName: string;
email: string;
phone?: string;
password: string; // Hashed in real app
createdAt: string;
lastLogin?: string;
}
interface UserSession {
userId: string;
token: string;
loginTime: string;
expiresAt: string;
}
// Local Storage Service
export const localStorageService = {
// User Management
async saveUser(userData: {
fullName: string;
email: string;
phone?: string;
password: string;
}): Promise<LocalUser> {
try {
const users = await this.getAllUsers();
// Check if user already exists
const existingUser = users.find(u => u.email === userData.email);
if (existingUser) {
throw new Error('User already exists with this email');
}
// Create new user
const newUser: LocalUser = {
id: Date.now().toString(),
fullName: userData.fullName,
email: userData.email,
phone: userData.phone,
password: userData.password, // In real app, hash this
createdAt: new Date().toISOString(),
};
// Add to users list
users.push(newUser);
await AsyncStorage.setItem(STORAGE_KEYS.USERS, JSON.stringify(users));
console.log('User saved locally:', { email: newUser.email, id: newUser.id });
return newUser;
} catch (error) {
console.error('Error saving user:', error);
throw error;
}
},
async getAllUsers(): Promise<LocalUser[]> {
try {
const usersJson = await AsyncStorage.getItem(STORAGE_KEYS.USERS);
return usersJson ? JSON.parse(usersJson) : [];
} catch (error) {
console.error('Error getting users:', error);
return [];
}
},
async findUserByEmail(email: string): Promise<LocalUser | null> {
try {
const users = await this.getAllUsers();
return users.find(u => u.email === email) || null;
} catch (error) {
console.error('Error finding user:', error);
return null;
}
},
async validateUser(email: string, password: string): Promise<LocalUser | null> {
try {
const user = await this.findUserByEmail(email);
if (user && user.password === password) {
// Update last login
user.lastLogin = new Date().toISOString();
await this.updateUser(user);
return user;
}
return null;
} catch (error) {
console.error('Error validating user:', error);
return null;
}
},
async updateUser(userData: LocalUser): Promise<void> {
try {
const users = await this.getAllUsers();
const userIndex = users.findIndex(u => u.id === userData.id);
if (userIndex !== -1) {
users[userIndex] = userData;
await AsyncStorage.setItem(STORAGE_KEYS.USERS, JSON.stringify(users));
}
} catch (error) {
console.error('Error updating user:', error);
throw error;
}
},
// Session Management
async createSession(user: LocalUser): Promise<string> {
try {
const token = `local_token_${user.id}_${Date.now()}`;
const session: UserSession = {
userId: user.id,
token,
loginTime: new Date().toISOString(),
expiresAt: new Date(Date.now() + 30 * 24 * 60 * 60 * 1000).toISOString(), // 30 days
};
// Save current session
await AsyncStorage.setItem(STORAGE_KEYS.AUTH_TOKEN, token);
await AsyncStorage.setItem(STORAGE_KEYS.CURRENT_USER, JSON.stringify(user));
// Save to sessions history
const sessions = await this.getAllSessions();
sessions.push(session);
await AsyncStorage.setItem(STORAGE_KEYS.USER_SESSIONS, JSON.stringify(sessions));
console.log('Session created:', { userId: user.id, token });
return token;
} catch (error) {
console.error('Error creating session:', error);
throw error;
}
},
async getCurrentUser(): Promise<User | null> {
try {
const userJson = await AsyncStorage.getItem(STORAGE_KEYS.CURRENT_USER);
if (userJson) {
const localUser: LocalUser = JSON.parse(userJson);
// Convert to User format (without password)
return {
id: localUser.id,
fullName: localUser.fullName,
email: localUser.email,
phone: localUser.phone,
};
}
return null;
} catch (error) {
console.error('Error getting current user:', error);
return null;
}
},
async getCurrentToken(): Promise<string | null> {
try {
return await AsyncStorage.getItem(STORAGE_KEYS.AUTH_TOKEN);
} catch (error) {
console.error('Error getting current token:', error);
return null;
}
},
async getAllSessions(): Promise<UserSession[]> {
try {
const sessionsJson = await AsyncStorage.getItem(STORAGE_KEYS.USER_SESSIONS);
return sessionsJson ? JSON.parse(sessionsJson) : [];
} catch (error) {
console.error('Error getting sessions:', error);
return [];
}
},
async isValidSession(token: string): Promise<boolean> {
try {
const sessions = await this.getAllSessions();
const session = sessions.find(s => s.token === token);
if (!session) return false;
// Check if session is expired
const now = new Date();
const expiresAt = new Date(session.expiresAt);
return now < expiresAt;
} catch (error) {
console.error('Error validating session:', error);
return false;
}
},
// Logout
async logout(): Promise<void> {
try {
await AsyncStorage.removeItem(STORAGE_KEYS.AUTH_TOKEN);
await AsyncStorage.removeItem(STORAGE_KEYS.CURRENT_USER);
console.log('User logged out locally');
} catch (error) {
console.error('Error during logout:', error);
throw error;
}
},
// Data Management
async clearAllData(): Promise<void> {
try {
await AsyncStorage.multiRemove([
STORAGE_KEYS.USERS,
STORAGE_KEYS.CURRENT_USER,
STORAGE_KEYS.AUTH_TOKEN,
STORAGE_KEYS.USER_SESSIONS,
]);
console.log('All local data cleared');
} catch (error) {
console.error('Error clearing data:', error);
throw error;
}
},
async getStorageInfo(): Promise<{
totalUsers: number;
currentUser: string | null;
activeSessions: number;
hasValidSession: boolean;
}> {
try {
const users = await this.getAllUsers();
const currentUser = await this.getCurrentUser();
const sessions = await this.getAllSessions();
const token = await this.getCurrentToken();
const hasValidSession = token ? await this.isValidSession(token) : false;
return {
totalUsers: users.length,
currentUser: currentUser?.email || null,
activeSessions: sessions.length,
hasValidSession,
};
} catch (error) {
console.error('Error getting storage info:', error);
return {
totalUsers: 0,
currentUser: null,
activeSessions: 0,
hasValidSession: false,
};
}
},
// Initialize with default test user
async initializeDefaultUsers(): Promise<void> {
try {
const users = await this.getAllUsers();
// Add default test user if no users exist
if (users.length === 0) {
await this.saveUser({
fullName: 'Test User',
email: 'test@example.com',
phone: '+1234567890',
password: 'password123',
});
// Also add the ReqRes test user
await this.saveUser({
fullName: 'Eve Holt',
email: 'eve.holt@reqres.in',
phone: '+1987654321',
password: 'cityslicka',
});
console.log('Default test users created');
}
} catch (error) {
console.error('Error initializing default users:', error);
}
},
};

View File

@@ -0,0 +1,136 @@
// Network Detection Service
// Detects online/offline status and manages connectivity
import NetInfo from '@react-native-community/netinfo';
interface NetworkState {
isConnected: boolean;
isInternetReachable: boolean;
type: string;
}
export class NetworkService {
private static instance: NetworkService;
private networkState: NetworkState = {
isConnected: false,
isInternetReachable: false,
type: 'unknown',
};
private listeners: ((state: NetworkState) => void)[] = [];
static getInstance(): NetworkService {
if (!NetworkService.instance) {
NetworkService.instance = new NetworkService();
}
return NetworkService.instance;
}
constructor() {
this.initialize();
}
private async initialize() {
try {
// Get initial network state
const state = await NetInfo.fetch();
this.updateNetworkState(state);
// Listen for network changes
NetInfo.addEventListener(state => {
this.updateNetworkState(state);
});
} catch (error) {
console.error('Error initializing network service:', error);
// Fallback to assuming offline
this.networkState = {
isConnected: false,
isInternetReachable: false,
type: 'unknown',
};
}
}
private updateNetworkState(state: any) {
const newState: NetworkState = {
isConnected: state.isConnected ?? false,
isInternetReachable: state.isInternetReachable ?? false,
type: state.type || 'unknown',
};
const wasOnline = this.networkState.isConnected;
const isNowOnline = newState.isConnected;
this.networkState = newState;
// Log network changes
if (wasOnline !== isNowOnline) {
console.log(`Network status changed: ${isNowOnline ? 'ONLINE' : 'OFFLINE'}`);
}
// Notify listeners
this.listeners.forEach(listener => {
try {
listener(newState);
} catch (error) {
console.error('Error in network listener:', error);
}
});
}
// Get current network state
getNetworkState(): NetworkState {
return { ...this.networkState };
}
// Check if device is online
isOnline(): boolean {
return this.networkState.isConnected && this.networkState.isInternetReachable !== false;
}
// Check if device is offline
isOffline(): boolean {
return !this.isOnline();
}
// Add network state listener
addListener(listener: (state: NetworkState) => void): () => void {
this.listeners.push(listener);
// Return unsubscribe function
return () => {
const index = this.listeners.indexOf(listener);
if (index > -1) {
this.listeners.splice(index, 1);
}
};
}
// Test internet connectivity
async testConnectivity(): Promise<boolean> {
try {
// Try to fetch a small resource
const controller = new AbortController();
const timeoutId = setTimeout(() => controller.abort(), 5000);
const response = await fetch('https://httpbin.org/status/200', {
method: 'HEAD',
signal: controller.signal,
});
clearTimeout(timeoutId);
return response.ok;
} catch (error) {
console.log('Connectivity test failed:', error);
return false;
}
}
// Get network info for debugging
getNetworkInfo(): string {
const state = this.networkState;
return `Connected: ${state.isConnected}, Internet: ${state.isInternetReachable}, Type: ${state.type}`;
}
}
// Export singleton instance
export const networkService = NetworkService.getInstance();

34
src/types/auth.ts Normal file
View File

@@ -0,0 +1,34 @@
// Simple authentication types for open API
export interface LoginRequest {
email: string;
password: string;
}
export interface RegisterRequest {
fullName: string;
email: string;
phone: string;
password: string;
}
export interface User {
id: string;
fullName: string;
email: string;
phone?: string;
}
export interface AuthResponse {
success: boolean;
message: string;
user?: User;
token?: string;
error?: string;
}
export interface ApiError {
success: false;
message: string;
error: string;
}

5
tsconfig.json Normal file
View File

@@ -0,0 +1,5 @@
{
"extends": "@react-native/typescript-config",
"include": ["**/*.ts", "**/*.tsx"],
"exclude": ["**/node_modules", "**/Pods"]
}