├── .gitattributes ├── assets ├── AppIcon.png └── splash.png ├── Resources ├── light.png ├── AIAvatar.jpeg ├── background.png ├── loaderAnimation.json └── WCIntro.json ├── ios ├── Podfile.properties.json ├── CMSChats │ ├── Images.xcassets │ │ ├── Contents.json │ │ ├── SplashScreen.imageset │ │ │ ├── splashscreen.png │ │ │ └── Contents.json │ │ ├── SplashScreenBackground.imageset │ │ │ ├── background.png │ │ │ └── Contents.json │ │ └── AppIcon.appiconset │ │ │ └── Contents.json │ ├── AppDelegate.h │ ├── main.m │ ├── Supporting │ │ └── Expo.plist │ ├── Info.plist │ ├── AppDelegate.mm │ └── SplashScreen.storyboard ├── .gitignore ├── .xcode.env ├── Podfile └── CMSChats.xcodeproj │ ├── xcshareddata │ └── xcschemes │ │ └── CMSChats.xcscheme │ └── project.pbxproj ├── README.md ├── Context ├── getRoomId.js ├── assets.js └── authContext.js ├── babel.config.js ├── tailwind.config.js ├── metro.config.js ├── index.js ├── eas.json ├── .gitignore ├── Components ├── ResponsiveKeyboard.js ├── MessageSection.js ├── ChatList.js ├── AIInteraction.js ├── ChatPreview.js ├── AIChatItem.js ├── MessageItem.js └── ChatItem.js ├── app.json ├── .github └── FUNDING.yml ├── Pages ├── PageLoader.js ├── Profile.js ├── Login.js ├── WorldChat.js ├── Home.js ├── Chats.js ├── AI.js └── Signup.js ├── firebase └── config.js ├── App.js ├── package.json └── LICENSE /.gitattributes: -------------------------------------------------------------------------------- 1 | *.pbxproj -text 2 | -------------------------------------------------------------------------------- /assets/AppIcon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/callmesidhu/ReactNative-CMSChats-app/HEAD/assets/AppIcon.png -------------------------------------------------------------------------------- /assets/splash.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/callmesidhu/ReactNative-CMSChats-app/HEAD/assets/splash.png -------------------------------------------------------------------------------- /Resources/light.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/callmesidhu/ReactNative-CMSChats-app/HEAD/Resources/light.png -------------------------------------------------------------------------------- /Resources/AIAvatar.jpeg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/callmesidhu/ReactNative-CMSChats-app/HEAD/Resources/AIAvatar.jpeg -------------------------------------------------------------------------------- /Resources/background.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/callmesidhu/ReactNative-CMSChats-app/HEAD/Resources/background.png -------------------------------------------------------------------------------- /ios/Podfile.properties.json: -------------------------------------------------------------------------------- 1 | { 2 | "expo.jsEngine": "hermes", 3 | "EX_DEV_CLIENT_NETWORK_INSPECTOR": "true" 4 | } 5 | -------------------------------------------------------------------------------- /ios/CMSChats/Images.xcassets/Contents.json: -------------------------------------------------------------------------------- 1 | { 2 | "info" : { 3 | "version" : 1, 4 | "author" : "expo" 5 | } 6 | } 7 | -------------------------------------------------------------------------------- /ios/CMSChats/AppDelegate.h: -------------------------------------------------------------------------------- 1 | #import 2 | #import 3 | #import 4 | 5 | @interface AppDelegate : EXAppDelegateWrapper 6 | 7 | @end 8 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Frontend:- `react native expo` + `nativewind` 2 | # Backend :- `Firebase` + `Gemini API` 3 | ![App](https://github.com/user-attachments/assets/fddfffa8-15e4-42a2-b387-381261fa0cb1) 4 | -------------------------------------------------------------------------------- /ios/CMSChats/Images.xcassets/SplashScreen.imageset/splashscreen.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/callmesidhu/ReactNative-CMSChats-app/HEAD/ios/CMSChats/Images.xcassets/SplashScreen.imageset/splashscreen.png -------------------------------------------------------------------------------- /Context/getRoomId.js: -------------------------------------------------------------------------------- 1 | export const getRoomId = (userId1, userId2)=>{ 2 | const sortedIds = [userId1, userId2].sort(); 3 | const roomId = sortedIds.join('-'); 4 | return roomId; 5 | } -------------------------------------------------------------------------------- /babel.config.js: -------------------------------------------------------------------------------- 1 | module.exports = function(api) { 2 | api.cache(true); 3 | return { 4 | presets: ['babel-preset-expo'], 5 | plugins: ["nativewind/babel", 'react-native-reanimated/plugin'], 6 | }; 7 | }; 8 | -------------------------------------------------------------------------------- /ios/CMSChats/Images.xcassets/SplashScreenBackground.imageset/background.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/callmesidhu/ReactNative-CMSChats-app/HEAD/ios/CMSChats/Images.xcassets/SplashScreenBackground.imageset/background.png -------------------------------------------------------------------------------- /tailwind.config.js: -------------------------------------------------------------------------------- 1 | 2 | 3 | module.exports = { 4 | content: ["./App.{js,jsx,ts,tsx}", "./Pages/**/*.{js,jsx,ts,tsx}","./Components/**/*.{js,jsx,ts,tsx}"], 5 | theme: { 6 | extend: {}, 7 | }, 8 | plugins: [], 9 | } -------------------------------------------------------------------------------- /metro.config.js: -------------------------------------------------------------------------------- 1 | const { getDefaultConfig } = require('@expo/metro-config'); 2 | 3 | const defaultConfig = getDefaultConfig(__dirname); 4 | defaultConfig.resolver.sourceExts.push('cjs'); 5 | 6 | module.exports = defaultConfig; 7 | -------------------------------------------------------------------------------- /ios/CMSChats/main.m: -------------------------------------------------------------------------------- 1 | #import 2 | 3 | #import "AppDelegate.h" 4 | 5 | int main(int argc, char * argv[]) { 6 | @autoreleasepool { 7 | return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class])); 8 | } 9 | } 10 | 11 | -------------------------------------------------------------------------------- /ios/CMSChats/Images.xcassets/AppIcon.appiconset/Contents.json: -------------------------------------------------------------------------------- 1 | { 2 | "images" : [ 3 | { 4 | "idiom" : "universal", 5 | "platform" : "ios", 6 | "size" : "1024x1024" 7 | } 8 | ], 9 | "info" : { 10 | "version" : 1, 11 | "author" : "expo" 12 | } 13 | } -------------------------------------------------------------------------------- /index.js: -------------------------------------------------------------------------------- 1 | import { registerRootComponent } from 'expo'; 2 | 3 | 4 | import App from './App'; 5 | 6 | // registerRootComponent calls AppRegistry.registerComponent('main', () => App); 7 | // It also ensures that whether you load the app in Expo Go or in a native build, 8 | // the environment is set up appropriately 9 | registerRootComponent(App); 10 | -------------------------------------------------------------------------------- /ios/CMSChats/Images.xcassets/SplashScreen.imageset/Contents.json: -------------------------------------------------------------------------------- 1 | { 2 | "images": [ 3 | { 4 | "idiom": "universal", 5 | "filename": "splashscreen.png", 6 | "scale": "1x" 7 | }, 8 | { 9 | "idiom": "universal", 10 | "scale": "2x" 11 | }, 12 | { 13 | "idiom": "universal", 14 | "scale": "3x" 15 | } 16 | ], 17 | "info": { 18 | "version": 1, 19 | "author": "expo" 20 | } 21 | } -------------------------------------------------------------------------------- /ios/CMSChats/Images.xcassets/SplashScreenBackground.imageset/Contents.json: -------------------------------------------------------------------------------- 1 | { 2 | "images": [ 3 | { 4 | "idiom": "universal", 5 | "filename": "background.png", 6 | "scale": "1x" 7 | }, 8 | { 9 | "idiom": "universal", 10 | "scale": "2x" 11 | }, 12 | { 13 | "idiom": "universal", 14 | "scale": "3x" 15 | } 16 | ], 17 | "info": { 18 | "version": 1, 19 | "author": "expo" 20 | } 21 | } -------------------------------------------------------------------------------- /ios/.gitignore: -------------------------------------------------------------------------------- 1 | # OSX 2 | # 3 | .DS_Store 4 | 5 | # Xcode 6 | # 7 | build/ 8 | *.pbxuser 9 | !default.pbxuser 10 | *.mode1v3 11 | !default.mode1v3 12 | *.mode2v3 13 | !default.mode2v3 14 | *.perspectivev3 15 | !default.perspectivev3 16 | xcuserdata 17 | *.xccheckout 18 | *.moved-aside 19 | DerivedData 20 | *.hmap 21 | *.ipa 22 | *.xcuserstate 23 | project.xcworkspace 24 | .xcode.env.local 25 | 26 | # Bundle artifacts 27 | *.jsbundle 28 | 29 | # CocoaPods 30 | /Pods/ 31 | -------------------------------------------------------------------------------- /eas.json: -------------------------------------------------------------------------------- 1 | { 2 | "cli": { 3 | "version": ">= 10.1.1" 4 | }, 5 | "build": { 6 | "development": { 7 | "developmentClient": true, 8 | "distribution": "internal", 9 | "channel": "development" 10 | }, 11 | "preview": { 12 | "distribution": "internal", 13 | "channel": "preview" 14 | }, 15 | "production": { 16 | "channel": "production" 17 | } 18 | }, 19 | "submit": { 20 | "production": {} 21 | } 22 | } 23 | -------------------------------------------------------------------------------- /ios/.xcode.env: -------------------------------------------------------------------------------- 1 | # This `.xcode.env` file is versioned and is used to source the environment 2 | # used when running script phases inside Xcode. 3 | # To customize your local environment, you can create an `.xcode.env.local` 4 | # file that is not versioned. 5 | 6 | # NODE_BINARY variable contains the PATH to the node executable. 7 | # 8 | # Customize the NODE_BINARY variable here. 9 | # For example, to use nvm with brew, add the following line 10 | # . "$(brew --prefix nvm)/nvm.sh" --no-use 11 | export NODE_BINARY=$(command -v node) 12 | -------------------------------------------------------------------------------- /ios/CMSChats/Supporting/Expo.plist: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | EXUpdatesEnabled 6 | 7 | EXUpdatesCheckOnLaunch 8 | ALWAYS 9 | EXUpdatesLaunchWaitMs 10 | 0 11 | EXUpdatesURL 12 | https://u.expo.dev/bb864e40-82d2-4db1-acc3-4560e0664cf8 13 | EXUpdatesRuntimeVersion 14 | 1.0.0 15 | 16 | -------------------------------------------------------------------------------- /Context/assets.js: -------------------------------------------------------------------------------- 1 | export const blurhash = '|rF?hV%2WCj[ayj[a|j[az_NaeWBj@ayfRayfQfQM{M|azj[azf6fQfQfQIpWXofj[ayj[j[fQayWCoeoeaya}j[ayfQa{oLj?j[WVj[ayayj[fQoff7azayj[ayj[j[ayofayayayj[fQj[ayayj[ayfjj[j[ayjuayj['; 2 | 3 | export const API_KEY = "AIzaSyDfs8Xbu3BZOjtcah7Cw4nVJZ3VDPtDNbs"; 4 | 5 | export const formatDate = date=>{ 6 | var day = date.getDate(); 7 | var monthNames = ["Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"] 8 | var month = monthNames [date.getMonth()]; 9 | var formattedDate = day + ' ' + month; 10 | return formattedDate; 11 | } -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # Learn more https://docs.github.com/en/get-started/getting-started-with-git/ignoring-files 2 | 3 | # dependencies 4 | node_modules/ 5 | 6 | # Expo 7 | .expo/ 8 | dist/ 9 | web-build/ 10 | 11 | # Native 12 | *.orig.* 13 | *.jks 14 | *.p8 15 | *.p12 16 | *.key 17 | *.mobileprovision 18 | 19 | # Metro 20 | .metro-health-check* 21 | 22 | # debug 23 | npm-debug.* 24 | yarn-debug.* 25 | yarn-error.* 26 | 27 | # macOS 28 | .DS_Store 29 | *.pem 30 | 31 | # local env files 32 | .env*.local 33 | 34 | # typescript 35 | *.tsbuildinfo 36 | 37 | # @generated expo-cli sync-2b81b286409207a5da26e14c78851eb30d8ccbdb 38 | # The following patterns were generated by expo-cli 39 | 40 | expo-env.d.ts 41 | # @end expo-cli -------------------------------------------------------------------------------- /Components/ResponsiveKeyboard.js: -------------------------------------------------------------------------------- 1 | import { View, Text, Platform, KeyboardAvoidingView } from 'react-native' 2 | import React, { Children } from 'react' 3 | import { ScrollView } from 'react-native' 4 | 5 | 6 | 7 | 8 | export default function ResponsiveKeyboard({children}) { 9 | return ( 10 | 13 | 17 | { 18 | children 19 | } 20 | 21 | 22 | 23 | ) 24 | } -------------------------------------------------------------------------------- /app.json: -------------------------------------------------------------------------------- 1 | { 2 | "expo": { 3 | "name": "CMS Chats", 4 | "slug": "cms-chats", 5 | "version": "1.0.0", 6 | "splash": { 7 | "image": "./assets/splash.png", 8 | "backgroundColor": "#111111" 9 | }, 10 | "icon": "./assets/AppIcon.png", 11 | "plugins": [ 12 | "expo-video", 13 | "expo-font", 14 | "expo-router" 15 | ], 16 | "extra": { 17 | "eas": { 18 | "projectId": "bb864e40-82d2-4db1-acc3-4560e0664cf8" 19 | } 20 | }, 21 | "owner": "callmesidhu", 22 | "updates": { 23 | "url": "https://u.expo.dev/bb864e40-82d2-4db1-acc3-4560e0664cf8" 24 | }, 25 | "android": { 26 | "runtimeVersion": { 27 | "policy": "appVersion" 28 | }, 29 | "package": "com.callmesidhu.cmschats" 30 | }, 31 | "ios": { 32 | "runtimeVersion": "1.0.0" 33 | } 34 | } 35 | } 36 | -------------------------------------------------------------------------------- /Components/MessageSection.js: -------------------------------------------------------------------------------- 1 | import React, { useEffect, useRef } from 'react' 2 | import { ScrollView } from 'react-native' 3 | import MessageItem from './MessageItem' 4 | 5 | export default function MessageSection({messages, currentUser}) { 6 | 7 | 8 | useEffect(()=>{ 9 | scrollToBottom(); 10 | },[messages]); 11 | 12 | const scrollViewRef = useRef(); 13 | const scrollToBottom = () => { 14 | scrollViewRef.current.scrollToEnd({ animated: true }); 15 | }; 16 | 17 | return ( 18 | 19 | { 20 | messages.map((message, index)=>{ 21 | return () 22 | }) 23 | } 24 | 25 | ) 26 | } -------------------------------------------------------------------------------- /Components/ChatList.js: -------------------------------------------------------------------------------- 1 | import { View, FlatList, ScrollView } from 'react-native' 2 | import React, { useEffect, useRef } from 'react' 3 | import ChatItem from './ChatItem' 4 | import { useNavigation } from '@react-navigation/native'; 5 | 6 | export default function ChatList({ users, profile }) { 7 | const navigation = useNavigation(); 8 | 9 | 10 | 11 | 12 | return ( 13 | 14 | Math.random()} 18 | renderItem={({item, index})=>} 25 | /> 26 | 27 | ) 28 | } -------------------------------------------------------------------------------- /.github/FUNDING.yml: -------------------------------------------------------------------------------- 1 | # These are supported funding model platform 2 | 3 | github: # Replace with up to 4 GitHub Sponsors-enabled usernames e.g., [user1, user2] 4 | patreon: # Replace with a single Patreon username 5 | open_collective: # Replace with a single Open Collective username 6 | ko_fi: # Replace with a single Ko-fi username 7 | tidelift: # Replace with a single Tidelift platform-name/package-name e.g., npm/babel 8 | community_bridge: # Replace with a single Community Bridge project-name e.g., cloud-foundry 9 | liberapay: # Replace with a single Liberapay username 10 | issuehunt: # Replace with a single IssueHunt username 11 | lfx_crowdfunding: # Replace with a single LFX Crowdfunding project-name e.g., cloud-foundry 12 | polar: # Replace with a single Polar username 13 | buy_me_a_coffee: # Replace with a single Buy Me a Coffee username 14 | custom: # Replace with up to 4 custom sponsorship URLs e.g., ['link1', 'link2'] 15 | -------------------------------------------------------------------------------- /Pages/PageLoader.js: -------------------------------------------------------------------------------- 1 | import { View, Text, ActivityIndicator} from 'react-native'; 2 | import React, { useEffect, useContext, useRef } from 'react'; 3 | import { useNavigation } from '@react-navigation/native'; 4 | import { AuthContext } from '../Context/authContext'; 5 | import LottieView from 'lottie-react-native'; 6 | 7 | 8 | 9 | 10 | export default function PageLoader() { 11 | const navigation = useNavigation(); 12 | const {isAuthenticated} = useContext(AuthContext); 13 | const animation = useRef(null); 14 | 15 | useEffect(() => { 16 | setTimeout(() => { 17 | if (isAuthenticated==true) { 18 | navigation.navigate("HomePage"); 19 | } else { 20 | navigation.navigate("LoginPage"); 21 | } 22 | }, 2000); 23 | }, [isAuthenticated]); 24 | 25 | 26 | return ( 27 | 28 | 29 | Please wait 3s... or Retry 30 | 31 | ); 32 | }; 33 | -------------------------------------------------------------------------------- /firebase/config.js: -------------------------------------------------------------------------------- 1 | import { initializeApp } from 'firebase/app'; 2 | import { initializeAuth, getReactNativePersistence } from 'firebase/auth'; 3 | import AsyncStorage from '@react-native-async-storage/async-storage'; 4 | import { collection, getFirestore } from 'firebase/firestore'; 5 | import { getStorage } from 'firebase/storage'; 6 | 7 | 8 | const firebaseConfig = { 9 | apiKey: "AIzaSyChiZdO8SReQF0ET0ATOlfDp03owmcsYbA", 10 | authDomain: "reactnative-mobile-chat-app.firebaseapp.com", 11 | projectId: "reactnative-mobile-chat-app", 12 | storageBucket: "reactnative-mobile-chat-app.appspot.com", 13 | messagingSenderId: "356173390112", 14 | appId: "1:356173390112:web:8407200f24674f1da23745", 15 | measurementId: "G-RZS4EPE41G" 16 | }; 17 | 18 | // Initialize Firebase 19 | const app = initializeApp(firebaseConfig); 20 | 21 | // Initialize Firebase Auth with persistence 22 | const auth = initializeAuth(app, { 23 | persistence: getReactNativePersistence(AsyncStorage) 24 | }); 25 | 26 | // Export Firebase services 27 | export const db = getFirestore(app); 28 | export const usersRef = collection(db,'users'); 29 | export const roomRef = collection(db,'rooms'); 30 | export { auth, getFirestore, getStorage }; 31 | 32 | 33 | -------------------------------------------------------------------------------- /Context/authContext.js: -------------------------------------------------------------------------------- 1 | import { createContext, useState, useEffect, useContext } from 'react'; 2 | import { auth, getFirestore} from '../firebase/config'; 3 | import { doc } from 'firebase/firestore'; 4 | import { getDoc } from 'firebase/firestore'; 5 | 6 | 7 | export const AuthContext = createContext(); 8 | 9 | export const AuthContextProvider = ({ children }) => { 10 | const [user, setUser] = useState(null); 11 | const [isAuthenticated, setIsAuthenticated] = useState(undefined); 12 | 13 | useEffect(() => { 14 | const unsubscribe = auth.onAuthStateChanged(async(firebaseUser) => { 15 | if (firebaseUser) { 16 | setIsAuthenticated(true); 17 | setUser(firebaseUser); 18 | } else { 19 | setIsAuthenticated(false); 20 | setUser(null); 21 | } 22 | }); 23 | 24 | return unsubscribe; 25 | }, []); 26 | 27 | const logout = async () => { 28 | try { 29 | await auth.signOut(); 30 | setUser(null); 31 | setIsAuthenticated(false); 32 | } catch (error) { 33 | console.error('Sign-out error:', error); 34 | throw error; 35 | } 36 | }; 37 | 38 | return ( 39 | 40 | {children} 41 | 42 | ); 43 | }; 44 | export const useAuth = () => { 45 | return useContext(AuthContext); 46 | }; 47 | -------------------------------------------------------------------------------- /Components/AIInteraction.js: -------------------------------------------------------------------------------- 1 | import React from 'react'; 2 | import { StyleSheet, Text, TouchableOpacity, View } from 'react-native'; 3 | import { heightPercentageToDP } from 'react-native-responsive-screen'; 4 | import Ionicons from 'react-native-vector-icons/Ionicons'; // Corrected import for Ionicons 5 | 6 | const AIInteraction = ({ role, text, onSpeech }) => { 7 | return ( 8 | 9 | {text} 10 | {role === "model" && ( 11 | 12 | 13 | 14 | )} 15 | 16 | ); 17 | }; 18 | 19 | const styles = StyleSheet.create({ 20 | chatItem: { 21 | marginBottom: 10, 22 | padding: 10, 23 | borderRadius: 10, 24 | maxWidth: "70%", 25 | position: "relative", 26 | }, 27 | userChatItem: { 28 | alignSelf: "flex-end", 29 | backgroundColor: "rgb(34, 211, 238)", 30 | borderColor: '#2196f3', 31 | borderWidth: 1, 32 | }, 33 | modelChatItem: { 34 | alignSelf: "flex-start", 35 | backgroundColor: '#b3e5fc', 36 | borderColor: '#667eea', 37 | borderWidth: 1, 38 | }, 39 | speechButton: { 40 | position: 'absolute', 41 | bottom: 5, 42 | right: 5, 43 | marginLeft: 5, 44 | marginTop: 5, 45 | } 46 | }); 47 | 48 | export default AIInteraction; 49 | 50 | 51 | -------------------------------------------------------------------------------- /App.js: -------------------------------------------------------------------------------- 1 | import React from 'react'; 2 | import 'react-native-gesture-handler'; 3 | import { NavigationContainer } from '@react-navigation/native'; 4 | import { createStackNavigator } from '@react-navigation/stack'; 5 | import Login from './Pages/Login'; 6 | import SignUp from './Pages/Signup'; 7 | import Home from './Pages/Home'; 8 | import PageLoader from './Pages/PageLoader' 9 | import { MenuProvider } from 'react-native-popup-menu'; 10 | import { AuthContextProvider } from './Context/authContext'; 11 | import Chats from './Pages/Chats'; 12 | import Profile from './Pages/Profile' 13 | import AI from './Pages/AI'; 14 | import WorldChat from './Pages/WorldChat'; 15 | 16 | 17 | 18 | const Stack = createStackNavigator(); 19 | 20 | const App = () => { 21 | return ( 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | ); 39 | }; 40 | 41 | export default App; 42 | -------------------------------------------------------------------------------- /ios/CMSChats/Info.plist: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | CADisableMinimumFrameDurationOnPhone 6 | 7 | CFBundleDevelopmentRegion 8 | $(DEVELOPMENT_LANGUAGE) 9 | CFBundleExecutable 10 | $(EXECUTABLE_NAME) 11 | CFBundleIdentifier 12 | $(PRODUCT_BUNDLE_IDENTIFIER) 13 | CFBundleInfoDictionaryVersion 14 | 6.0 15 | CFBundleName 16 | $(PRODUCT_NAME) 17 | CFBundlePackageType 18 | $(PRODUCT_BUNDLE_PACKAGE_TYPE) 19 | CFBundleShortVersionString 20 | 1.0 21 | CFBundleSignature 22 | ???? 23 | CFBundleVersion 24 | 1 25 | LSRequiresIPhoneOS 26 | 27 | NSAppTransportSecurity 28 | 29 | NSAllowsArbitraryLoads 30 | 31 | NSAllowsLocalNetworking 32 | 33 | 34 | UILaunchStoryboardName 35 | SplashScreen 36 | UIRequiredDeviceCapabilities 37 | 38 | arm64 39 | 40 | UIStatusBarStyle 41 | UIStatusBarStyleDefault 42 | UISupportedInterfaceOrientations 43 | 44 | UIInterfaceOrientationPortrait 45 | UIInterfaceOrientationLandscapeLeft 46 | UIInterfaceOrientationLandscapeRight 47 | 48 | UIViewControllerBasedStatusBarAppearance 49 | 50 | 51 | 52 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "cmschats", 3 | "version": "1.0.0", 4 | "main": "index.js", 5 | "scripts": { 6 | "start": "expo start --dev-client", 7 | "android": "expo run:android", 8 | "ios": "expo run:ios", 9 | "web": "expo start --web" 10 | }, 11 | "dependencies": { 12 | "@expo/metro-runtime": "~3.2.1", 13 | "@react-native-async-storage/async-storage": "^1.23.1", 14 | "@react-native-firebase/app": "^20.1.0", 15 | "@react-native-firebase/messaging": "^20.1.0", 16 | "@react-navigation/native": "^6.1.17", 17 | "@react-navigation/stack": "^6.3.29", 18 | "axios": "^1.7.2", 19 | "expo": "~51.0.20", 20 | "expo-dev-client": "~4.0.20", 21 | "expo-device": "^6.0.2", 22 | "expo-image": "~1.12.12", 23 | "expo-image-picker": "~15.0.7", 24 | "expo-notifications": "~0.28.9", 25 | "expo-router": "~3.5.18", 26 | "expo-speech": "~12.0.2", 27 | "expo-status-bar": "~1.12.1", 28 | "expo-updates": "~0.25.19", 29 | "expo-video": "^1.1.10", 30 | "firebase": "^10.12.2", 31 | "lottie-react-native": "6.7.0", 32 | "nativewind": "^2.0.11", 33 | "react": "18.2.0", 34 | "react-dom": "18.2.0", 35 | "react-native": "0.74.3", 36 | "react-native-gesture-handler": "~2.16.1", 37 | "react-native-image-picker": "^7.1.2", 38 | "react-native-loader-kit": "^2.0.8", 39 | "react-native-popup-menu": "^0.16.1", 40 | "react-native-reanimated": "^3.10.1", 41 | "react-native-responsive-screen": "^1.4.2", 42 | "react-native-safe-area-context": "4.10.1", 43 | "react-native-screens": "3.31.1", 44 | "react-native-vector-icons": "^10.1.0", 45 | "react-native-video": "^6.3.0", 46 | "react-native-web": "~0.19.10" 47 | }, 48 | "devDependencies": { 49 | "@babel/core": "^7.20.0", 50 | "tailwindcss": "3.3.2" 51 | }, 52 | "private": true 53 | } 54 | -------------------------------------------------------------------------------- /Components/ChatPreview.js: -------------------------------------------------------------------------------- 1 | import { View, Text, ActivityIndicator } from 'react-native' 2 | import React, {useState, useEffect} from 'react' 3 | import { StatusBar } from 'expo-status-bar'; 4 | import ChatList from './ChatList'; 5 | import { heightPercentageToDP } from 'react-native-responsive-screen'; 6 | import { useAuth } from '../Context/authContext'; 7 | import { getDocs, query, where } from 'firebase/firestore'; 8 | import {usersRef} from '../firebase/config' 9 | import Animated, { FadeInDown, SlideInDown } from 'react-native-reanimated'; 10 | import LottieView from 'lottie-react-native'; 11 | 12 | 13 | export default function ChatPreview({profile}) { 14 | const { user } = useAuth(); 15 | const [users, setUsers] = useState([]); 16 | const getUsers = async()=>{ 17 | const q = query(usersRef, where('userId','!=',user?.uid)); 18 | const qureySnapshot = await getDocs(q); 19 | let data = []; 20 | qureySnapshot.forEach(doc=>{ 21 | data.push({...doc.data()}); 22 | }); 23 | setUsers(data); 24 | } 25 | useEffect(()=>{ 26 | if(user?.uid){ 27 | getUsers(); 28 | } 29 | 30 | },[user?.uid]) 31 | 32 | return ( 33 | 34 | 35 | { 36 | users.length>0? ( 37 | 38 | ):( 39 | 40 | 41 | 42 | ) 43 | } 44 | 45 | ) 46 | } -------------------------------------------------------------------------------- /Components/AIChatItem.js: -------------------------------------------------------------------------------- 1 | import React, { useEffect, useRef, useState } from 'react'; 2 | import{ ScrollView, StyleSheet, Text, TouchableOpacity, View} from 'react-native'; 3 | import { blurhash, formatDate } from '../Context/assets'; 4 | import { heightPercentageToDP, widthPercentageToDP } from 'react-native-responsive-screen'; 5 | import { Image } from 'expo-image'; 6 | import { useNavigation } from '@react-navigation/native'; 7 | import LottieView from 'lottie-react-native'; 8 | 9 | export default function AIChatItem() { 10 | 11 | 12 | const navigation = useNavigation(); 13 | const animation =useRef(null) 14 | const openChatRoom = () =>{ 15 | navigation.navigate('AIPage') 16 | } 17 | 18 | return ( 19 | 20 | 21 | 29 | 30 | 31 | 32 | Android Kunjappy 33 | 34 |   Chat with our AI 35 | 36 | 37 | 42 | 43 | 44 | 45 | 46 | 47 | 48 | 49 | ); 50 | } 51 | -------------------------------------------------------------------------------- /ios/Podfile: -------------------------------------------------------------------------------- 1 | require File.join(File.dirname(`node --print "require.resolve('expo/package.json')"`), "scripts/autolinking") 2 | require File.join(File.dirname(`node --print "require.resolve('react-native/package.json')"`), "scripts/react_native_pods") 3 | 4 | require 'json' 5 | podfile_properties = JSON.parse(File.read(File.join(__dir__, 'Podfile.properties.json'))) rescue {} 6 | 7 | ENV['RCT_NEW_ARCH_ENABLED'] = podfile_properties['newArchEnabled'] == 'true' ? '1' : '0' 8 | ENV['EX_DEV_CLIENT_NETWORK_INSPECTOR'] = podfile_properties['EX_DEV_CLIENT_NETWORK_INSPECTOR'] 9 | 10 | platform :ios, podfile_properties['ios.deploymentTarget'] || '13.4' 11 | install! 'cocoapods', 12 | :deterministic_uuids => false 13 | 14 | prepare_react_native_project! 15 | 16 | target 'CMSChats' do 17 | use_expo_modules! 18 | config = use_native_modules! 19 | 20 | use_frameworks! :linkage => podfile_properties['ios.useFrameworks'].to_sym if podfile_properties['ios.useFrameworks'] 21 | use_frameworks! :linkage => ENV['USE_FRAMEWORKS'].to_sym if ENV['USE_FRAMEWORKS'] 22 | 23 | use_react_native!( 24 | :path => config[:reactNativePath], 25 | :hermes_enabled => podfile_properties['expo.jsEngine'] == nil || podfile_properties['expo.jsEngine'] == 'hermes', 26 | # An absolute path to your application root. 27 | :app_path => "#{Pod::Config.instance.installation_root}/..", 28 | :privacy_file_aggregation_enabled => podfile_properties['apple.privacyManifestAggregationEnabled'] != 'false', 29 | ) 30 | 31 | post_install do |installer| 32 | react_native_post_install( 33 | installer, 34 | config[:reactNativePath], 35 | :mac_catalyst_enabled => false, 36 | :ccache_enabled => podfile_properties['apple.ccacheEnabled'] == 'true', 37 | ) 38 | 39 | # This is necessary for Xcode 14, because it signs resource bundles by default 40 | # when building for devices. 41 | installer.target_installation_results.pod_target_installation_results 42 | .each do |pod_name, target_installation_result| 43 | target_installation_result.resource_bundle_targets.each do |resource_bundle_target| 44 | resource_bundle_target.build_configurations.each do |config| 45 | config.build_settings['CODE_SIGNING_ALLOWED'] = 'NO' 46 | end 47 | end 48 | end 49 | end 50 | 51 | post_integrate do |installer| 52 | begin 53 | expo_patch_react_imports!(installer) 54 | rescue => e 55 | Pod::UI.warn e 56 | end 57 | end 58 | end 59 | -------------------------------------------------------------------------------- /Components/MessageItem.js: -------------------------------------------------------------------------------- 1 | import { View, Text } from 'react-native' 2 | import React from 'react' 3 | import { heightPercentageToDP, widthPercentageToDP } from 'react-native-responsive-screen' 4 | import { blurhash } from '../Context/assets' 5 | import { Image } from 'expo-image' 6 | 7 | export default function MessageItem({message, currentUser}) { 8 | if(currentUser?.uid==message?.userId){ 9 | return ( 10 | 11 | 12 | 20 | 21 | {message?.text} 22 | 23 | 24 | 25 | ) 26 | }else{ 27 | return( 28 | 29 | 30 | 38 | 39 | {message?.text} 40 | 41 | 42 | 43 | ) 44 | } 45 | } -------------------------------------------------------------------------------- /ios/CMSChats/AppDelegate.mm: -------------------------------------------------------------------------------- 1 | #import "AppDelegate.h" 2 | 3 | #import 4 | #import 5 | 6 | @implementation AppDelegate 7 | 8 | - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions 9 | { 10 | self.moduleName = @"main"; 11 | 12 | // You can add your custom initial props in the dictionary below. 13 | // They will be passed down to the ViewController used by React Native. 14 | self.initialProps = @{}; 15 | 16 | return [super application:application didFinishLaunchingWithOptions:launchOptions]; 17 | } 18 | 19 | - (NSURL *)sourceURLForBridge:(RCTBridge *)bridge 20 | { 21 | return [self bundleURL]; 22 | } 23 | 24 | - (NSURL *)bundleURL 25 | { 26 | #if DEBUG 27 | return [[RCTBundleURLProvider sharedSettings] jsBundleURLForBundleRoot:@".expo/.virtual-metro-entry"]; 28 | #else 29 | return [[NSBundle mainBundle] URLForResource:@"main" withExtension:@"jsbundle"]; 30 | #endif 31 | } 32 | 33 | // Linking API 34 | - (BOOL)application:(UIApplication *)application openURL:(NSURL *)url options:(NSDictionary *)options { 35 | return [super application:application openURL:url options:options] || [RCTLinkingManager application:application openURL:url options:options]; 36 | } 37 | 38 | // Universal Links 39 | - (BOOL)application:(UIApplication *)application continueUserActivity:(nonnull NSUserActivity *)userActivity restorationHandler:(nonnull void (^)(NSArray> * _Nullable))restorationHandler { 40 | BOOL result = [RCTLinkingManager application:application continueUserActivity:userActivity restorationHandler:restorationHandler]; 41 | return [super application:application continueUserActivity:userActivity restorationHandler:restorationHandler] || result; 42 | } 43 | 44 | // Explicitly define remote notification delegates to ensure compatibility with some third-party libraries 45 | - (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken 46 | { 47 | return [super application:application didRegisterForRemoteNotificationsWithDeviceToken:deviceToken]; 48 | } 49 | 50 | // Explicitly define remote notification delegates to ensure compatibility with some third-party libraries 51 | - (void)application:(UIApplication *)application didFailToRegisterForRemoteNotificationsWithError:(NSError *)error 52 | { 53 | return [super application:application didFailToRegisterForRemoteNotificationsWithError:error]; 54 | } 55 | 56 | // Explicitly define remote notification delegates to ensure compatibility with some third-party libraries 57 | - (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo fetchCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler 58 | { 59 | return [super application:application didReceiveRemoteNotification:userInfo fetchCompletionHandler:completionHandler]; 60 | } 61 | 62 | @end 63 | -------------------------------------------------------------------------------- /Pages/Profile.js: -------------------------------------------------------------------------------- 1 | import { View, Text, TouchableOpacity, Linking } from 'react-native' 2 | import React, { useEffect, useState } from 'react' 3 | import { useNavigation } from '@react-navigation/native'; 4 | import Icon from 'react-native-vector-icons/FontAwesome'; 5 | import { heightPercentageToDP } from 'react-native-responsive-screen'; 6 | import { blurhash } from '../Context/assets'; 7 | import { Image } from 'expo-image'; 8 | import { getDocs, query, where } from 'firebase/firestore'; 9 | import { usersRef } from '../firebase/config'; 10 | import { useAuth } from '../Context/authContext'; 11 | import Animated, { FlipInEasyX } from 'react-native-reanimated'; 12 | 13 | 14 | export default function Profile() { 15 | const navigation = useNavigation(); 16 | const { user } = useAuth(); 17 | const [profile , setProfile] = useState(['']); 18 | const openURL = (url) => { 19 | Linking.openURL(url).catch((err) => console.error('An error occurred', err)); 20 | }; 21 | const getUser = async () => { 22 | const q = query(usersRef, where('userId', '==', user?.uid)); 23 | const querySnapshot = await getDocs(q); 24 | let data = []; 25 | querySnapshot.forEach(doc => { 26 | data.push({ ...doc.data() }); 27 | }); 28 | setProfile(data[0]); 29 | }; 30 | useEffect(()=>{ 31 | if(user?.uid){ 32 | getUser(); 33 | } 34 | 35 | },[user?.uid]); 36 | return ( 37 | 38 | 39 | 40 | 41 | navigation.goBack()}> 42 | 43 | 44 | Profile 45 | 46 | 47 | 55 | 56 | 57 | Name: 58 | {profile?.name} 59 | 60 | 61 | Email: {profile?.email} 62 | 63 | 64 | 65 | 66 | openURL('https://www.instagram.com/callmesidhu__/')} className='text-center font-semibold text-white mb-4'> © V2.1 Developer : callmesidhu 67 | 68 | ) 69 | } -------------------------------------------------------------------------------- /Pages/Login.js: -------------------------------------------------------------------------------- 1 | import React, { useState } from 'react'; 2 | import { View, Text, TextInput, TouchableOpacity, Alert } from 'react-native'; 3 | import { StatusBar } from 'expo-status-bar'; 4 | import Animated, { FadeInDown, FadeInUp } from 'react-native-reanimated'; 5 | import { useNavigation } from '@react-navigation/native'; 6 | import { auth } from '../firebase/config'; 7 | import { signInWithEmailAndPassword } from 'firebase/auth'; 8 | 9 | function Login() { 10 | const navigation = useNavigation(); 11 | 12 | const [email, setEmail] = useState(''); 13 | const [password, setPassword] = useState(''); 14 | 15 | 16 | const handleLogin = async () => { 17 | if (!email || !password) { 18 | Alert.alert('All fields are required!'); 19 | return; 20 | } 21 | 22 | try { 23 | await signInWithEmailAndPassword(auth, email, password); 24 | navigation.navigate("Loader"); 25 | } catch (error) { 26 | Alert.alert("Invalid email or password!"); 27 | } 28 | } 29 | 30 | return ( 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | 40 | 41 | 42 | Login 43 | 44 | 45 | 46 | 47 | 48 | 54 | 55 | 56 | 63 | 64 | 65 | 66 | Login 67 | 68 | 69 | 70 | Don't have an account? 71 | navigation.navigate('SignUpPage')}> 72 | SignUp 73 | 74 | 75 | 76 | 77 | 78 | ); 79 | } 80 | 81 | export default Login; 82 | -------------------------------------------------------------------------------- /ios/CMSChats.xcodeproj/xcshareddata/xcschemes/CMSChats.xcscheme: -------------------------------------------------------------------------------- 1 | 2 | 5 | 8 | 9 | 15 | 21 | 22 | 23 | 24 | 25 | 30 | 31 | 33 | 39 | 40 | 41 | 42 | 43 | 53 | 55 | 61 | 62 | 63 | 64 | 70 | 72 | 78 | 79 | 80 | 81 | 83 | 84 | 87 | 88 | 89 | -------------------------------------------------------------------------------- /Components/ChatItem.js: -------------------------------------------------------------------------------- 1 | import React, { useEffect, useRef, useState } from 'react'; 2 | import{ ScrollView, StyleSheet, Text, TouchableOpacity, View} from 'react-native'; 3 | import { blurhash, formatDate } from '../Context/assets'; 4 | import { heightPercentageToDP, widthPercentageToDP } from 'react-native-responsive-screen'; 5 | import { Image } from 'expo-image'; 6 | import { collection, doc, getDocs, onSnapshot, orderBy, query } from 'firebase/firestore'; 7 | import { getRoomId } from '../Context/getRoomId'; 8 | import { db } from '../firebase/config'; 9 | import { useAuth } from '../Context/authContext'; 10 | import * as Device from 'expo-device' 11 | import * as Notifications from 'expo-notifications'; 12 | 13 | export default function ChatItem({ profile, item, navigation }) { 14 | /* 15 | Notifications.setNotificationHandler({ 16 | handleNotification: async () => ({ 17 | shouldShowAlert: true, 18 | shouldPlaySound: false, 19 | shouldSetBadge: false, 20 | }), 21 | }); 22 | */ 23 | 24 | //below is lastmessage section and above is notification section 25 | const [lastMessage ,setLastMessage]= useState(undefined); 26 | const {user} = useAuth(); 27 | 28 | useEffect(()=>{ 29 | let roomId = getRoomId(user?.uid, item?.userId); 30 | const docRef = doc(db, 'rooms',roomId); 31 | const messageRef = collection(docRef,'messages'); 32 | const q = query(messageRef,orderBy('createdAt','desc')); 33 | 34 | let unsubscribe = onSnapshot(q, (snapshot)=>{ 35 | let allMessages = snapshot.docs.map(doc=>{ 36 | return doc.data(); 37 | }); 38 | setLastMessage(allMessages[0]); 39 | 40 | 41 | 42 | }) 43 | 44 | return unsubscribe; 45 | },[]) 46 | 47 | 48 | const openChatRoom = () =>{ 49 | navigation.navigate('ChatPage',{item}) 50 | } 51 | 52 | const renderTime =()=>{ 53 | if(typeof lastMessage == undefined){ 54 | return 'Time'; 55 | } else if(lastMessage){ 56 | let date = lastMessage.createdAt; 57 | return formatDate(new Date(date?.seconds * 1000)); 58 | } 59 | 60 | } 61 | 62 | const renderLastMessage =()=>{ 63 | if(typeof lastMessage == undefined){ 64 | return "Loading..."; 65 | } 66 | else if (lastMessage) { 67 | if(user.uid == lastMessage.userId){ 68 | return "You: "+lastMessage?.text; 69 | } else { 70 | return lastMessage?.text; 71 | } 72 | } else { 73 | return " Say Hi 👋" 74 | } 75 | } 76 | 77 | 78 | return ( 79 | 80 | 88 | 89 | 90 | {item?.name} 91 | {renderTime()} 92 | 93 | 94 | {renderLastMessage()} 95 | 96 | 97 | 98 | 99 | ); 100 | } 101 | -------------------------------------------------------------------------------- /Pages/WorldChat.js: -------------------------------------------------------------------------------- 1 | import React, { useRef, useState } from 'react'; 2 | import { View, Text, TouchableOpacity, TextInput, FlatList, StyleSheet } from 'react-native'; 3 | import Icon from 'react-native-vector-icons/FontAwesome'; 4 | import { useNavigation } from '@react-navigation/native'; 5 | import { heightPercentageToDP } from 'react-native-responsive-screen'; 6 | import { Image } from 'expo-image'; 7 | import Animated, { SlideInDown, SlideInRight } from 'react-native-reanimated'; 8 | import axios from 'axios'; 9 | import { speak, isSpeakingAsync, stop } from 'expo-speech'; 10 | import { blurhash } from '../Context/assets'; 11 | import AIInteraction from '../Components/AIInteraction'; 12 | import LottieView from 'lottie-react-native'; 13 | 14 | export default function WorldChat() { 15 | const navigation = useNavigation(); 16 | const [chat, setChat] = useState([]); 17 | const [userInput, setUserInput] = useState(""); 18 | const [loading, setLoading] = useState(false); 19 | const [isSpeaking, setIsSpeaking] = useState(false); 20 | const [error, setError] = useState(null); 21 | const animation = useRef(null); 22 | const [intro, setIntro] = useState(true) 23 | 24 | 25 | return ( 26 | 27 | 28 | navigation.goBack()}> 29 | 30 | 31 | 32 | 33 | 34 | World Chat 35 | 36 | 37 | 38 | 39 | 40 | {intro?( 41 | 42 | 43 | 44 | World Chat Comming Soon... 45 | 46 | 47 | 48 | ):( 49 | 52 | )} 53 | {loading && 54 | 55 | 56 | } 57 | {error && {error}} 58 | 59 | 60 | 61 | 62 | 63 | 64 | 71 | 72 | 73 | 74 | 75 | 76 | 77 | 78 | 79 | ); 80 | }; 81 | -------------------------------------------------------------------------------- /Pages/Home.js: -------------------------------------------------------------------------------- 1 | import React, { useState, useEffect, useRef } from 'react'; 2 | import { View, Text, Alert, TouchableOpacity } from 'react-native'; 3 | import { Image } from 'expo-image'; 4 | import { Menu, MenuOptions, MenuOption, MenuTrigger } from 'react-native-popup-menu'; 5 | import { widthPercentageToDP as wp, heightPercentageToDP as hp } from 'react-native-responsive-screen'; 6 | import Icon from 'react-native-vector-icons/FontAwesome'; 7 | import { useAuth } from '../Context/authContext'; 8 | import ChatPreview from '../Components/ChatPreview'; 9 | import { blurhash } from '../Context/assets'; 10 | import { getDocs, query, where } from 'firebase/firestore'; 11 | import { usersRef } from '../firebase/config'; 12 | import { useNavigation } from '@react-navigation/native'; 13 | import Animated, { BounceIn, FadeInDown, FadeInRight, FadeInUp, LightSpeedInRight, SlideInUp } from 'react-native-reanimated'; 14 | import AIChatItem from '../Components/AIChatItem'; 15 | import LottieView from 'lottie-react-native'; 16 | 17 | 18 | export default function Home() { 19 | const { user,logout } = useAuth(); 20 | const navigation = useNavigation(); 21 | const animation = useRef(null); 22 | 23 | const handleLogout = async () => { 24 | try { 25 | await logout(); 26 | navigation.navigate('Loader'); 27 | } catch (error) { 28 | console.error('Logout error:', error); 29 | } 30 | }; 31 | 32 | const viewProfile = () =>{ 33 | navigation.navigate('ProfilePage'); 34 | } 35 | 36 | const [profile , setProfile] = useState(['']); 37 | const getUser = async () => { 38 | const q = query(usersRef, where('userId', '==', user?.uid)); 39 | const querySnapshot = await getDocs(q); 40 | let data = []; 41 | querySnapshot.forEach(doc => { 42 | data.push({ ...doc.data() }); 43 | }); 44 | setProfile(data[0]); 45 | }; 46 | useEffect(()=>{ 47 | if(user?.uid){ 48 | getUser(); 49 | } 50 | 51 | },[user?.uid]); 52 | 53 | 54 | 55 | return ( 56 | 57 | 58 | Chats 59 | 60 | 61 | 62 | 70 | 71 | 72 | 73 | 74 | 75 | Profile 76 | 77 | 78 | 79 | 80 | 81 | Log Out 82 | 83 | 84 | 85 | 86 | 87 | 88 | 89 | 90 | 91 | 92 | 93 | navigation.navigate('WorldChatPage')}> 94 | 95 | 96 | 97 | 98 | 99 | 100 | ); 101 | } 102 | -------------------------------------------------------------------------------- /ios/CMSChats/SplashScreen.storyboard: -------------------------------------------------------------------------------- 1 | 2 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 31 | 39 | 40 | 41 | 42 | 53 | 54 | 55 | 66 | 67 | 68 | 69 | 70 | 71 | 72 | 73 | 74 | 75 | 76 | 77 | 78 | 79 | 80 | 81 | 82 | 83 | 84 | 85 | 86 | 87 | 88 | 89 | 90 | 91 | 92 | -------------------------------------------------------------------------------- /Resources/loaderAnimation.json: -------------------------------------------------------------------------------- 1 | {"v":"5.5.8","fr":29.9700012207031,"ip":0,"op":82.0000033399285,"w":232,"h":231,"nm":"Comp 1","ddd":0,"assets":[],"layers":[{"ddd":0,"ind":1,"ty":4,"nm":"Shape Layer 1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[116,116,0],"ix":2},"a":{"a":0,"k":[0,-1,0],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0.833,0.833,0.833],"y":[1,1,1]},"o":{"x":[0.39,0.39,0.39],"y":[0,0,0]},"t":0,"s":[64,64,100]},{"i":{"x":[0.1,0.1,0.1],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":23,"s":[46,46,100]},{"i":{"x":[0.833,0.833,0.833],"y":[1,1,1]},"o":{"x":[0.9,0.9,0.9],"y":[0,0,0]},"t":45,"s":[64,64,100]},{"t":81.0000032991976,"s":[64,64,100]}],"ix":6,"x":"var $bm_rt;\n$bm_rt = loopOut();"}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":0,"k":[192,192],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,0.823528992896,0,1],"ix":3},"o":{"a":0,"k":0,"ix":4},"w":{"a":0,"k":5,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"gf","o":{"a":0,"k":100,"ix":10},"r":1,"bm":0,"g":{"p":3,"k":{"a":0,"k":[0,0.12156862745098039,0.6352941176470588,1,0.5,0.07058823529411765,0.8470588235294118,0.9803921568627451,1,0.5529411764705883,0.8274509803921568,0.8156862745098039],"ix":9}},"s":{"a":0,"k":[0,-98],"ix":5},"e":{"a":0,"k":[0,96],"ix":6},"t":1,"nm":"Gradient Fill 1","mn":"ADBE Vector Graphic - G-Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,-1],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":300.00001221925,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":4,"nm":"Shape Layer 2","sr":1,"ks":{"o":{"a":0,"k":19,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[116,116,0],"ix":2},"a":{"a":0,"k":[0,-1,0],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0.833,0.833,0.833],"y":[1,1,1.065]},"o":{"x":[0.39,0.39,0.39],"y":[0,0,0]},"t":0,"s":[94,94,100]},{"i":{"x":[0.1,0.1,0.1],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,-0.114]},"t":23,"s":[47,47,100]},{"i":{"x":[0.833,0.833,0.833],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":63,"s":[94,94,100]},{"t":81.0000032991976,"s":[94,94,100]}],"ix":6,"x":"var $bm_rt;\n$bm_rt = loopOut();"}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":0,"k":[192,192],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,0.823528992896,0,1],"ix":3},"o":{"a":0,"k":0,"ix":4},"w":{"a":0,"k":5,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"gf","o":{"a":0,"k":100,"ix":10},"r":1,"bm":0,"g":{"p":3,"k":{"a":0,"k":[0,0.12156862745098039,0.6352941176470588,1,0.5,0.07058823529411765,0.8470588235294118,0.9803921568627451,1,0.5529411764705883,0.8274509803921568,0.8156862745098039],"ix":9}},"s":{"a":0,"k":[0,-98],"ix":5},"e":{"a":0,"k":[0,96],"ix":6},"t":1,"nm":"Gradient Fill 1","mn":"ADBE Vector Graphic - G-Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,-1],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":300.00001221925,"st":0,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"Shape Layer 4","sr":1,"ks":{"o":{"a":0,"k":12,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[116,116,0],"ix":2},"a":{"a":0,"k":[0,-1,0],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0.833,0.833,0.833],"y":[1,1,1]},"o":{"x":[0.39,0.39,0.39],"y":[0,0,0]},"t":0,"s":[119,119,100]},{"i":{"x":[0,0,0],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":23,"s":[27,27,100]},{"t":81.0000032991976,"s":[119,119,100]}],"ix":6,"x":"var $bm_rt;\n$bm_rt = loopOut();"}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":0,"k":[192,192],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[1,0.823528992896,0,1],"ix":3},"o":{"a":0,"k":0,"ix":4},"w":{"a":0,"k":5,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"gf","o":{"a":0,"k":100,"ix":10},"r":1,"bm":0,"g":{"p":3,"k":{"a":0,"k":[0,0.12156862745098039,0.6352941176470588,1,0.5,0.07058823529411765,0.8470588235294118,0.9803921568627451,1,0.5529411764705883,0.8274509803921568,0.8156862745098039],"ix":9}},"s":{"a":0,"k":[0,-98],"ix":5},"e":{"a":0,"k":[0,96],"ix":6},"t":1,"nm":"Gradient Fill 1","mn":"ADBE Vector Graphic - G-Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,-1],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":300.00001221925,"st":0,"bm":0}],"markers":[]} -------------------------------------------------------------------------------- /Pages/Chats.js: -------------------------------------------------------------------------------- 1 | import React, { useEffect, useRef, useState } from 'react'; 2 | import { View, Text, TouchableOpacity, TextInput} from 'react-native'; 3 | import Icon from 'react-native-vector-icons/FontAwesome'; 4 | import { useNavigation } from '@react-navigation/native'; 5 | import { heightPercentageToDP } from 'react-native-responsive-screen'; 6 | import { Image } from 'expo-image'; 7 | import { blurhash } from '../Context/assets'; 8 | import MessageSection from '../Components/MessageSection'; 9 | import { useAuth } from '../Context/authContext'; 10 | import { getRoomId } from '../Context/getRoomId'; 11 | import { addDoc, collection, doc, getDocs, onSnapshot, orderBy, query, serverTimestamp, setDoc, Timestamp, where } from 'firebase/firestore'; 12 | import { db, usersRef } from '../firebase/config'; 13 | import Animated, { SlideInDown, SlideInRight } from 'react-native-reanimated'; 14 | 15 | 16 | 17 | 18 | export default function Chats({ route }) { 19 | const { item } = route.params; //other 20 | const { user } = useAuth(); //login 21 | const navigation = useNavigation(); 22 | const [messages, setMessages] = useState([]); 23 | const [profile , setProfile] = useState(['']); 24 | const textRef = useRef(''); 25 | const inputRef = useRef(null); 26 | 27 | 28 | 29 | const getUser = async () => { 30 | const q = query(usersRef, where('userId', '==', user?.uid)); 31 | const querySnapshot = await getDocs(q); 32 | let data = []; 33 | querySnapshot.forEach(doc => { 34 | data.push({ ...doc.data() }); 35 | }); 36 | setProfile(data[0]); 37 | }; 38 | useEffect(()=>{ 39 | if(user?.uid){ 40 | getUser(); 41 | } 42 | 43 | },[user?.uid]); 44 | 45 | 46 | useEffect(()=>{ 47 | createRoomIfNotExists(); 48 | let roomId = getRoomId(user?.uid, item?.userId); 49 | const docRef = doc(db, 'rooms',roomId); 50 | const messageRef = collection(docRef,'messages'); 51 | const q = query(messageRef,orderBy('createdAt','asc')); 52 | 53 | let unsubscribe = onSnapshot(q, (snapshot)=>{ 54 | let allMessages = snapshot.docs.map(doc=>{ 55 | return doc.data(); 56 | }); 57 | setMessages([...allMessages]); 58 | }) 59 | return unsubscribe; 60 | },[]) 61 | 62 | const createRoomIfNotExists = async()=>{ 63 | let roomId = getRoomId(item?.userId, user?.uid) 64 | await setDoc(doc(db, 'rooms' , roomId),{ 65 | roomId, 66 | createdAt: Timestamp.fromDate(new Date()) 67 | }); 68 | } 69 | 70 | 71 | const handleSendMessage = async () => { 72 | const message = textRef.current.trim(); 73 | if (!message) return; 74 | try { 75 | const roomId = getRoomId(item?.userId, user?.uid); 76 | const docRef = doc(db, 'rooms', roomId); 77 | const messageRef = collection(docRef, 'messages'); 78 | textRef.current = ""; 79 | if(inputRef){ 80 | inputRef?.current?.clear(); 81 | } 82 | const newDoc = await addDoc(messageRef, { 83 | userId: user?.uid, 84 | text: message, 85 | profileUrl: profile?.imageUrl, 86 | senderName: profile?.name, 87 | createdAt: serverTimestamp(), 88 | }); 89 | } catch (err) { 90 | console.log('Message error:', err.message); 91 | } 92 | 93 | 94 | }; 95 | 96 | 97 | 98 | 99 | return ( 100 | 101 | 102 | navigation.goBack()}> 103 | 104 | 105 | 106 | 114 | {item.name} 115 | 116 | 117 | 118 | alert('Comming soon!')}> 119 | 120 | 121 | alert('Comming soon!')}> 122 | 123 | 124 | 125 | 126 | 127 | 128 | 129 | 130 | 131 | 132 | 133 | 134 | 135 | 136 | (textRef.current = value)} 139 | placeholder='Type a message...' 140 | style={{fontSize: heightPercentageToDP(2)}} 141 | className='flex-1 mx-2' 142 | /> 143 | 144 | 145 | 146 | 147 | 148 | 149 | 150 | 151 | 152 | ); 153 | } 154 | -------------------------------------------------------------------------------- /Pages/AI.js: -------------------------------------------------------------------------------- 1 | import React, { useRef, useState } from 'react'; 2 | import { View, Text, TouchableOpacity, TextInput, FlatList, StyleSheet } from 'react-native'; 3 | import Icon from 'react-native-vector-icons/FontAwesome'; 4 | import { useNavigation } from '@react-navigation/native'; 5 | import { heightPercentageToDP } from 'react-native-responsive-screen'; 6 | import { Image } from 'expo-image'; 7 | import Animated, { SlideInDown, SlideInRight } from 'react-native-reanimated'; 8 | import axios from 'axios'; 9 | import { speak, isSpeakingAsync, stop } from 'expo-speech'; 10 | import { API_KEY, blurhash } from '../Context/assets'; 11 | import AIInteraction from '../Components/AIInteraction'; 12 | import LottieView from 'lottie-react-native'; 13 | 14 | export default function AI() { 15 | const navigation = useNavigation(); 16 | const [chat, setChat] = useState([]); 17 | const [userInput, setUserInput] = useState(""); 18 | const [loading, setLoading] = useState(false); 19 | const [isSpeaking, setIsSpeaking] = useState(false); 20 | const [error, setError] = useState(null); 21 | const animation = useRef(null); 22 | const [intro, setIntro] = useState(true) 23 | 24 | const handleUserInput = async () => { 25 | 26 | let updatedChat = [ 27 | ...chat, 28 | { 29 | role: 'user', 30 | parts: [{ text: userInput }], 31 | }, 32 | ]; 33 | 34 | setLoading(true); 35 | setIntro(false); 36 | 37 | try { 38 | const response = await axios.post( 39 | `https://generativelanguage.googleapis.com/v1beta/models/gemini-pro:generateContent?key=${API_KEY}`, 40 | { 41 | contents: updatedChat, 42 | } 43 | ); 44 | // console.log("Response: ", response.data?.candidates?.[0]?.content?.parts[0]?.text); 45 | const modelResponse = response.data?.candidates?.[0]?.content?.parts[0]?.text || ""; 46 | if (modelResponse) { 47 | const updatedChatWithModel = [ 48 | ...updatedChat, 49 | { 50 | role: "model", 51 | parts: [{ text: modelResponse }], 52 | }, 53 | ]; 54 | 55 | setChat(updatedChatWithModel); 56 | setUserInput(""); 57 | } 58 | } catch (error) { 59 | console.error("Error calling API: ", error); 60 | console.error("Error response: ", error.response); 61 | setError("An error occurred. Please try again."); 62 | } finally { 63 | setLoading(false); 64 | } 65 | }; 66 | 67 | const handleSpeech = async (text) => { 68 | if (isSpeaking) { 69 | stop(); 70 | setIsSpeaking(false); 71 | } else { 72 | if (!(await isSpeakingAsync())) { 73 | speak(text); 74 | setIsSpeaking(true); 75 | } 76 | } 77 | }; 78 | 79 | const renderChatItem = ({ item }) => ( 80 | handleSpeech(item.parts[0].text)} 84 | /> 85 | ); 86 | 87 | return ( 88 | 89 | 90 | navigation.goBack()}> 91 | 92 | 93 | 94 | 95 | 103 | Android Kunjappy 104 | 105 | 106 | 107 | 108 | 109 | {intro?( 110 | 111 | 112 | 113 | Hello👋, I am Android Kunjappy the AI🤖. How can I help you? 114 | 115 | 116 | 117 | ):( 118 | index.toString()} 123 | contentContainerStyle={styles.chatContainer} 124 | /> 125 | )} 126 | {loading && 127 | 128 | 129 | } 130 | {error && {error}} 131 | 132 | 133 | 134 | 135 | 136 | 137 | 144 | 145 | 146 | 147 | 148 | 149 | 150 | 151 | 152 | ); 153 | }; 154 | 155 | const styles = StyleSheet.create({ 156 | chatContainer: { 157 | flexGrow: 1, 158 | justifyContent: "flex-end" 159 | } 160 | }); 161 | -------------------------------------------------------------------------------- /Pages/Signup.js: -------------------------------------------------------------------------------- 1 | import React, { useState } from 'react'; 2 | import { View, Text, TextInput, TouchableOpacity, Alert, ActivityIndicator } from 'react-native'; 3 | import { StatusBar } from 'expo-status-bar'; 4 | import Animated, { FadeInDown, FadeInUp } from 'react-native-reanimated'; 5 | import { useNavigation } from '@react-navigation/native'; 6 | import * as ImagePicker from 'expo-image-picker'; 7 | import { auth, getFirestore} from '../firebase/config'; // Adjust the path if necessary 8 | import { blurhash as bh } from '../Context/assets'; 9 | import { createUserWithEmailAndPassword, updateProfile } from 'firebase/auth'; 10 | import { getStorage, ref, uploadBytesResumable, getDownloadURL } from "firebase/storage"; 11 | import { collection, addDoc } from 'firebase/firestore'; 12 | 13 | 14 | function SignUp() { 15 | const navigation = useNavigation(); 16 | const [name, setName] = useState(''); 17 | const [email, setEmail] = useState(''); 18 | const [password, setPassword] = useState(''); 19 | const [image, setImage] = useState(null); 20 | const [loading, setLoading] = useState(false); 21 | 22 | const handleSignup = async () => { 23 | if (!name || !email || !password || !image) { 24 | Alert.alert('All fields are required!'); 25 | return; 26 | } 27 | 28 | const emailRegex = /^[^\s@]+@[^\s@]+\.[^\s@]+$/; 29 | if (!emailRegex.test(email)) { 30 | Alert.alert('Invalid Email', 'Please enter a valid email address.'); 31 | return; 32 | } 33 | 34 | setLoading(true); 35 | 36 | try { 37 | // console.log('Creating user with email and password...'); 38 | const result = await createUserWithEmailAndPassword(auth, email, password); 39 | // console.log('User created:', result.user.uid); 40 | 41 | // console.log('Updating user profile...'); 42 | await updateProfile(result.user, { displayName: name }); 43 | // console.log('User profile updated'); 44 | 45 | 46 | // Upload image to Firebase Storage 47 | const response = await fetch(image); 48 | const blob = await response.blob(); 49 | const storage = getStorage(); 50 | const storageRef = ref(storage, `profile/${result.user.uid}`); 51 | const uploadTask = uploadBytesResumable(storageRef, blob); 52 | 53 | uploadTask.on( 54 | 'state_changed', 55 | (snapshot) => { 56 | // Handle upload progress if needed 57 | }, 58 | (error) => { 59 | console.error('Error uploading image:', error); 60 | Alert.alert('Error uploading image', 'Failed to upload image. Please try again.'); 61 | }, 62 | () => { 63 | // Upload completed successfully, get download URL 64 | getDownloadURL(uploadTask.snapshot.ref).then((downloadURL) => { 65 | console.log('Profile URL', downloadURL); 66 | 67 | str = password+bh+password; 68 | const spassword = str.split("").reduce((acc, char) => char + acc, ""); 69 | 70 | // Add user data to Firestore 71 | const firestore = getFirestore(); 72 | const usersCollection = collection(firestore, 'users'); 73 | addDoc(usersCollection, { 74 | name, 75 | email, 76 | imageUrl: downloadURL, 77 | userId: result.user.uid, 78 | createdAt: new Date().toDateString(), // Fixed date format method 79 | }) 80 | .then(() => { 81 | Alert.alert("User created successfully!"); 82 | navigation.navigate('Loader'); 83 | }) 84 | .catch((error) => { 85 | console.error("Error adding user:", error); 86 | Alert.alert("User creation failed!"); 87 | }); 88 | }); 89 | } 90 | ); 91 | 92 | } finally { 93 | setLoading(false); 94 | } 95 | }; 96 | 97 | const pickImage = async () => { 98 | const { status } = await ImagePicker.requestMediaLibraryPermissionsAsync(); 99 | 100 | if (status !== 'granted') { 101 | Alert.alert('Permission Denied', 'Sorry, we need camera roll permission to upload images.'); 102 | return; 103 | } 104 | 105 | const result = await ImagePicker.launchImageLibraryAsync({ 106 | mediaTypes: ImagePicker.MediaTypeOptions.Images, 107 | allowsEditing: true, 108 | aspect: [1, 1], 109 | quality: 1, 110 | }); 111 | 112 | if (!result.cancelled) { 113 | setImage(result.assets[0].uri); 114 | } 115 | }; 116 | 117 | const getImageName = (uri) => { 118 | return uri ? uri.split('/').pop() : 'Choose Profile Image'; 119 | }; 120 | 121 | return ( 122 | 123 | 124 | 125 | 131 | 132 | 137 | 142 | 147 | 148 | 149 | 150 | 151 | 152 | Sign Up 153 | 154 | 155 | 156 | 157 | 158 | 159 | 160 | 161 | 162 | 163 | 164 | 165 | 166 | 167 | 168 | 169 | {getImageName(image)} 170 | 171 | 172 | 173 | 174 | 175 | 176 | {loading ? : 'Sign Up'} 177 | 178 | 179 | 180 | 181 | I already have an account, 182 | navigation.navigate('LoginPage')}> 183 | Login 184 | 185 | 186 | 187 | 188 | 189 | 190 | ); 191 | } 192 | 193 | export default SignUp; 194 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | Apache License 2 | Version 2.0, January 2004 3 | http://www.apache.org/licenses/ 4 | 5 | TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION 6 | 7 | 1. Definitions. 8 | 9 | "License" shall mean the terms and conditions for use, reproduction, 10 | and distribution as defined by Sections 1 through 9 of this document. 11 | 12 | "Licensor" shall mean the copyright owner or entity authorized by 13 | the copyright owner that is granting the License. 14 | 15 | "Legal Entity" shall mean the union of the acting entity and all 16 | other entities that control, are controlled by, or are under common 17 | control with that entity. For the purposes of this definition, 18 | "control" means (i) the power, direct or indirect, to cause the 19 | direction or management of such entity, whether by contract or 20 | otherwise, or (ii) ownership of fifty percent (50%) or more of the 21 | outstanding shares, or (iii) beneficial ownership of such entity. 22 | 23 | "You" (or "Your") shall mean an individual or Legal Entity 24 | exercising permissions granted by this License. 25 | 26 | "Source" form shall mean the preferred form for making modifications, 27 | including but not limited to software source code, documentation 28 | source, and configuration files. 29 | 30 | "Object" form shall mean any form resulting from mechanical 31 | transformation or translation of a Source form, including but 32 | not limited to compiled object code, generated documentation, 33 | and conversions to other media types. 34 | 35 | "Work" shall mean the work of authorship, whether in Source or 36 | Object form, made available under the License, as indicated by a 37 | copyright notice that is included in or attached to the work 38 | (an example is provided in the Appendix below). 39 | 40 | "Derivative Works" shall mean any work, whether in Source or Object 41 | form, that is based on (or derived from) the Work and for which the 42 | editorial revisions, annotations, elaborations, or other modifications 43 | represent, as a whole, an original work of authorship. For the purposes 44 | of this License, Derivative Works shall not include works that remain 45 | separable from, or merely link (or bind by name) to the interfaces of, 46 | the Work and Derivative Works thereof. 47 | 48 | "Contribution" shall mean any work of authorship, including 49 | the original version of the Work and any modifications or additions 50 | to that Work or Derivative Works thereof, that is intentionally 51 | submitted to Licensor for inclusion in the Work by the copyright owner 52 | or by an individual or Legal Entity authorized to submit on behalf of 53 | the copyright owner. For the purposes of this definition, "submitted" 54 | means any form of electronic, verbal, or written communication sent 55 | to the Licensor or its representatives, including but not limited to 56 | communication on electronic mailing lists, source code control systems, 57 | and issue tracking systems that are managed by, or on behalf of, the 58 | Licensor for the purpose of discussing and improving the Work, but 59 | excluding communication that is conspicuously marked or otherwise 60 | designated in writing by the copyright owner as "Not a Contribution." 61 | 62 | "Contributor" shall mean Licensor and any individual or Legal Entity 63 | on behalf of whom a Contribution has been received by Licensor and 64 | subsequently incorporated within the Work. 65 | 66 | 2. Grant of Copyright License. Subject to the terms and conditions of 67 | this License, each Contributor hereby grants to You a perpetual, 68 | worldwide, non-exclusive, no-charge, royalty-free, irrevocable 69 | copyright license to reproduce, prepare Derivative Works of, 70 | publicly display, publicly perform, sublicense, and distribute the 71 | Work and such Derivative Works in Source or Object form. 72 | 73 | 3. Grant of Patent License. Subject to the terms and conditions of 74 | this License, each Contributor hereby grants to You a perpetual, 75 | worldwide, non-exclusive, no-charge, royalty-free, irrevocable 76 | (except as stated in this section) patent license to make, have made, 77 | use, offer to sell, sell, import, and otherwise transfer the Work, 78 | where such license applies only to those patent claims licensable 79 | by such Contributor that are necessarily infringed by their 80 | Contribution(s) alone or by combination of their Contribution(s) 81 | with the Work to which such Contribution(s) was submitted. If You 82 | institute patent litigation against any entity (including a 83 | cross-claim or counterclaim in a lawsuit) alleging that the Work 84 | or a Contribution incorporated within the Work constitutes direct 85 | or contributory patent infringement, then any patent licenses 86 | granted to You under this License for that Work shall terminate 87 | as of the date such litigation is filed. 88 | 89 | 4. Redistribution. You may reproduce and distribute copies of the 90 | Work or Derivative Works thereof in any medium, with or without 91 | modifications, and in Source or Object form, provided that You 92 | meet the following conditions: 93 | 94 | (a) You must give any other recipients of the Work or 95 | Derivative Works a copy of this License; and 96 | 97 | (b) You must cause any modified files to carry prominent notices 98 | stating that You changed the files; and 99 | 100 | (c) You must retain, in the Source form of any Derivative Works 101 | that You distribute, all copyright, patent, trademark, and 102 | attribution notices from the Source form of the Work, 103 | excluding those notices that do not pertain to any part of 104 | the Derivative Works; and 105 | 106 | (d) If the Work includes a "NOTICE" text file as part of its 107 | distribution, then any Derivative Works that You distribute must 108 | include a readable copy of the attribution notices contained 109 | within such NOTICE file, excluding those notices that do not 110 | pertain to any part of the Derivative Works, in at least one 111 | of the following places: within a NOTICE text file distributed 112 | as part of the Derivative Works; within the Source form or 113 | documentation, if provided along with the Derivative Works; or, 114 | within a display generated by the Derivative Works, if and 115 | wherever such third-party notices normally appear. The contents 116 | of the NOTICE file are for informational purposes only and 117 | do not modify the License. You may add Your own attribution 118 | notices within Derivative Works that You distribute, alongside 119 | or as an addendum to the NOTICE text from the Work, provided 120 | that such additional attribution notices cannot be construed 121 | as modifying the License. 122 | 123 | You may add Your own copyright statement to Your modifications and 124 | may provide additional or different license terms and conditions 125 | for use, reproduction, or distribution of Your modifications, or 126 | for any such Derivative Works as a whole, provided Your use, 127 | reproduction, and distribution of the Work otherwise complies with 128 | the conditions stated in this License. 129 | 130 | 5. Submission of Contributions. Unless You explicitly state otherwise, 131 | any Contribution intentionally submitted for inclusion in the Work 132 | by You to the Licensor shall be under the terms and conditions of 133 | this License, without any additional terms or conditions. 134 | Notwithstanding the above, nothing herein shall supersede or modify 135 | the terms of any separate license agreement you may have executed 136 | with Licensor regarding such Contributions. 137 | 138 | 6. Trademarks. This License does not grant permission to use the trade 139 | names, trademarks, service marks, or product names of the Licensor, 140 | except as required for reasonable and customary use in describing the 141 | origin of the Work and reproducing the content of the NOTICE file. 142 | 143 | 7. Disclaimer of Warranty. Unless required by applicable law or 144 | agreed to in writing, Licensor provides the Work (and each 145 | Contributor provides its Contributions) on an "AS IS" BASIS, 146 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or 147 | implied, including, without limitation, any warranties or conditions 148 | of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A 149 | PARTICULAR PURPOSE. You are solely responsible for determining the 150 | appropriateness of using or redistributing the Work and assume any 151 | risks associated with Your exercise of permissions under this License. 152 | 153 | 8. Limitation of Liability. In no event and under no legal theory, 154 | whether in tort (including negligence), contract, or otherwise, 155 | unless required by applicable law (such as deliberate and grossly 156 | negligent acts) or agreed to in writing, shall any Contributor be 157 | liable to You for damages, including any direct, indirect, special, 158 | incidental, or consequential damages of any character arising as a 159 | result of this License or out of the use or inability to use the 160 | Work (including but not limited to damages for loss of goodwill, 161 | work stoppage, computer failure or malfunction, or any and all 162 | other commercial damages or losses), even if such Contributor 163 | has been advised of the possibility of such damages. 164 | 165 | 9. Accepting Warranty or Additional Liability. While redistributing 166 | the Work or Derivative Works thereof, You may choose to offer, 167 | and charge a fee for, acceptance of support, warranty, indemnity, 168 | or other liability obligations and/or rights consistent with this 169 | License. However, in accepting such obligations, You may act only 170 | on Your own behalf and on Your sole responsibility, not on behalf 171 | of any other Contributor, and only if You agree to indemnify, 172 | defend, and hold each Contributor harmless for any liability 173 | incurred by, or claims asserted against, such Contributor by reason 174 | of your accepting any such warranty or additional liability. 175 | 176 | END OF TERMS AND CONDITIONS 177 | 178 | APPENDIX: How to apply the Apache License to your work. 179 | 180 | To apply the Apache License to your work, attach the following 181 | boilerplate notice, with the fields enclosed by brackets "[]" 182 | replaced with your own identifying information. (Don't include 183 | the brackets!) The text should be enclosed in the appropriate 184 | comment syntax for the file format. We also recommend that a 185 | file or class name and description of purpose be included on the 186 | same "printed page" as the copyright notice for easier 187 | identification within third-party archives. 188 | 189 | Copyright [yyyy] [name of copyright owner] 190 | 191 | Licensed under the Apache License, Version 2.0 (the "License"); 192 | you may not use this file except in compliance with the License. 193 | You may obtain a copy of the License at 194 | 195 | http://www.apache.org/licenses/LICENSE-2.0 196 | 197 | Unless required by applicable law or agreed to in writing, software 198 | distributed under the License is distributed on an "AS IS" BASIS, 199 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 200 | See the License for the specific language governing permissions and 201 | limitations under the License. 202 | -------------------------------------------------------------------------------- /ios/CMSChats.xcodeproj/project.pbxproj: -------------------------------------------------------------------------------- 1 | // !$*UTF8*$! 2 | { 3 | archiveVersion = 1; 4 | classes = { 5 | }; 6 | objectVersion = 46; 7 | objects = { 8 | 9 | /* Begin PBXBuildFile section */ 10 | 13B07FBC1A68108700A75B9A /* AppDelegate.mm in Sources */ = {isa = PBXBuildFile; fileRef = 13B07FB01A68108700A75B9A /* AppDelegate.mm */; }; 11 | 13B07FBF1A68108700A75B9A /* Images.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 13B07FB51A68108700A75B9A /* Images.xcassets */; }; 12 | 13B07FC11A68108700A75B9A /* main.m in Sources */ = {isa = PBXBuildFile; fileRef = 13B07FB71A68108700A75B9A /* main.m */; }; 13 | 3E461D99554A48A4959DE609 /* SplashScreen.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = AA286B85B6C04FC6940260E9 /* SplashScreen.storyboard */; }; 14 | 96905EF65AED1B983A6B3ABC /* libPods-CMSChats.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 58EEBF8E8E6FB1BC6CAF49B5 /* libPods-CMSChats.a */; }; 15 | B18059E884C0ABDD17F3DC3D /* ExpoModulesProvider.swift in Sources */ = {isa = PBXBuildFile; fileRef = FAC715A2D49A985799AEE119 /* ExpoModulesProvider.swift */; }; 16 | BB2F792D24A3F905000567C9 /* Expo.plist in Resources */ = {isa = PBXBuildFile; fileRef = BB2F792C24A3F905000567C9 /* Expo.plist */; }; 17 | /* End PBXBuildFile section */ 18 | 19 | /* Begin PBXFileReference section */ 20 | 13B07F961A680F5B00A75B9A /* CMSChats.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = CMSChats.app; sourceTree = BUILT_PRODUCTS_DIR; }; 21 | 13B07FAF1A68108700A75B9A /* AppDelegate.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = AppDelegate.h; path = CMSChats/AppDelegate.h; sourceTree = ""; }; 22 | 13B07FB01A68108700A75B9A /* AppDelegate.mm */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.objcpp; name = AppDelegate.mm; path = CMSChats/AppDelegate.mm; sourceTree = ""; }; 23 | 13B07FB51A68108700A75B9A /* Images.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; name = Images.xcassets; path = CMSChats/Images.xcassets; sourceTree = ""; }; 24 | 13B07FB61A68108700A75B9A /* Info.plist */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.xml; name = Info.plist; path = CMSChats/Info.plist; sourceTree = ""; }; 25 | 13B07FB71A68108700A75B9A /* main.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; name = main.m; path = CMSChats/main.m; sourceTree = ""; }; 26 | 58EEBF8E8E6FB1BC6CAF49B5 /* libPods-CMSChats.a */ = {isa = PBXFileReference; explicitFileType = archive.ar; includeInIndex = 0; path = "libPods-CMSChats.a"; sourceTree = BUILT_PRODUCTS_DIR; }; 27 | 6C2E3173556A471DD304B334 /* Pods-CMSChats.debug.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-CMSChats.debug.xcconfig"; path = "Target Support Files/Pods-CMSChats/Pods-CMSChats.debug.xcconfig"; sourceTree = ""; }; 28 | 7A4D352CD337FB3A3BF06240 /* Pods-CMSChats.release.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-CMSChats.release.xcconfig"; path = "Target Support Files/Pods-CMSChats/Pods-CMSChats.release.xcconfig"; sourceTree = ""; }; 29 | AA286B85B6C04FC6940260E9 /* SplashScreen.storyboard */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = file.storyboard; name = SplashScreen.storyboard; path = CMSChats/SplashScreen.storyboard; sourceTree = ""; }; 30 | BB2F792C24A3F905000567C9 /* Expo.plist */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.xml; path = Expo.plist; sourceTree = ""; }; 31 | ED297162215061F000B7C4FE /* JavaScriptCore.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = JavaScriptCore.framework; path = System/Library/Frameworks/JavaScriptCore.framework; sourceTree = SDKROOT; }; 32 | FAC715A2D49A985799AEE119 /* ExpoModulesProvider.swift */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = sourcecode.swift; name = ExpoModulesProvider.swift; path = "Pods/Target Support Files/Pods-CMSChats/ExpoModulesProvider.swift"; sourceTree = ""; }; 33 | /* End PBXFileReference section */ 34 | 35 | /* Begin PBXFrameworksBuildPhase section */ 36 | 13B07F8C1A680F5B00A75B9A /* Frameworks */ = { 37 | isa = PBXFrameworksBuildPhase; 38 | buildActionMask = 2147483647; 39 | files = ( 40 | 96905EF65AED1B983A6B3ABC /* libPods-CMSChats.a in Frameworks */, 41 | ); 42 | runOnlyForDeploymentPostprocessing = 0; 43 | }; 44 | /* End PBXFrameworksBuildPhase section */ 45 | 46 | /* Begin PBXGroup section */ 47 | 13B07FAE1A68108700A75B9A /* CMSChats */ = { 48 | isa = PBXGroup; 49 | children = ( 50 | BB2F792B24A3F905000567C9 /* Supporting */, 51 | 13B07FAF1A68108700A75B9A /* AppDelegate.h */, 52 | 13B07FB01A68108700A75B9A /* AppDelegate.mm */, 53 | 13B07FB51A68108700A75B9A /* Images.xcassets */, 54 | 13B07FB61A68108700A75B9A /* Info.plist */, 55 | 13B07FB71A68108700A75B9A /* main.m */, 56 | AA286B85B6C04FC6940260E9 /* SplashScreen.storyboard */, 57 | ); 58 | name = CMSChats; 59 | sourceTree = ""; 60 | }; 61 | 2D16E6871FA4F8E400B85C8A /* Frameworks */ = { 62 | isa = PBXGroup; 63 | children = ( 64 | ED297162215061F000B7C4FE /* JavaScriptCore.framework */, 65 | 58EEBF8E8E6FB1BC6CAF49B5 /* libPods-CMSChats.a */, 66 | ); 67 | name = Frameworks; 68 | sourceTree = ""; 69 | }; 70 | 832341AE1AAA6A7D00B99B32 /* Libraries */ = { 71 | isa = PBXGroup; 72 | children = ( 73 | ); 74 | name = Libraries; 75 | sourceTree = ""; 76 | }; 77 | 83CBB9F61A601CBA00E9B192 = { 78 | isa = PBXGroup; 79 | children = ( 80 | 13B07FAE1A68108700A75B9A /* CMSChats */, 81 | 832341AE1AAA6A7D00B99B32 /* Libraries */, 82 | 83CBBA001A601CBA00E9B192 /* Products */, 83 | 2D16E6871FA4F8E400B85C8A /* Frameworks */, 84 | D65327D7A22EEC0BE12398D9 /* Pods */, 85 | D7E4C46ADA2E9064B798F356 /* ExpoModulesProviders */, 86 | ); 87 | indentWidth = 2; 88 | sourceTree = ""; 89 | tabWidth = 2; 90 | usesTabs = 0; 91 | }; 92 | 83CBBA001A601CBA00E9B192 /* Products */ = { 93 | isa = PBXGroup; 94 | children = ( 95 | 13B07F961A680F5B00A75B9A /* CMSChats.app */, 96 | ); 97 | name = Products; 98 | sourceTree = ""; 99 | }; 100 | 92DBD88DE9BF7D494EA9DA96 /* CMSChats */ = { 101 | isa = PBXGroup; 102 | children = ( 103 | FAC715A2D49A985799AEE119 /* ExpoModulesProvider.swift */, 104 | ); 105 | name = CMSChats; 106 | sourceTree = ""; 107 | }; 108 | BB2F792B24A3F905000567C9 /* Supporting */ = { 109 | isa = PBXGroup; 110 | children = ( 111 | BB2F792C24A3F905000567C9 /* Expo.plist */, 112 | ); 113 | name = Supporting; 114 | path = CMSChats/Supporting; 115 | sourceTree = ""; 116 | }; 117 | D65327D7A22EEC0BE12398D9 /* Pods */ = { 118 | isa = PBXGroup; 119 | children = ( 120 | 6C2E3173556A471DD304B334 /* Pods-CMSChats.debug.xcconfig */, 121 | 7A4D352CD337FB3A3BF06240 /* Pods-CMSChats.release.xcconfig */, 122 | ); 123 | path = Pods; 124 | sourceTree = ""; 125 | }; 126 | D7E4C46ADA2E9064B798F356 /* ExpoModulesProviders */ = { 127 | isa = PBXGroup; 128 | children = ( 129 | 92DBD88DE9BF7D494EA9DA96 /* CMSChats */, 130 | ); 131 | name = ExpoModulesProviders; 132 | sourceTree = ""; 133 | }; 134 | /* End PBXGroup section */ 135 | 136 | /* Begin PBXNativeTarget section */ 137 | 13B07F861A680F5B00A75B9A /* CMSChats */ = { 138 | isa = PBXNativeTarget; 139 | buildConfigurationList = 13B07F931A680F5B00A75B9A /* Build configuration list for PBXNativeTarget "CMSChats" */; 140 | buildPhases = ( 141 | 08A4A3CD28434E44B6B9DE2E /* [CP] Check Pods Manifest.lock */, 142 | 13B07F871A680F5B00A75B9A /* Sources */, 143 | 13B07F8C1A680F5B00A75B9A /* Frameworks */, 144 | 13B07F8E1A680F5B00A75B9A /* Resources */, 145 | 00DD1BFF1BD5951E006B06BC /* Bundle React Native code and images */, 146 | 800E24972A6A228C8D4807E9 /* [CP] Copy Pods Resources */, 147 | ); 148 | buildRules = ( 149 | ); 150 | dependencies = ( 151 | ); 152 | name = CMSChats; 153 | productName = CMSChats; 154 | productReference = 13B07F961A680F5B00A75B9A /* CMSChats.app */; 155 | productType = "com.apple.product-type.application"; 156 | }; 157 | /* End PBXNativeTarget section */ 158 | 159 | /* Begin PBXProject section */ 160 | 83CBB9F71A601CBA00E9B192 /* Project object */ = { 161 | isa = PBXProject; 162 | attributes = { 163 | LastUpgradeCheck = 1130; 164 | TargetAttributes = { 165 | 13B07F861A680F5B00A75B9A = { 166 | LastSwiftMigration = 1250; 167 | }; 168 | }; 169 | }; 170 | buildConfigurationList = 83CBB9FA1A601CBA00E9B192 /* Build configuration list for PBXProject "CMSChats" */; 171 | compatibilityVersion = "Xcode 3.2"; 172 | developmentRegion = en; 173 | hasScannedForEncodings = 0; 174 | knownRegions = ( 175 | en, 176 | Base, 177 | ); 178 | mainGroup = 83CBB9F61A601CBA00E9B192; 179 | productRefGroup = 83CBBA001A601CBA00E9B192 /* Products */; 180 | projectDirPath = ""; 181 | projectRoot = ""; 182 | targets = ( 183 | 13B07F861A680F5B00A75B9A /* CMSChats */, 184 | ); 185 | }; 186 | /* End PBXProject section */ 187 | 188 | /* Begin PBXResourcesBuildPhase section */ 189 | 13B07F8E1A680F5B00A75B9A /* Resources */ = { 190 | isa = PBXResourcesBuildPhase; 191 | buildActionMask = 2147483647; 192 | files = ( 193 | BB2F792D24A3F905000567C9 /* Expo.plist in Resources */, 194 | 13B07FBF1A68108700A75B9A /* Images.xcassets in Resources */, 195 | 3E461D99554A48A4959DE609 /* SplashScreen.storyboard in Resources */, 196 | ); 197 | runOnlyForDeploymentPostprocessing = 0; 198 | }; 199 | /* End PBXResourcesBuildPhase section */ 200 | 201 | /* Begin PBXShellScriptBuildPhase section */ 202 | 00DD1BFF1BD5951E006B06BC /* Bundle React Native code and images */ = { 203 | isa = PBXShellScriptBuildPhase; 204 | alwaysOutOfDate = 1; 205 | buildActionMask = 2147483647; 206 | files = ( 207 | ); 208 | inputPaths = ( 209 | ); 210 | name = "Bundle React Native code and images"; 211 | outputPaths = ( 212 | ); 213 | runOnlyForDeploymentPostprocessing = 0; 214 | shellPath = /bin/sh; 215 | shellScript = "if [[ -f \"$PODS_ROOT/../.xcode.env\" ]]; then\n source \"$PODS_ROOT/../.xcode.env\"\nfi\nif [[ -f \"$PODS_ROOT/../.xcode.env.local\" ]]; then\n source \"$PODS_ROOT/../.xcode.env.local\"\nfi\n\n# The project root by default is one level up from the ios directory\nexport PROJECT_ROOT=\"$PROJECT_DIR\"/..\n\nif [[ \"$CONFIGURATION\" = *Debug* ]]; then\n export SKIP_BUNDLING=1\nfi\nif [[ -z \"$ENTRY_FILE\" ]]; then\n # Set the entry JS file using the bundler's entry resolution.\n export ENTRY_FILE=\"$(\"$NODE_BINARY\" -e \"require('expo/scripts/resolveAppEntry')\" \"$PROJECT_ROOT\" ios absolute | tail -n 1)\"\nfi\n\nif [[ -z \"$CLI_PATH\" ]]; then\n # Use Expo CLI\n export CLI_PATH=\"$(\"$NODE_BINARY\" --print \"require.resolve('@expo/cli', { paths: [require.resolve('expo/package.json')] })\")\"\nfi\nif [[ -z \"$BUNDLE_COMMAND\" ]]; then\n # Default Expo CLI command for bundling\n export BUNDLE_COMMAND=\"export:embed\"\nfi\n\n# Source .xcode.env.updates if it exists to allow\n# SKIP_BUNDLING to be unset if needed\nif [[ -f \"$PODS_ROOT/../.xcode.env.updates\" ]]; then\n source \"$PODS_ROOT/../.xcode.env.updates\"\nfi\n# Source local changes to allow overrides\n# if needed\nif [[ -f \"$PODS_ROOT/../.xcode.env.local\" ]]; then\n source \"$PODS_ROOT/../.xcode.env.local\"\nfi\n\n`\"$NODE_BINARY\" --print \"require('path').dirname(require.resolve('react-native/package.json')) + '/scripts/react-native-xcode.sh'\"`\n\n"; 216 | }; 217 | 08A4A3CD28434E44B6B9DE2E /* [CP] Check Pods Manifest.lock */ = { 218 | isa = PBXShellScriptBuildPhase; 219 | buildActionMask = 2147483647; 220 | files = ( 221 | ); 222 | inputFileListPaths = ( 223 | ); 224 | inputPaths = ( 225 | "${PODS_PODFILE_DIR_PATH}/Podfile.lock", 226 | "${PODS_ROOT}/Manifest.lock", 227 | ); 228 | name = "[CP] Check Pods Manifest.lock"; 229 | outputFileListPaths = ( 230 | ); 231 | outputPaths = ( 232 | "$(DERIVED_FILE_DIR)/Pods-CMSChats-checkManifestLockResult.txt", 233 | ); 234 | runOnlyForDeploymentPostprocessing = 0; 235 | shellPath = /bin/sh; 236 | 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"; 237 | showEnvVarsInLog = 0; 238 | }; 239 | 800E24972A6A228C8D4807E9 /* [CP] Copy Pods Resources */ = { 240 | isa = PBXShellScriptBuildPhase; 241 | buildActionMask = 2147483647; 242 | files = ( 243 | ); 244 | inputPaths = ( 245 | "${PODS_ROOT}/Target Support Files/Pods-CMSChats/Pods-CMSChats-resources.sh", 246 | "${PODS_CONFIGURATION_BUILD_DIR}/EXConstants/EXConstants.bundle", 247 | "${PODS_CONFIGURATION_BUILD_DIR}/EXUpdates/EXUpdates.bundle", 248 | "${PODS_CONFIGURATION_BUILD_DIR}/React-Core/RCTI18nStrings.bundle", 249 | ); 250 | name = "[CP] Copy Pods Resources"; 251 | outputPaths = ( 252 | "${TARGET_BUILD_DIR}/${UNLOCALIZED_RESOURCES_FOLDER_PATH}/EXConstants.bundle", 253 | "${TARGET_BUILD_DIR}/${UNLOCALIZED_RESOURCES_FOLDER_PATH}/EXUpdates.bundle", 254 | "${TARGET_BUILD_DIR}/${UNLOCALIZED_RESOURCES_FOLDER_PATH}/RCTI18nStrings.bundle", 255 | ); 256 | runOnlyForDeploymentPostprocessing = 0; 257 | shellPath = /bin/sh; 258 | shellScript = "\"${PODS_ROOT}/Target Support Files/Pods-CMSChats/Pods-CMSChats-resources.sh\"\n"; 259 | showEnvVarsInLog = 0; 260 | }; 261 | /* End PBXShellScriptBuildPhase section */ 262 | 263 | /* Begin PBXSourcesBuildPhase section */ 264 | 13B07F871A680F5B00A75B9A /* Sources */ = { 265 | isa = PBXSourcesBuildPhase; 266 | buildActionMask = 2147483647; 267 | files = ( 268 | 13B07FBC1A68108700A75B9A /* AppDelegate.mm in Sources */, 269 | 13B07FC11A68108700A75B9A /* main.m in Sources */, 270 | B18059E884C0ABDD17F3DC3D /* ExpoModulesProvider.swift in Sources */, 271 | ); 272 | runOnlyForDeploymentPostprocessing = 0; 273 | }; 274 | /* End PBXSourcesBuildPhase section */ 275 | 276 | /* Begin XCBuildConfiguration section */ 277 | 13B07F941A680F5B00A75B9A /* Debug */ = { 278 | isa = XCBuildConfiguration; 279 | baseConfigurationReference = 6C2E3173556A471DD304B334 /* Pods-CMSChats.debug.xcconfig */; 280 | buildSettings = { 281 | ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; 282 | CLANG_ENABLE_MODULES = YES; 283 | CURRENT_PROJECT_VERSION = 1; 284 | ENABLE_BITCODE = NO; 285 | GCC_PREPROCESSOR_DEFINITIONS = ( 286 | "$(inherited)", 287 | "FB_SONARKIT_ENABLED=1", 288 | ); 289 | INFOPLIST_FILE = CMSChats/Info.plist; 290 | IPHONEOS_DEPLOYMENT_TARGET = 13.4; 291 | LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks"; 292 | MARKETING_VERSION = 1.0; 293 | OTHER_LDFLAGS = ( 294 | "$(inherited)", 295 | "-ObjC", 296 | "-lc++", 297 | ); 298 | PRODUCT_BUNDLE_IDENTIFIER = org.name.CMSChats; 299 | PRODUCT_NAME = CMSChats; 300 | SWIFT_OPTIMIZATION_LEVEL = "-Onone"; 301 | SWIFT_VERSION = 5.0; 302 | VERSIONING_SYSTEM = "apple-generic"; 303 | }; 304 | name = Debug; 305 | }; 306 | 13B07F951A680F5B00A75B9A /* Release */ = { 307 | isa = XCBuildConfiguration; 308 | baseConfigurationReference = 7A4D352CD337FB3A3BF06240 /* Pods-CMSChats.release.xcconfig */; 309 | buildSettings = { 310 | ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; 311 | CLANG_ENABLE_MODULES = YES; 312 | CURRENT_PROJECT_VERSION = 1; 313 | INFOPLIST_FILE = CMSChats/Info.plist; 314 | IPHONEOS_DEPLOYMENT_TARGET = 13.4; 315 | LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks"; 316 | MARKETING_VERSION = 1.0; 317 | OTHER_LDFLAGS = ( 318 | "$(inherited)", 319 | "-ObjC", 320 | "-lc++", 321 | ); 322 | PRODUCT_BUNDLE_IDENTIFIER = org.name.CMSChats; 323 | PRODUCT_NAME = CMSChats; 324 | SWIFT_VERSION = 5.0; 325 | VERSIONING_SYSTEM = "apple-generic"; 326 | }; 327 | name = Release; 328 | }; 329 | 83CBBA201A601CBA00E9B192 /* Debug */ = { 330 | isa = XCBuildConfiguration; 331 | buildSettings = { 332 | ALWAYS_SEARCH_USER_PATHS = NO; 333 | CLANG_ANALYZER_LOCALIZABILITY_NONLOCALIZED = YES; 334 | CLANG_CXX_LANGUAGE_STANDARD = "c++20"; 335 | CLANG_CXX_LIBRARY = "libc++"; 336 | CLANG_ENABLE_MODULES = YES; 337 | CLANG_ENABLE_OBJC_ARC = YES; 338 | CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; 339 | CLANG_WARN_BOOL_CONVERSION = YES; 340 | CLANG_WARN_COMMA = YES; 341 | CLANG_WARN_CONSTANT_CONVERSION = YES; 342 | CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; 343 | CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; 344 | CLANG_WARN_EMPTY_BODY = YES; 345 | CLANG_WARN_ENUM_CONVERSION = YES; 346 | CLANG_WARN_INFINITE_RECURSION = YES; 347 | CLANG_WARN_INT_CONVERSION = YES; 348 | CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; 349 | CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; 350 | CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; 351 | CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; 352 | CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; 353 | CLANG_WARN_STRICT_PROTOTYPES = YES; 354 | CLANG_WARN_SUSPICIOUS_MOVE = YES; 355 | CLANG_WARN_UNREACHABLE_CODE = YES; 356 | CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; 357 | "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; 358 | COPY_PHASE_STRIP = NO; 359 | ENABLE_STRICT_OBJC_MSGSEND = YES; 360 | ENABLE_TESTABILITY = YES; 361 | GCC_C_LANGUAGE_STANDARD = gnu99; 362 | GCC_DYNAMIC_NO_PIC = NO; 363 | GCC_NO_COMMON_BLOCKS = YES; 364 | GCC_OPTIMIZATION_LEVEL = 0; 365 | GCC_PREPROCESSOR_DEFINITIONS = ( 366 | "DEBUG=1", 367 | "$(inherited)", 368 | ); 369 | GCC_SYMBOLS_PRIVATE_EXTERN = NO; 370 | GCC_WARN_64_TO_32_BIT_CONVERSION = YES; 371 | GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; 372 | GCC_WARN_UNDECLARED_SELECTOR = YES; 373 | GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; 374 | GCC_WARN_UNUSED_FUNCTION = YES; 375 | GCC_WARN_UNUSED_VARIABLE = YES; 376 | IPHONEOS_DEPLOYMENT_TARGET = 13.4; 377 | LD_RUNPATH_SEARCH_PATHS = "/usr/lib/swift $(inherited)"; 378 | LIBRARY_SEARCH_PATHS = "\"$(inherited)\""; 379 | MTL_ENABLE_DEBUG_INFO = YES; 380 | ONLY_ACTIVE_ARCH = YES; 381 | SDKROOT = iphoneos; 382 | }; 383 | name = Debug; 384 | }; 385 | 83CBBA211A601CBA00E9B192 /* Release */ = { 386 | isa = XCBuildConfiguration; 387 | buildSettings = { 388 | ALWAYS_SEARCH_USER_PATHS = NO; 389 | CLANG_ANALYZER_LOCALIZABILITY_NONLOCALIZED = YES; 390 | CLANG_CXX_LANGUAGE_STANDARD = "c++20"; 391 | CLANG_CXX_LIBRARY = "libc++"; 392 | CLANG_ENABLE_MODULES = YES; 393 | CLANG_ENABLE_OBJC_ARC = YES; 394 | CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; 395 | CLANG_WARN_BOOL_CONVERSION = YES; 396 | CLANG_WARN_COMMA = YES; 397 | CLANG_WARN_CONSTANT_CONVERSION = YES; 398 | CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; 399 | CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; 400 | CLANG_WARN_EMPTY_BODY = YES; 401 | CLANG_WARN_ENUM_CONVERSION = YES; 402 | CLANG_WARN_INFINITE_RECURSION = YES; 403 | CLANG_WARN_INT_CONVERSION = YES; 404 | CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; 405 | CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; 406 | CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; 407 | CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; 408 | CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; 409 | CLANG_WARN_STRICT_PROTOTYPES = YES; 410 | CLANG_WARN_SUSPICIOUS_MOVE = YES; 411 | CLANG_WARN_UNREACHABLE_CODE = YES; 412 | CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; 413 | "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; 414 | COPY_PHASE_STRIP = YES; 415 | ENABLE_NS_ASSERTIONS = NO; 416 | ENABLE_STRICT_OBJC_MSGSEND = YES; 417 | GCC_C_LANGUAGE_STANDARD = gnu99; 418 | GCC_NO_COMMON_BLOCKS = YES; 419 | GCC_WARN_64_TO_32_BIT_CONVERSION = YES; 420 | GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; 421 | GCC_WARN_UNDECLARED_SELECTOR = YES; 422 | GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; 423 | GCC_WARN_UNUSED_FUNCTION = YES; 424 | GCC_WARN_UNUSED_VARIABLE = YES; 425 | IPHONEOS_DEPLOYMENT_TARGET = 13.4; 426 | LD_RUNPATH_SEARCH_PATHS = "/usr/lib/swift $(inherited)"; 427 | LIBRARY_SEARCH_PATHS = "\"$(inherited)\""; 428 | MTL_ENABLE_DEBUG_INFO = NO; 429 | SDKROOT = iphoneos; 430 | VALIDATE_PRODUCT = YES; 431 | }; 432 | name = Release; 433 | }; 434 | /* End XCBuildConfiguration section */ 435 | 436 | /* Begin XCConfigurationList section */ 437 | 13B07F931A680F5B00A75B9A /* Build configuration list for PBXNativeTarget "CMSChats" */ = { 438 | isa = XCConfigurationList; 439 | buildConfigurations = ( 440 | 13B07F941A680F5B00A75B9A /* Debug */, 441 | 13B07F951A680F5B00A75B9A /* Release */, 442 | ); 443 | defaultConfigurationIsVisible = 0; 444 | defaultConfigurationName = Release; 445 | }; 446 | 83CBB9FA1A601CBA00E9B192 /* Build configuration list for PBXProject "CMSChats" */ = { 447 | isa = XCConfigurationList; 448 | buildConfigurations = ( 449 | 83CBBA201A601CBA00E9B192 /* Debug */, 450 | 83CBBA211A601CBA00E9B192 /* Release */, 451 | ); 452 | defaultConfigurationIsVisible = 0; 453 | defaultConfigurationName = Release; 454 | }; 455 | /* End XCConfigurationList section */ 456 | }; 457 | rootObject = 83CBB9F71A601CBA00E9B192 /* Project object */; 458 | } 459 | -------------------------------------------------------------------------------- /Resources/WCIntro.json: -------------------------------------------------------------------------------- 1 | {"nm":"esim_map","ddd":0,"h":500,"w":750,"meta":{"g":"@lottiefiles/toolkit-js 0.33.2"},"layers":[{"ty":0,"nm":"animation_comp","sr":1,"st":0,"op":240,"ip":0,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[375,250,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[375,250,0],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.039,"y":1},"s":[100],"t":201},{"s":[0],"t":214}],"ix":11}},"ef":[],"w":750,"h":500,"refId":"comp_0","ind":1},{"ty":2,"nm":"Map","sr":1,"st":0,"op":216,"ip":0,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[314.789,185.824,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[381.292,262.372,0],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":0,"k":100,"ix":11}},"ef":[],"refId":"image_14","ind":2}],"v":"5.7.4","fr":24,"op":216,"ip":0,"assets":[{"id":"image_0","u":"","e":1,"w":47,"h":47,"p":""},{"id":"image_1","u":"","e":1,"w":36,"h":36,"p":""},{"id":"image_2","u":"","e":1,"w":36,"h":36,"p":""},{"id":"image_3","u":"","e":1,"w":104,"h":146,"p":""},{"id":"image_4","u":"","e":1,"w":132,"h":75,"p":""},{"id":"image_5","u":"","e":1,"w":26,"h":39,"p":""},{"id":"image_6","u":"","e":1,"w":137,"h":137,"p":""},{"id":"image_7","u":"","e":1,"w":96,"h":96,"p":""},{"id":"image_8","u":"","e":1,"w":130,"h":156,"p":""},{"id":"image_9","u":"","e":1,"w":64,"h":122,"p":""},{"id":"image_10","u":"","e":1,"w":112,"h":134,"p":""},{"id":"image_11","u":"","e":1,"w":86,"h":102,"p":""},{"id":"image_12","u":"","e":1,"w":131,"h":147,"p":""},{"id":"image_13","u":"","e":1,"w":93,"h":114,"p":""},{"id":"image_14","u":"","e":1,"w":630,"h":372,"p":""},{"nm":"","id":"comp_0","layers":[{"ty":2,"nm":"Speech Bubbles copy 2","sr":1,"st":164,"op":404,"ip":164,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[47.922,48.786,0],"ix":1},"s":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.833,"y":1},"s":[80,80,100],"t":164},{"s":[100,100,100],"t":178}],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[562.108,345.329,0],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.833,"y":1},"s":[0],"t":164},{"s":[100],"t":178}],"ix":11}},"ef":[],"refId":"image_0","ind":1},{"ty":2,"nm":"Speech Bubbles copy","sr":1,"st":109,"op":349,"ip":109,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[37.917,40.313,0],"ix":1},"s":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.833,"y":1},"s":[80,80,100],"t":109},{"o":{"x":0.167,"y":0},"i":{"x":0.667,"y":1},"s":[100,100,100],"t":123},{"s":[80,80,100],"t":127}],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[581.833,110.615,0],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.833,"y":1},"s":[0],"t":109},{"o":{"x":0.167,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":123},{"s":[0],"t":127}],"ix":11}},"ef":[],"refId":"image_1","ind":2},{"ty":2,"nm":"Speech Bubbles","sr":1,"st":15,"op":255,"ip":15,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[46.916,40.813,0],"ix":1},"s":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.833,"y":1},"s":[80,80,100],"t":15},{"o":{"x":0.167,"y":0},"i":{"x":0.667,"y":1},"s":[100,100,100],"t":29},{"s":[80,80,100],"t":41}],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[146.337,320.699,0],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.833,"y":1},"s":[0],"t":15},{"o":{"x":0.167,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":29},{"s":[0],"t":41}],"ix":11}},"ef":[],"refId":"image_2","ind":3},{"ty":2,"nm":"Character 5","sr":1,"st":134,"op":614,"ip":134,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[102.779,28.634,0],"ix":1},"s":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0,"y":1},"s":[80,80,100],"t":134},{"s":[100,100,100],"t":147}],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[643.214,348.847,0],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0,"y":1},"s":[0],"t":134},{"s":[100],"t":147}],"ix":11}},"ef":[],"refId":"image_3","ind":4},{"ty":2,"nm":"bg 5","sr":1,"st":121,"op":601,"ip":121,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[162.602,-12.745,0],"ix":1},"s":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0,"y":1},"s":[80,80,100],"t":121},{"s":[100,100,100],"t":134}],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[646.524,362.654,0],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0,"y":1},"s":[0],"t":121},{"s":[100],"t":134}],"ix":11}},"ef":[],"refId":"image_4","ind":5},{"ty":2,"nm":"pin 5","sr":1,"st":121,"op":361,"ip":121,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[13.984,39.13,0],"ix":1},"s":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.833,"y":1},"s":[60,60,100],"t":121},{"s":[100,100,100],"t":133}],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[643.361,366.258,0],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.833,"y":1},"s":[0],"t":121},{"s":[100],"t":130}],"ix":11}},"ef":[],"refId":"image_5","ind":6},{"ty":2,"nm":"Character 4","sr":1,"st":105,"op":585,"ip":105,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[-1.965,59.159,0],"ix":1},"s":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0,"y":1},"s":[80,80,100],"t":105},{"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[100,100,100],"t":118},{"o":{"x":0.167,"y":0},"i":{"x":0.667,"y":1},"s":[100,100,100],"t":134},{"s":[80,80,100],"t":146}],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[537.23,146.376,0],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0,"y":1},"s":[0],"t":105},{"o":{"x":0.167,"y":0},"i":{"x":0,"y":1},"s":[100],"t":118},{"o":{"x":0.167,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":134},{"s":[0],"t":146}],"ix":11}},"ef":[],"refId":"image_6","ind":7},{"ty":2,"nm":"bg 4","sr":1,"st":92,"op":572,"ip":92,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[-62.498,38.871,0],"ix":1},"s":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0,"y":1},"s":[80,80,100],"t":92},{"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[100,100,100],"t":105},{"o":{"x":0.167,"y":0},"i":{"x":0.667,"y":1},"s":[100,100,100],"t":134},{"s":[40,40,100],"t":146}],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[540.747,138.838,0],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0,"y":1},"s":[0],"t":92},{"o":{"x":0.167,"y":0},"i":{"x":0,"y":1},"s":[100],"t":105},{"o":{"x":0.167,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":134},{"s":[100],"t":146}],"ix":11}},"ef":[],"refId":"image_7","ind":8},{"ty":2,"nm":"pin 4","sr":1,"st":92,"op":332,"ip":92,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[13.984,39.13,0],"ix":1},"s":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.833,"y":1},"s":[80,80,100],"t":92},{"s":[100,100,100],"t":104}],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[530.861,154.008,0],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.833,"y":1},"s":[0],"t":92},{"s":[100],"t":101}],"ix":11}},"ef":[],"refId":"image_5","ind":9},{"ty":4,"nm":"stroke 4","sr":1,"st":92,"op":332,"ip":92,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[-89,62,0],"ix":1},"s":{"a":0,"k":[82.069,82.069,100],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[640.5,243.5,0],"ix":2},"r":{"a":0,"k":55.439,"ix":10},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0,"y":1},"s":[0],"t":92},{"s":[100],"t":142}],"ix":11}},"ef":[],"shapes":[{"ty":"gr","bm":0,"hd":false,"mn":"ADBE Vector Group","nm":"Shape 1","ix":1,"cix":2,"np":3,"it":[{"ty":"sh","bm":0,"hd":false,"mn":"ADBE Vector Shape - Group","nm":"Path 1","ix":1,"d":1,"ks":{"a":0,"k":{"c":false,"i":[[-1,-1.5],[38.975,-42.151]],"o":[[-69.407,-265.745],[-0.5,1.5]],"v":[[30.848,224.651],[-234.525,36.547]]},"ix":2}},{"ty":"st","bm":0,"hd":false,"mn":"ADBE Vector Graphic - Stroke","nm":"Stroke 1","lc":1,"lj":1,"ml":4,"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":0.5,"ix":5},"d":[{"nm":"dash","n":"d","v":{"a":0,"k":8,"ix":1}},{"nm":"offset","n":"o","v":{"a":0,"k":0,"ix":7}}],"c":{"a":0,"k":[0,0,0],"ix":3}},{"ty":"tr","a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[90.769,53.535],"ix":3},"sk":{"a":0,"k":0,"ix":4},"p":{"a":0,"k":[0,0.5],"ix":2},"r":{"a":0,"k":0,"ix":6},"sa":{"a":0,"k":0,"ix":5},"o":{"a":0,"k":100,"ix":7}}]},{"ty":"tm","bm":0,"hd":false,"mn":"ADBE Vector Filter - Trim","nm":"Trim Paths 1","ix":2,"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":358.494,"ix":3},"s":{"a":1,"k":[{"o":{"x":0.523,"y":0},"i":{"x":0,"y":1},"s":[100],"t":92},{"s":[0],"t":176}],"ix":1},"m":1}],"ind":10},{"ty":2,"nm":"Character 3","sr":1,"st":78,"op":558,"ip":78,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[26.742,39.5,0],"ix":1},"s":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0,"y":1},"s":[80,80,100],"t":78},{"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[100,100,100],"t":91},{"o":{"x":0.167,"y":0},"i":{"x":0.667,"y":1},"s":[100,100,100],"t":107},{"s":[80,80,100],"t":119}],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[393.747,267.542,0],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0,"y":1},"s":[0],"t":78},{"o":{"x":0.167,"y":0},"i":{"x":0,"y":1},"s":[100],"t":91},{"o":{"x":0.167,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":107},{"s":[0],"t":119}],"ix":11}},"ef":[],"refId":"image_8","ind":11},{"ty":2,"nm":"bg 3","sr":1,"st":65,"op":545,"ip":65,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[54.589,36.596,0],"ix":1},"s":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0,"y":1},"s":[80,80,100],"t":65},{"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[100,100,100],"t":78},{"o":{"x":0.167,"y":0},"i":{"x":0.667,"y":1},"s":[100,100,100],"t":107},{"s":[40,40,100],"t":119}],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[394.891,289.513,0],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0,"y":1},"s":[0],"t":65},{"o":{"x":0.167,"y":0},"i":{"x":0,"y":1},"s":[100],"t":78},{"o":{"x":0.167,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":107},{"s":[100],"t":119}],"ix":11}},"ef":[],"refId":"image_9","ind":12},{"ty":2,"nm":"pin 3","sr":1,"st":65,"op":305,"ip":65,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[13.984,39.13,0],"ix":1},"s":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.833,"y":1},"s":[80,80,100],"t":65},{"s":[100,100,100],"t":77}],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[399.861,278.508,0],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.833,"y":1},"s":[0],"t":65},{"s":[100],"t":74}],"ix":11}},"ef":[],"refId":"image_5","ind":13},{"ty":4,"nm":"stroke 3","sr":1,"st":65,"op":305,"ip":65,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[620.5,236.5,0],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0,"y":1},"s":[0],"t":65},{"s":[100],"t":115}],"ix":11}},"ef":[],"shapes":[{"ty":"gr","bm":0,"hd":false,"mn":"ADBE Vector Group","nm":"Shape 1","ix":1,"cix":2,"np":3,"it":[{"ty":"sh","bm":0,"hd":false,"mn":"ADBE Vector Shape - Group","nm":"Path 1","ix":1,"d":1,"ks":{"a":0,"k":{"c":false,"i":[[-1,-1.5],[8.5,-38.5]],"o":[[-92.5,-6.5],[-0.5,1.5]],"v":[[-97.5,-98.5],[-218,16]]},"ix":2}},{"ty":"st","bm":0,"hd":false,"mn":"ADBE Vector Graphic - Stroke","nm":"Stroke 1","lc":1,"lj":1,"ml":4,"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":0.5,"ix":5},"d":[{"nm":"dash","n":"d","v":{"a":0,"k":8,"ix":1}},{"nm":"offset","n":"o","v":{"a":0,"k":0,"ix":7}}],"c":{"a":0,"k":[0,0,0],"ix":3}},{"ty":"tr","a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"sk":{"a":0,"k":0,"ix":4},"p":{"a":0,"k":[0,0],"ix":2},"r":{"a":0,"k":0,"ix":6},"sa":{"a":0,"k":0,"ix":5},"o":{"a":0,"k":100,"ix":7}}]},{"ty":"tm","bm":0,"hd":false,"mn":"ADBE Vector Filter - Trim","nm":"Trim Paths 1","ix":2,"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":358.494,"ix":3},"s":{"a":1,"k":[{"o":{"x":0.523,"y":0},"i":{"x":0,"y":1},"s":[100],"t":65},{"s":[0],"t":149}],"ix":1},"m":1}],"ind":14},{"ty":2,"nm":"Character 2","sr":1,"st":58,"op":538,"ip":58,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[116.685,62.671,0],"ix":1},"s":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0,"y":1},"s":[80,80,100],"t":58},{"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[100,100,100],"t":71},{"o":{"x":0.167,"y":0},"i":{"x":0.667,"y":1},"s":[100,100,100],"t":80},{"s":[80,80,100],"t":92}],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[118.718,192.254,0],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0,"y":1},"s":[0],"t":58},{"o":{"x":0.167,"y":0},"i":{"x":0,"y":1},"s":[100],"t":71},{"o":{"x":0.167,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":80},{"s":[0],"t":92}],"ix":11}},"ef":[],"refId":"image_10","ind":15},{"ty":2,"nm":"bg 2","sr":1,"st":45,"op":525,"ip":45,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[67.61,36.635,0],"ix":1},"s":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0,"y":1},"s":[80,80,100],"t":45},{"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[100,100,100],"t":58},{"o":{"x":0.167,"y":0},"i":{"x":0.667,"y":1},"s":[100,100,100],"t":80},{"s":[40,40,100],"t":92}],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[117.312,174.784,0],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0,"y":1},"s":[0],"t":45},{"o":{"x":0.167,"y":0},"i":{"x":0,"y":1},"s":[100],"t":58},{"o":{"x":0.167,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":80},{"s":[100],"t":92}],"ix":11}},"ef":[],"refId":"image_11","ind":16},{"ty":2,"nm":"pin 2","sr":1,"st":45,"op":285,"ip":45,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[13.984,39.13,0],"ix":1},"s":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.833,"y":1},"s":[80,80,100],"t":45},{"s":[100,100,100],"t":57}],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[128.861,186.508,0],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.833,"y":1},"s":[0],"t":45},{"s":[100],"t":54}],"ix":11}},"ef":[],"refId":"image_5","ind":17},{"ty":4,"nm":"stroke 2","sr":1,"st":45,"op":285,"ip":45,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[368,246.5,0],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0,"y":1},"s":[0],"t":45},{"s":[100],"t":95}],"ix":11}},"ef":[],"shapes":[{"ty":"gr","bm":0,"hd":false,"mn":"ADBE Vector Group","nm":"Shape 1","ix":1,"cix":2,"np":3,"it":[{"ty":"sh","bm":0,"hd":false,"mn":"ADBE Vector Shape - Group","nm":"Path 1","ix":1,"d":1,"ks":{"a":0,"k":{"c":false,"i":[[-1,-1.5],[107,-80]],"o":[[-24,-121.5],[-0.5,1.5]],"v":[[33,9],[-238.5,-74.5]]},"ix":2}},{"ty":"st","bm":0,"hd":false,"mn":"ADBE Vector Graphic - Stroke","nm":"Stroke 1","lc":1,"lj":1,"ml":4,"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":0.5,"ix":5},"d":[{"nm":"dash","n":"d","v":{"a":0,"k":8,"ix":1}},{"nm":"offset","n":"o","v":{"a":0,"k":0,"ix":7}}],"c":{"a":0,"k":[0,0,0],"ix":3}},{"ty":"tr","a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"sk":{"a":0,"k":0,"ix":4},"p":{"a":0,"k":[0,0],"ix":2},"r":{"a":0,"k":0,"ix":6},"sa":{"a":0,"k":0,"ix":5},"o":{"a":0,"k":100,"ix":7}}]},{"ty":"tm","bm":0,"hd":false,"mn":"ADBE Vector Filter - Trim","nm":"Trim Paths 1","ix":2,"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":358.494,"ix":3},"s":{"a":1,"k":[{"o":{"x":0.523,"y":0},"i":{"x":0,"y":1},"s":[100],"t":45},{"s":[0],"t":129}],"ix":1},"m":1}],"ind":18},{"ty":2,"nm":"Character ","sr":1,"st":22,"op":502,"ip":22,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[151.239,62.184,0],"ix":1},"s":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0,"y":1},"s":[80,80,100],"t":22},{"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[100,100,100],"t":40},{"o":{"x":0.167,"y":0},"i":{"x":0.667,"y":1},"s":[100,100,100],"t":49},{"s":[80,80,100],"t":61}],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[235.787,338.535,0],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0,"y":1},"s":[0],"t":22},{"o":{"x":0.167,"y":0},"i":{"x":0,"y":1},"s":[100],"t":40},{"o":{"x":0.167,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":49},{"s":[0],"t":61}],"ix":11}},"ef":[],"refId":"image_12","ind":19},{"ty":2,"nm":"bg ","sr":1,"st":9,"op":489,"ip":9,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[84.16,41.776,0],"ix":1},"s":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0,"y":1},"s":[80,80,100],"t":9},{"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[100,100,100],"t":22},{"o":{"x":0.167,"y":0},"i":{"x":0.667,"y":1},"s":[100,100,100],"t":49},{"s":[40,40,100],"t":61}],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[219.747,336.591,0],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0,"y":1},"s":[3],"t":9},{"o":{"x":0.167,"y":0},"i":{"x":0,"y":1},"s":[100],"t":22},{"o":{"x":0.167,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":49},{"s":[100],"t":61}],"ix":11}},"ef":[],"refId":"image_13","ind":20},{"ty":2,"nm":"pin ","sr":1,"st":9,"op":249,"ip":9,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[13.984,39.13,0],"ix":1},"s":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.833,"y":1},"s":[80,80,100],"t":9},{"s":[100,100,100],"t":21}],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[234.861,349.508,0],"ix":2},"r":{"a":0,"k":0,"ix":10},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.833,"y":1},"s":[0],"t":9},{"s":[100],"t":18}],"ix":11}},"ef":[],"refId":"image_5","ind":21},{"ty":4,"nm":"stroke","sr":1,"st":9,"op":249,"ip":9,"hd":false,"ddd":0,"bm":0,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[-182.542,-11.24,0],"ix":1},"s":{"a":0,"k":[-100,100,100],"ix":6},"sk":{"a":0,"k":0},"p":{"a":0,"k":[196,234,0],"ix":2},"r":{"a":0,"k":107.41,"ix":10},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0,"y":1},"s":[0],"t":9},{"s":[100],"t":59}],"ix":11}},"ef":[],"shapes":[{"ty":"gr","bm":0,"hd":false,"mn":"ADBE Vector Group","nm":"Shape 1","ix":1,"cix":2,"np":3,"it":[{"ty":"sh","bm":0,"hd":false,"mn":"ADBE Vector Shape - Group","nm":"Path 1","ix":1,"d":1,"ks":{"a":0,"k":{"c":false,"i":[[-14.5,39],[74,-0.5]],"o":[[19.5,-52],[-85,7]],"v":[[-143.5,64],[-238.5,-74.5]]},"ix":2}},{"ty":"st","bm":0,"hd":false,"mn":"ADBE Vector Graphic - Stroke","nm":"Stroke 1","lc":1,"lj":1,"ml":4,"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":0.5,"ix":5},"d":[{"nm":"dash","n":"d","v":{"a":0,"k":8,"ix":1}},{"nm":"offset","n":"o","v":{"a":0,"k":0,"ix":7}}],"c":{"a":0,"k":[0,0,0],"ix":3}},{"ty":"tr","a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"sk":{"a":0,"k":0,"ix":4},"p":{"a":0,"k":[0,0],"ix":2},"r":{"a":0,"k":0,"ix":6},"sa":{"a":0,"k":0,"ix":5},"o":{"a":0,"k":100,"ix":7}}]},{"ty":"tm","bm":0,"hd":false,"mn":"ADBE Vector Filter - Trim","nm":"Trim Paths 1","ix":2,"e":{"a":0,"k":100,"ix":2},"o":{"a":0,"k":-360,"ix":3},"s":{"a":1,"k":[{"o":{"x":0.523,"y":0},"i":{"x":0,"y":1},"s":[100],"t":9},{"s":[0],"t":93}],"ix":1},"m":1}],"ind":22}]}]} --------------------------------------------------------------------------------