React Native II Flashcards
What are we going to learn in React Native II?
data:image/s3,"s3://crabby-images/c34fe/c34feb203799da13b10734de600621aea6d13528" alt=""
Mosh worked really hard on this course
Promise you’ll be a React-Native Ninja
Tons of Tips and Tricks
If we build an App or get a job, share story!
Advanced Concepts:
Native Device features (Android/iOS)
Navigation
Networking (Api calls)
Offline support
Authentication & Authorization
Push Notifications
Distrubution (App stores)
data:image/s3,"s3://crabby-images/a1ac0/a1ac0e407278d11d4a685cee10e0956d6849c1a1" alt=""
What do we need to know?
Ultimate React Native Part I
React Hooks
(useState, useEffect, custom hooks)
React Context
Javascript
(arrow fns, spread, async/await)
What native features are not available in React Native?
React Native has limited API for working with native features:
Camera, context not available
available as 3rd party library
Ex. image picker -> allows us to get an image from camera
have to install a library
run in iOS project
can eject from expo
or
use components by expo
lots of native features available
data:image/s3,"s3://crabby-images/44a19/44a193955d656c3cec74e93cc7fb2195d4e9a48c" alt=""
What is the ImagePicker component?
Can select a video or image from user’s library or camera
expo install expo-image-picker
What is useEffect (effect hook) for?
accessing lifecycle hooks like componentDidMount ( ) in functional component
data:image/s3,"s3://crabby-images/3c16a/3c16ac4038ba55ab62ed5839816962f39ec8897f" alt=""
How can we access a user’s camera?
ImagePicker component from ‘expo-image-picker’
data:image/s3,"s3://crabby-images/0be89/0be89c660313f73bc1fa969eeaccd9d6103e13eb" alt=""
What is the permissions API?
data:image/s3,"s3://crabby-images/19a0b/19a0b39a0698b33c006f56a9f45e6fd029d8fb8a" alt=""
another method for accessing camera, etc.
can pass multiple permission types
data:image/s3,"s3://crabby-images/692a5/692a5372f10fdc64c75367cd962fe3c9dd0d56c2" alt=""
How can we access a user’s library to add a photo and display on the screen?
data:image/s3,"s3://crabby-images/5ddd6/5ddd699c633b8a1d14937491f22a01167d8bd23a" alt=""
useState() hook
to set result.uri (from ImagePicker)
display imageUri in Image component
data:image/s3,"s3://crabby-images/4b572/4b572afcca1c9d8480a9554ec4bd82e7d99b1ba7" alt=""
How do we build a re-usable ImageInput component?
data:image/s3,"s3://crabby-images/bc9a5/bc9a522e6cd7f4697582709b6acce8b2e6690d9a" alt=""
data:image/s3,"s3://crabby-images/7f438/7f438f2165af1eedc38c2fb306edb03fb7b93b1d" alt=""
How do we handle the touch event of our image picker component?
consumer of ImageInput:
handles state using hooks
ImageInput:
props - imageUri, onChangeInput (reference to hook from consumer for updating data of imageUri)
wrap in a TouchableWithoutFeedback
set handlePress to fn for calling ImagePicker.launchImageLibrary
data:image/s3,"s3://crabby-images/88c05/88c0520a08236b2456e78a3748b0c318f3205466" alt=""
How do we build the image input list component?
data:image/s3,"s3://crabby-images/2a600/2a60003b79fdd794166f3b6e78b07eec82b6f970" alt=""
data:image/s3,"s3://crabby-images/90306/9030686d0f3983858ca4fe1c430cdced7fd69e75" alt=""
How do we solve this problem?
Hint: Horizontal scrolling
data:image/s3,"s3://crabby-images/4f6da/4f6dab9487224284393a13aa7d75401bd68b1218" alt=""
ScrollView component from react-native
data:image/s3,"s3://crabby-images/70d4f/70d4f7bbe89b2139912212568e4aa058172e2d3a" alt=""
wrap ScrollView in a View so it only takes up content space
use useRef() hook to get current element and scroll to end
How do we design a form image picker?
wrap our ImageInputList in a new component with an error message
Just like FormPicker, FormField
allow Formik to handle state
use the values field to get form values
use the SetFieldValue method to update form values
data:image/s3,"s3://crabby-images/99df9/99df98bfd1d9a0830cbe27d42d5c0fda0d0ccf45" alt=""
How do we add the Form Image Picker to our Listing Edit Screen?
data:image/s3,"s3://crabby-images/ba9ba/ba9baa9c9a872cdc1ac91f399de40ab695f08db0" alt=""
Consume FormImagePicker in ListingEditScreen
add validation to images using Yup schema in ListingEditScreen
data:image/s3,"s3://crabby-images/1b8f5/1b8f531551e68d05d73ca575889fd59df49ee819" alt=""
How do we get the user’s location?
define function:
getLocation( ) to async call Location.getLastKnownPosition()
useEffect :
to call getLocation( )
useState:
set location and get location
data:image/s3,"s3://crabby-images/c3883/c3883ec86b418d6699aed33c9c4348e96e7099b9" alt=""
What is a custom hook?
Encapsulate some state and logic around state
in a re-usable component
naming convention:
useLocation
*start with “use”
data:image/s3,"s3://crabby-images/36859/3685910a1e810fb1cca1cb960b53a2cb804d3cc3" alt=""
What are we going to learn in navigation?
data:image/s3,"s3://crabby-images/1a79b/1a79b09baec653400b7c1cc4402a4544973780da" alt=""
How to take App to next level
add navigation at bottom
can click on screens
App is becoming more functional
How do we implement navigation in our React Native Apps?
data:image/s3,"s3://crabby-images/d15b7/d15b7e20827e8e8aa9849152d547010effe665bc" alt=""
What kind of navigators do we have in React?
Stack - taking users from one page to another
Tab - going between tabs
Drawer - drawing things on screen navigations
**all of them are very similar and can learn if you know one!
data:image/s3,"s3://crabby-images/d35fa/d35faeaf7046172a3e87f02105f5fdcea769fb07" alt=""
How do we use the stack navigator?
data:image/s3,"s3://crabby-images/feeb8/feeb893ada557811b226bfcaff4e4d95dd437c14" alt=""
Each navigator is implemented in a separate library
data:image/s3,"s3://crabby-images/826df/826df6a9a18dda377a4168fd917ef605b1f63c1a" alt=""
We render StackNavigatorComponent
it knows how to navigate between components
What does React Navigation give our screen components?
data:image/s3,"s3://crabby-images/d8095/d8095f5123413d6cae797dd075c417e3081efe0d" alt=""
A special prop called navigation
can destructure and pick navigation property
Ex. navigation.navigate(“Target Location”)
only available in screen component
not available in child components
.push creates a new instance on the stack
.navigate creates a single instance on stack
but can use navigation hook to access navigation object
data:image/s3,"s3://crabby-images/d38a2/d38a225a2e17b442c72869883f155df963c016c1" alt=""
How do we access navigation prop using React Navigator hook?
useNavigation( ) hook
data:image/s3,"s3://crabby-images/a16db/a16db8f0750f19df19c207be4001a8e296dc61d4" alt=""
How do we pass parameters to React Navigator?
ex. pass ID of a specific tweet
data:image/s3,"s3://crabby-images/d599b/d599b0abcd5b6146f042e06568a6c0e0c61e78bd" alt=""
How can we set the title of a screen?
data:image/s3,"s3://crabby-images/0adca/0adcafd091a3909cc1e7b10334c0de9a1efdf018" alt=""
using the options prop of the Stack.Screen component
data:image/s3,"s3://crabby-images/2ba28/2ba28c7e0b6e8f07150d39213b8374f9beb19efa" alt=""
How do we customize the header of a screen?
data:image/s3,"s3://crabby-images/de79e/de79ea9880905cacebbcdb6d2759a733718015cf" alt=""
Locally (screen only)
use the options prop of the Stack.Screen component
headerStyle (pass a color)
headerTintColor
headerShown
Globally (all screens)
options prop of the Stack.Navigator component
can override for specific screen
data:image/s3,"s3://crabby-images/9da02/9da02c614a09e283e9626ee743a7c5d3d5242361" alt=""
How do we create a bottom tab navigator?
data:image/s3,"s3://crabby-images/a13e0/a13e0b0087d51485dd72311a2b70174449480abf" alt=""
npm install @react-navigation/bottom-tabs
createBottomTabNavigator()
returns an object
has Screen and Navigator properties
render
TabNavigator inside NavigationContainer
data:image/s3,"s3://crabby-images/ef7a1/ef7a100e2cae364b1916dcf77f5f5c227580c17a" alt=""
How do we add a background color to each tab and change the text when pressed?
createBottomTabNavigator()
tabBarOptions = styles
/>
data:image/s3,"s3://crabby-images/1f443/1f44345f72806a095881d6b8b702fdd809c1b9e8" alt=""
How can we add icons to our Tabs?
Really easy
Tab.Screen
options - set to a fn that renders a MaterialCommunityIcon
destructure the props from React Navigator
set size and color to props given by Navigator
data:image/s3,"s3://crabby-images/1065e/1065e47575a2f3de638450433c6480499deb67b8" alt=""
What is nesting navigators?
instead of passing a component to the Tab.Screen component (bottomTabNavigator)
passing a Stack Navigator
called nesting navigators
data:image/s3,"s3://crabby-images/f2bbe/f2bbedb846edc125ed1c7ed0d68186e23bee7aa8" alt=""
How do we build the welcome screen to login screen / register screen navigation?
AuthNavigator.js
use createStackNavigator( ) from @react-navigation/stack
from the return object
create Screen component and Navigator container
Consuming Screen:
inherits navigation prop
uses navigate method onPress of button
data:image/s3,"s3://crabby-images/e537a/e537afe597ee8d8b4635b9e59e95087daf98be88" alt=""
How do we set a custom theme for our app?
spread the …DefaultTheme
data:image/s3,"s3://crabby-images/b4a89/b4a899e203a0b717a6a3646800a2fa3879ce136a" alt=""
can update headers in Stack.Screen
pass options prop
headerStyle: { backgroundColor: ‘white’ }
How do we create the AppNavigator component?
data:image/s3,"s3://crabby-images/2eabd/2eabde605e27721376fef46af9a9be44f9d0ce2a" alt=""
data:image/s3,"s3://crabby-images/5103f/5103ff2f66cc4856edc24bfd027dbcc08bcb33b2" alt=""
How do we create navigation from the listings screen to press on an item and be directed to the item details screen?
data:image/s3,"s3://crabby-images/aac84/aac8452849471b3c0d1c81fa20fcbf62fbe791b7" alt=""
add TouchableWithoutFeedback to card component
assign the onPress event via props to consuming class
because we have registered the screen with our navigator….
can access the navigation or route props
handle onPress in ListingScreen by extracting navigation prop (ListingsScreen)
extract route param, use it to get item
listing.title, listing.price
dynamically display item
data:image/s3,"s3://crabby-images/d6541/d6541ec50ad5af6d4d2487c71eece20525172a60" alt=""
How can we navigate to an item dynamically?
data:image/s3,"s3://crabby-images/61b75/61b756ae4876425aa398aa6a3ce2fb43bee76b67" alt=""
wrap card component in a Touchable
set the event to props (onPress)
handle in ListingsScreen (navigation prop)
handle ListingDetailsScreen (route to get item[param])
data:image/s3,"s3://crabby-images/9319c/9319c46f164483841712eb14e74792c895f4d61c" alt=""
How do we make our listing details screen appear from the bottom vs from the side?
data:image/s3,"s3://crabby-images/25fc2/25fc2b99819adc9481f022a0d16880ec5ce71b7b" alt=""
Set Stack.Navigator prop to modal
data:image/s3,"s3://crabby-images/9c417/9c417f207c27e83cc2b16a1b67b1856d2c545528" alt=""
What does our account screen need?
data:image/s3,"s3://crabby-images/291bc/291bc7d57b1ad403a32af1c7dee9613062d5db5d" alt=""
It’s own stack navigator for navigating to it’s components
data:image/s3,"s3://crabby-images/c21f8/c21f8a6660b46cadb54fde8b8183fbb6d73d9ea9" alt=""
How do we make our AppNavigator tabs beautiful?
data:image/s3,"s3://crabby-images/018d2/018d2422a92e45583947b68b8a4a0e9287fb10d7" alt=""
create NewListingButton component
let consumer decide what should happen when onPress event
hook with prop
data:image/s3,"s3://crabby-images/ff02a/ff02a52c6661207758333e3c40f0931dbdbb1169" alt=""
What should we do to make our navigation cleaner?
data:image/s3,"s3://crabby-images/2c346/2c3467e851ea497abb64b9b365939c276ba3919c" alt=""
Get ride of hard coded route names
if there is a typo, app will not work
testing every feature manually or automated
if rename route, have to rename all instances
solns?
store all routes in a single place in app
Object.freeze ensures object cannnot be modified anywhere else in our app
data:image/s3,"s3://crabby-images/3a376/3a3763ca74f20c02986d530d66e8fdb4d0977669" alt=""
What are we going to learn in the networking section?
How to connect our React-Native app to a node.js server
data:image/s3,"s3://crabby-images/077a6/077a612fa980b1068ec2eb69946f5554b50abaad" alt=""
created a fake database
Why?
Don’t need to setup MongoDB, etc.
What library are we going to use for calling APIs?
Hint: npm i apisauce
data:image/s3,"s3://crabby-images/b2f34/b2f34a5d4883855595be1517fef178d181c1fe28" alt=""
Api Sauce
an abstraction over axios
can use fetch API, but using api sauce
why?
whenever call server, promise always resolved, even if an error
just check if there is an error
no try/catch needed
gives us standardized errors
data:image/s3,"s3://crabby-images/4b738/4b738a920fabac73d77edc0bf2a1a35a7593da78" alt=""
What should our components know?
data:image/s3,"s3://crabby-images/94d14/94d14ab0f9f3308ea6449c5d44d62ebf057b6529" alt=""
What UI looks like
How UI behaves
Why?
separation of concerns
NOT:
HTTP protocol
calling GET, POST, ETC.
not responsibility of component
Don’t do this, violates separation of concerns
Soln?
Create an API layer that handles all this
proper architecture
data:image/s3,"s3://crabby-images/f7174/f71745ec7a6d5b852970fb81859744af63e2e9fc" alt=""
How do we create our basic API layer?
data:image/s3,"s3://crabby-images/fc999/fc9993da457c4b0d4d61632409d61b6187df3611" alt=""
How can we get the list of items from our backend?
data:image/s3,"s3://crabby-images/6e0aa/6e0aa0210d135c5b4454434bdba6e4f7553b9167" alt=""
How do we debug our API calls to see if the server was called as well as the request and response objects?
data:image/s3,"s3://crabby-images/c0861/c08615729ea413f4e20186fc0ca229bc8fa8fbdc" alt=""
Look at network requests in React Native Debugger
data:image/s3,"s3://crabby-images/60394/6039473e27d8a8432f55bac135a5367d3edcf962" alt=""
How should we handle an error?
should provide a good user experience
Not a blank screen
How?
use another hook to catch an error from APISauce
conditionally render the error with a Text component
and a Button component
data:image/s3,"s3://crabby-images/53532/53532b75ba4b040fc9ab2b7071096790011b7966" alt=""
How do we simulate a slow network connection?
React native developer tools
makes our app slow
takes 2 seconds of a blank screen before listings
soln?
Show an animation before listings shown
data:image/s3,"s3://crabby-images/db8d4/db8d4f62db133a3cc433a86e4d8fba8e8bf88c0b" alt=""
How do we display an activity indicator?
data:image/s3,"s3://crabby-images/b9736/b973614a25fa096674bbcf6700a8ab7bb9ebf7cb" alt=""
ActivityIndicator component
defined in react native core
use a separate hook for loading, setLoading
render component if loading is true
kind of boring can replace with a beautiful animation
data:image/s3,"s3://crabby-images/ee8a5/ee8a5e718b2aed3316d7103b8d328e2614e5b66e" alt=""
What is Lottie?
data:image/s3,"s3://crabby-images/e03cd/e03cdb03890b1204016f56f4d0199bfceaf88f47" alt=""
an animation library built by AirBnB
Can add beautiful animations
JSON file exported from Adobe After Effects
LottieFiles.com
thousands of animations made by people
most free, some paid
data:image/s3,"s3://crabby-images/0d815/0d815f9a5e46be2b1f6881887e3e5216c7d5f97c" alt=""
How can we add a beautiful Lottie animation to our application?
Hint: expo install lottie-react-native
data:image/s3,"s3://crabby-images/106f6/106f68580eab926b123c04d7c5bf548ff29e44c6" alt=""
Go to LottieFiles.com
customize a free animation
download the JSON
upload into project
render using a special component
data:image/s3,"s3://crabby-images/4d250/4d250d0023a6eba6f22afdd3d7c9ba1b98c382a9" alt=""
What’s the problem with this implementation?
data:image/s3,"s3://crabby-images/befbe/befbe86a42340b53269cc29b10bd088ec86f491a" alt=""
have to repeat logic everytime want to call server
declare hooks, etc.
is there a better way?
repetative, redundant
Soln?
use hooks
encapsulate some state and logic around
can encapsulate details around a re-usable hook
data:image/s3,"s3://crabby-images/2bebc/2bebcdc059df4f7f88345a0a5e45e58f1f7a579b" alt=""
How do we create a re-useable API hook?
extract the useState (state)
extract the logic that modifies state
data:image/s3,"s3://crabby-images/09eaa/09eaaca29c79972403b7ad5b4acee5ba07297a22" alt=""
How do we pass props to our re-usable useApi hook?
use rest operator to allow request function in useApi
to take zero or more args
data:image/s3,"s3://crabby-images/14fd5/14fd5c67fb0d6eeba20159e700cc768e4e9ca9d9" alt=""
How can we call multiple Api’s in on a screen?
name each Api call and pass to useApi function
data:image/s3,"s3://crabby-images/d75b8/d75b8d9ddd122e8cfc462ffe3501a6b38ce5bc04" alt=""
How do we post data to the server?
(ie - send a new listing to the server?)
data:image/s3,"s3://crabby-images/1852d/1852df4135d02a2a9cbb6ede7a45e477771691f7" alt=""
How do we get upload progress?
pass a third object to axios (config)
{onUploadProgress: progress => console.log(progress)}
this object contains loaded and total bytes
(loaded bytes / total bytes) * 100 = progress
How?
Pass fn reference to child (raising event)
child calls event when event is raised
gives access to progress in UI layer
data:image/s3,"s3://crabby-images/6b889/6b88929859b37acf6c31ff734ecb16f86da0eb0f" alt=""
How do we build the upload screen?
UploadScreen
composed of a Modal
Props of progress, visible
ListingEditScreen
composed of UploadScreen
uses hooks to handle progress, visibility
data:image/s3,"s3://crabby-images/f78cd/f78cde5ec4523762a22492c0a41765c12a3f342e" alt=""
How do we add a beautiful progress animation to our application?
npm install react-native-progress –save
hook for storing progress / visible
pass progress / visible to UploadScreen
in UploadScreen import progress bar
data:image/s3,"s3://crabby-images/2ec88/2ec8886b3cfe05c390f45d2cb164dd27a97c3de1" alt=""
How do we display a beautiful done animation?
UploadScreen
conditionally render
Progress.Bar or LottieView
raise onDone event after onAnimationFinish
ListingEditScreen
handle onDone event
data:image/s3,"s3://crabby-images/32722/32722025668523a1a7d85a5996ab93c4e629d2ed" alt=""
How do we reset our form after making a post to the server?
pass resetForm from formik (fn)
ensure FormFields use formik setFieldValue, value from FormikContext so that resetForm can clear values
data:image/s3,"s3://crabby-images/acbcb/acbcb36c8d06cb127793bfe9a0747d3e11448a3c" alt=""
What will we learn in the offline support section?
A very important topic
lots of otherbooks / courses overlook topic
What?
if no internet connection
reloads listings from a cache
data:image/s3,"s3://crabby-images/2a3a4/2a3a4cac91d26bfdb754d95e88a0b4ef887e8fef" alt=""
What are strategies for offline capable apps?
*as we build more offline capability, cost increases
Think about time, budget, uses before implementing in apps
Four Strategies:
Notify the user
that there is no internet connection
Disable features
turn off certain feature capabilities while offline
Cache data
data available while offline
Store user actions
if user takes actions, store actions locally on device
when online is back, execute them on server
data:image/s3,"s3://crabby-images/76f7f/76f7f158f8cd7902eb238b4a70b2b688dd92a35a" alt=""
How do we detect the status of a users network connection?
data:image/s3,"s3://crabby-images/99a7f/99a7fbaf53ee7d8b3b6562dd566e58f2e918a6a8" alt=""
expo install @react-native-community/netinfo
NetInfo.fetch() returns an object…
isInternetReachable
check this property
isConnected
doesn’t mean has internet
data:image/s3,"s3://crabby-images/8cec7/8cec7b8c972ede42d81c4e027b8f1c9c606add21" alt=""
How can we subscribe to network status changes?
data:image/s3,"s3://crabby-images/19b23/19b23c8ae0261b935cddd8b3e2b05a03fc717c54" alt=""
Class Components:
ComponentDidMount
call const unsubscribe = NetInfo.addEventListener()
Function called everytime a change to network status
ComponentDidUnmount
unsubscribe ( )
prevent memory leaks
Functional Components
useNetInfo ( ) hook
subscribe / unsubscribe happens inside hook
data:image/s3,"s3://crabby-images/eedd5/eedd594628294d18652ae69c0443224535e4d4be" alt=""
What is caching?
When we call an API to retrieve data from server
storing successful request data locally on device, so if mobile device goes offline
can read data from cache
How?
AsyncStorage -
data:image/s3,"s3://crabby-images/1d32f/1d32ff95c3ffa15b94c9bc8f5e05c26b5549f44a" alt=""
What are the three options for caching successful API response data locally on the mobile device?
data:image/s3,"s3://crabby-images/7b2e4/7b2e4e3e854b5b591ab29b828d922752c6d68915" alt=""
AsyncStorage
values are not encrypted
like local storage in browsers
stores key value pairs
values are strings, not encrypted
6MB limited storage
don’t store large data or sensitive data
SecureStore
expo built API
2MB limit
SQLite
SQL like database
if query data using SQL, use this option
data:image/s3,"s3://crabby-images/0e3dd/0e3dd3d4848be6a4782cda194a307629d982f044" alt=""
How do we store a value in async storage?
Hint: expo install @react-native-community/async-storage
data:image/s3,"s3://crabby-images/a51da/a51dac1a68cea8a2fc439b4a17d64beadb9bfa72" alt=""
How can we debug our async storage in React Native debugger?
open react native debugger
in console
call showAsyncStorageContentInDev()
work directly with Async functions
$reactNative.AsyncStorage.clear()
if storage is corrupt, need to clear, or insert values
data:image/s3,"s3://crabby-images/aa467/aa4674ef9f2bd55a1dc8206e5402c9f48a5982ae" alt=""
How do we store listings in cache in our app so that if the network connection is down we can read data from our cache?
data:image/s3,"s3://crabby-images/04539/04539d306c28703c8555eae7bf1883cd4c733390" alt=""
caching listings:
don’t have complex data to query
don’t have sensitive information
use AsynStorage
How does AsyncStorage work under the hood?
We don’t have to worry about these implementation details
just store in AsyncStorage and let them figure out the details
hide behind a cache layer (don’t use directly)
Android
uses SQLite
iOS
uses dictionary (hash map) or separate files
data:image/s3,"s3://crabby-images/83ac8/83ac8fa1c5caf7c119d8354846895dcbbffd81e9" alt=""
What is a cache layer?
data:image/s3,"s3://crabby-images/ce830/ce8304f3cf48d81999841f2f3e5955caa7893880" alt=""
abstraction over AsyncStorage
will handle serialization / de-serialization logic
will apply a timestamp (expired or not)
automatically remove expired data
Why?
Encapsulate all details in a single place
Prevents data leakage
minimizes breaking changes if we change cache implementation
What is the command-query separation (CQS)?
data:image/s3,"s3://crabby-images/a26c9/a26c9d2e4cf4bce0b995b657436a7ed076e081d8" alt=""
an important programming principle
functions should either be commands or queries
Command
fn that changes state of system
Query
fn that returns the state of the system
What’s the problem with this code?
data:image/s3,"s3://crabby-images/a2a8c/a2a8c694eaec55b2f135f73b19a1a3e8f48754f5" alt=""
it breaks the command query separation principle
the function is both returning the state and changing the state in the same function
But…
we are consciously breaking the rules
we don’t want to store too much data in the cache
What’s the problem with this code?
data:image/s3,"s3://crabby-images/85a4d/85a4db2f7c5fb4ca1e46544ed006caff2fc3ae9b" alt=""
it’s violating the single responsibility principle
fn or method should have a single responsibility
get ( ) function has two responsibilities:
- get item from cache
- detecting if item is expired or not
Soln?
move logic somewhere else
define new function isExpired()
How do we bring the caching layer into the api layer?
data:image/s3,"s3://crabby-images/f8861/f886179c5ecbe156eadcf662c6c3a5b10b0c43f6" alt=""
in the client.js file
logic for caching should go where calling server
store get method in a constant
call original get ( )
perform some logic before / after
data:image/s3,"s3://crabby-images/9905b/9905b8aad0f3363c7c972bb57f4abbcb37d7cba4" alt=""
How do we cache images?
data:image/s3,"s3://crabby-images/5d43b/5d43b510a3078cfcd59a553bbd98832cc3cd86a7" alt=""
Hint:
npm i react-native-expo-image-cache
npm i expo-blur
popular library gives replacement imagecomponent
(vs react native image component)
Why?
Supports caching, progress loading, authorization headers
But?
Not available for expo aps
Soln?
react-native-expo-image-cache
How can we cache images in our app?
data:image/s3,"s3://crabby-images/52e16/52e1612d9d3d7e7e685296a624ed1b8456bb7110" alt=""
use Image component from expo-image-cache
slighly different API
images now stored in files on device
data:image/s3,"s3://crabby-images/10227/102279f336420e333218bb028ebc16a745d33bd2" alt=""
What is Redux?
data:image/s3,"s3://crabby-images/64e52/64e529c942440d2839e85b75ef2e7962ef1eae77" alt=""
Used in React and React Native Apps
many people use Redux for caching and offline support
store entire app state in a single object
like a database for front end
redux-persist
persist and rehydrate a redux store
BUT…
Many people can’t live without redux
it brings extra complexity
simplicity is the ultimate sophistication
data:image/s3,"s3://crabby-images/cce9f/cce9f123332b08311afa2cf0a4ad8805b3a05c09" alt=""
Why is using Redux not an elegant solution to caching problem?
Adds a lot more complexity
Ask yourself..
Does undue/redue really matter for your app?
know state of app and go back in time really matter?
Only use Redux if you have a complex data flow in your app
don’t use redux for every single app
When use?
When having multiple screens, sharing data, updating
Screens must be in synch
otherwise, don’t use redux
data:image/s3,"s3://crabby-images/fb47d/fb47d139667d60dd1282d7425b2f962beda64c5c" alt=""
How can we store user’s actions and execute when back online?
store post request locally in app, execute later
modify API layer - post, put methods to execute later
complex app with various apps?
complex actions to handle like retrying actions, conflict resolution, optimistic updates
don’t implement by hand
use Firebase
real time database, offline support, auto synchronization
Fire saves changes automatically
Firebase would be backend of app
use Realm
automatically synchs with database
ask?
Is this necessary for my application
Twitter doesn’t allow posting tweets when offline
data:image/s3,"s3://crabby-images/0773d/0773ddacf5e19c5c9a2c5bf6da9e34df794b0c62" alt=""
How do we display a notice to the user if the mobile device is not connected to the internet?
useNetInfo ( ) hook
data:image/s3,"s3://crabby-images/3e6b0/3e6b0fc4b294e0cf0241b286415238f25f5ed91e" alt=""
What will we learn in the authentication and authorization section?
Users will be able to register and login
taking App to the next level
Why should we subscribe to React context vs. use Redux?
React Context replaces prop drilling
Redux is useful for large objects that change frequently
Why?
any changes to a context object, causes entire app tree to re-render
redux store components subscribe individually
only components that use / need data are re-rendered
data:image/s3,"s3://crabby-images/f2f84/f2f84ed802a0a090df8053d78631541554e0e840" alt=""
How do we login a user?
create an auth.js api file
use client (abstraction over apiSauce)
to call endpoint and pass email, password
LoginScreen
onSubmit (handleSubmit)
define login for handleSubmit
use authApi to login by sending email / password from form
if error, use state variable to display error message
data:image/s3,"s3://crabby-images/9616c/9616cbd9bb265657987f2f586039753f6894298d" alt=""
How do we decode the JWT sent from the server to access the payload?
data:image/s3,"s3://crabby-images/6e7b3/6e7b3f797fbab3fe4a6af47a4921cd7295797ebd" alt=""
npm i jwt-decode
data:image/s3,"s3://crabby-images/19cc8/19cc806a07ebf1fec5171dffb0f2e6107fcc5de4" alt=""
How do we give the current user to all the screens in our application?
wrap in Context (react context)
pass hook (user, setUser) to all components using context
data:image/s3,"s3://crabby-images/3277b/3277bcf039fa3a094b03bca836b6fd6201b2cb90" alt=""
How do we replace the hard coded data in the account screen with the current user?
data:image/s3,"s3://crabby-images/21bdf/21bdfd16fb58b1804200a93e4df85f4cd4b091d7" alt=""
get user from context
useContext(AuthContext)
data:image/s3,"s3://crabby-images/5a53c/5a53c7868c52082316582ffbf9da04663e4e93b9" alt=""
How do we implement logging a user out?
AccountScreen
extract setUser from context
onPress () => setUser(null)
redirects to the login / register page
data:image/s3,"s3://crabby-images/87334/873344203435b709bd7d5dba99474267e22d5129" alt=""
How do we persist the authentication?
data:image/s3,"s3://crabby-images/98812/98812e1991cf94a20893f1d48cf08e723202db1f" alt=""
Auth Context vs. SecureStorage
two places in memory to persist
store auth token on device
use Secure Store
storage.js
responsible for storing / retrieving auth token
like AsynStorage but for secure data
data:image/s3,"s3://crabby-images/bf953/bf953175ce83d33789f86805e7b995478185cd32" alt=""
What is the AppLoading component?
data:image/s3,"s3://crabby-images/59378/5937852b1b3d96dbeb6da3e4cecbafedffd9bd4c" alt=""
data:image/s3,"s3://crabby-images/c9d42/c9d4210133cb9f179abcd74451657be6f42a6a26" alt=""
What is the problem with this code?
data:image/s3,"s3://crabby-images/fddb7/fddb7b3205e7f67adec9986c9eaab7877fa1fe10" alt=""
App component
should be like a manager
delegating, not implementing
too many details of authentication inside App
Soln?
Refactoring by moving logic to auth module
data:image/s3,"s3://crabby-images/989fc/989fc6fcef54a8764a66dd2cdb4904ebc36931f3" alt=""
How can we make this code better?
data:image/s3,"s3://crabby-images/9d5de/9d5de6bde38d4e455bd3c5508c14ec40597725a1" alt=""
All business logic into custom hook
App is only responsible for delegation
How?
extract useContext(AuthContext)
into a module that abstracts these details
instead of importing the context hook
and the authContext hook
only import custom hook
Encapsulate
setUser()
authStorage.removeToken()
into custom hook
data:image/s3,"s3://crabby-images/37269/37269252cca11ef2facc55884e3dab22266c2df7" alt=""
How can we refactor this code?
data:image/s3,"s3://crabby-images/69acd/69acd2060fed45e293115e8c09311bc327dc6d16" alt=""
separate concerns
encapsulate logic for logIn ( )
fn updates Context and SecureStorage
both are encapsulated in custom hook
handle logic around the state there
export logIn function
import in LoginScreen
cleaner code
data:image/s3,"s3://crabby-images/06024/060245c93d0bf88dac99a00dd49f8588c7650d59" alt=""
How can we access protected routes?
data:image/s3,"s3://crabby-images/2b600/2b60007c0cad59f456935eb15387a7215119278a" alt=""
update client.js
transform (middleware) request
automatically include authToken
data:image/s3,"s3://crabby-images/bb129/bb12919ec100e5c97bc1f28295ad8bb21d831df1" alt=""
How do we register a new user?
usersApi
create users.js in api module
handles calling api
RegisterScreen
pass formik object (userInfo) to usersApi
How?
Register user
usersApi.register(userInfo)
formik object
Login user
authApi.login
data:image/s3,"s3://crabby-images/4101a/4101a4d62995f6756a014dc0c71795d48ccd3961" alt=""
How do we add an overlay to the listing screen?
data:image/s3,"s3://crabby-images/7ea58/7ea58b9264c474c4152e2310f8f15f7aedf27d10" alt=""
How can we send push notifications?
Expo Push Notification
simplest
hides all complexity
by far the best
OneSignal, Amazon, etc.
have to eject from expo
make changes to native projects
lots of extra work
if you miss any steps, don’t work
review all the steps
Recommendation?
Use expo unless job says you have to use something else
data:image/s3,"s3://crabby-images/69339/69339b8561d5f6f0291c55c9912b053cdaddfcd2" alt=""
What is Amazon Simple Notification Service?
data:image/s3,"s3://crabby-images/edc41/edc41d00bb61517c9bdb7351a3f340d193fbca50" alt=""
have to make changes to native projects
to setup notifications
any steps missed or not done properly
have to review all steps, painfully
data:image/s3,"s3://crabby-images/afb98/afb983b545f4176ba9e5c417c5913afbc6742aeb" alt=""
What are the four steps we need to execute to setup push notifications using expo?
Register app with expo
get push notification token
uniquely identifies each device
can send notification to unique device
store token on device
somewhere stored a collection of users
each user - Id, name, email, pushNotificationToken
can send notifications to a user or group of users
Send notification
happens on server
not on the client
UI has a box for typing a message
onSubmit sent to server
server stores message in DB
server notifies user
Handle device recieved notifications
decide what to do when user taps notification
data:image/s3,"s3://crabby-images/61390/61390fa11559c2911c3b0c00d6b1aba048d98f0c" alt=""
How do we handle push notifications in our app?
using expo notifications
data:image/s3,"s3://crabby-images/7ec02/7ec02a5bb3ef1e4b8adc75cab1d9145eadeb50c5" alt=""
How do we get the push token?
use Permissions to ask permission
use Notifications to getExpoPushTokenAsync( )
useEffect ( ) hook
call a function
Permissions.askAsync
await Notifications.getExpoPushTokenAsync()
user will get same push token
unless they re-install the app
data:image/s3,"s3://crabby-images/8ca97/8ca97415ddbcd158191fa22b94a8513b4d8feffc" alt=""
How do we send a notification token to the server?
data:image/s3,"s3://crabby-images/86a4a/86a4a5e3fde4d7c211bdf2612c5726171c2c98c1" alt=""
AppNavigator component
ask permissions for push notifications
generate a push notification token
send it to the server using expoPushTokensApi
an abstraction over the client (axios)
data:image/s3,"s3://crabby-images/dc168/dc16868868dab8673f007b4948773ce360660f2c" alt=""
How do we send test notifications in Expo during development?
data:image/s3,"s3://crabby-images/0f6ad/0f6adc83eb0d28ea4420b9785a6b5f53de3955fd" alt=""
How does the backend send push notifications using expo?
data:image/s3,"s3://crabby-images/57365/573657dd53ec1aaa82f3eebfd3972aca8f73926e" alt=""
How can we handle push notifications in our app?
data:image/s3,"s3://crabby-images/fc661/fc661cf080ab73780c272cb9cbb669132b6bc86f" alt=""
addNotificationRecievedListener
data:image/s3,"s3://crabby-images/a34f8/a34f80532979199bef299c3af969b91028192067" alt=""
How can we navigate a user to a specific page when the notification is recieved?
data:image/s3,"s3://crabby-images/b0135/b0135056c16562142eb8b595ac9885eba8849e06" alt=""
Can’t?
use navigation prop
useNavigation hook
Why?
Only available in screens wrapped in Navigator
Soln?
create a reference object (ref to Navigation)
from that reference, can access navigation object
access ref property ‘current’
get access to navigation object
rootNavigation.js file
ref object called when Navigation container created
check if .current is null (will throw error if called too early)
data:image/s3,"s3://crabby-images/3c303/3c303b660ab806ad5a336a772ef6374edde2d08f" alt=""
What is the challenge with this code?
data:image/s3,"s3://crabby-images/0ad03/0ad03f81af8ff0b50330a5082f6e576e001c6fba" alt=""
the notifications logic:
getting users permission
generating a pushToken
handling a notification response
is polluting our AppNavigator component
UI components should only have logic for presentation
Soln?
extract logic into a custom hook (re-usable)
data:image/s3,"s3://crabby-images/c00f7/c00f7a1205d6233bad22d1151398fdc954799eb1" alt=""
How can we make this code re-usable?
data:image/s3,"s3://crabby-images/5f0f8/5f0f8cd8fcb5d92377850525cb436c789d37aeed" alt=""
we don’t want to hardcode the action everytime a listener is added
data:image/s3,"s3://crabby-images/36307/363078e1e7df2aae4923c61964336003909ef9ae" alt=""
pass action down to component via props
What are the two types of notifications?
Remote (push)
sent by expo backend
Local
created locally in the app
data:image/s3,"s3://crabby-images/29595/29595646fd9495cfa9923866a7ea764c3002256e" alt=""
How can we schedule a local notification for the future?
Ex. 10 minutes after reached cart page?
data:image/s3,"s3://crabby-images/646aa/646aa077b1808c58873dfdf3dae0e1388a9f2015" alt=""
How do we send a message to the seller and display a notification for the user?
data:image/s3,"s3://crabby-images/7dc8e/7dc8e0d44fd0adfb8731fef2c5c4fc3ea7fe0f84" alt=""
create a new form component
import Form (Formik abstraction)
FormField (error handling, etc.)
handleSubmit ( )
send message to server with messagesApi (client abstraction)
use Expo notifications to send user notification
data:image/s3,"s3://crabby-images/718cf/718cf829cf1d20ea2397b62dbcba60f5f568757a" alt=""
What will we learn in the final section of this course?
how to submit app to iOS store
GooglePlay store should be similar
Have to give app proper Icon
data:image/s3,"s3://crabby-images/b0164/b016456e3ef70e34624b56cd840b0c0d0ffc40a9" alt=""
How do we create a proper app icon for distributing our app?
data:image/s3,"s3://crabby-images/500ba/500baca54be666067bcb810d38de340b4a2a2baf" alt=""
https://developer.apple.com/design/human-interface-guidelines/ios/icons-and-images/app-icon/
https://developer.android.com/guide/practices/ui_guidelines/icon_design_adaptive
How do we optimize the assets in our project?
Delete non essential assets
Why?
Bundled in app during production
only make our app larger
Soln?
include only absolute essential assets
assets that must be immediately available when starting our app
anything not essential, downloaded later from internet
data:image/s3,"s3://crabby-images/b17d1/b17d1a5d7fbee6d61f7bc05ff9c17fb7ef6da1f7" alt=""
How else can we optimize our assets before building the app for production?
sudo npm i -g sharp-cli
Why?
reduce the size of our images
How?
npx expo-optimize
Why?
uses sharp cli behind scenes
reduces images without losing quality
data:image/s3,"s3://crabby-images/03444/0344449bd4ee32c067e4b20555537c2399cc838b" alt=""
What happens when we build our app?
all our javascript code gets combined, minified into a bundle
app loads, bundle is loaded into memory
Javascript virtual machine on mobile, parses code
Why?
More code in the bundle, more time to load and parse
data:image/s3,"s3://crabby-images/47243/472430c0c6b98174b1d641d947daeaa645eed32c" alt=""
As a best practice, what should we do?
Always check:
is there a more lightweight version of a package used?
Ex. Moment.js
bringing a lot of code we don’t need
in app, only using once, using a small functionality
moment has a lot of functionality, not used
Soln?
Replace with a more lightweight library
npm i dayjs
data:image/s3,"s3://crabby-images/c0f93/c0f939cb1954e7f2a48762dbebfb20179776c0cb" alt=""
How can we see the cost of our import statements?
data:image/s3,"s3://crabby-images/89018/89018851948b7411cc6d6e4f532c3a74baaa6f82" alt=""
cost of moment.js is 258k (71kb compressed)
data:image/s3,"s3://crabby-images/b12a3/b12a3f4630b348c7387d60c7d546b1631d34c1e4" alt=""
vs.
7.2k kb for dayjs (compressed)
What else can effect our bundle size?
How we import things from a library
Why?
can import needed code vs. all library code
How?
some libraries are split into small packages
data:image/s3,"s3://crabby-images/b7d47/b7d47724e0db85c1e271d3000d390499a289b723" alt=""
How can we log errors when our app is in the app store?
data:image/s3,"s3://crabby-images/411be/411bee902e705d68ef905dc03e15da37390ad287" alt=""
use error reporting tools
Talked about sentry in React course
LOVES bugsnap (new tool)
a bit pricey
report handle / unhandled
identify users who encountered errors
data:image/s3,"s3://crabby-images/398ea/398ea0476ccdc0f7ae58d0f5bc0d280901adbe12" alt=""
What is bugsnag?
data:image/s3,"s3://crabby-images/caf27/caf27dd991a858b3a9f6fda1c9640ebbd466db92" alt=""
an error reporting tool
Mosh loves it!
Lift, Square, Airbnb, Mailchimp
How?
upload sourcemaps allows us to see stack trace in production
data:image/s3,"s3://crabby-images/ddd0f/ddd0f98db2a24c2a88039531c49c06f472209287" alt=""
How do we report errors in our production app?
data:image/s3,"s3://crabby-images/25eea/25eea0f16662f476d9650340d7d86eb57bddad31" alt=""
use bugsnag
create logger.js (in utility) folder
can see error rate over time (very powerful)
can see device iOS and version of React (very powerful)
data:image/s3,"s3://crabby-images/c81a2/c81a2925fdc5f9dcc897d17c84acdb105481af53" alt=""
How can we manage the settings for different environments?
data:image/s3,"s3://crabby-images/b95a7/b95a711e5965ab4aff1aba1a6e49e2cdf1711d53" alt=""
Why?
baseUrl is for production backend
in production app, want to use backend hosted in cloud
How?
settings.js (config module)
define settings for different environments
data:image/s3,"s3://crabby-images/831c2/831c22c2a6cd9fd885ea76352b1c77d4dd6ad981" alt=""
How do we publish to expo?
data:image/s3,"s3://crabby-images/954fc/954fc4eb3ee267027658a693d805daadefbfe812" alt=""
publish on Expo
expo publish –release-channel staging
build standalone binaries for iOS and Android
data:image/s3,"s3://crabby-images/99d69/99d69f7e951c184582ef10e45db8a2c802d0fd10" alt=""
How can we build standalone binaries for deployment to app stores (Google and Apple)?
Expo publishing only used for dev, testing
Why?
don’t want users to have to download expo app to use our app
soln?
Expo’s build service
Mosh absolutely loves this service
it’s exceptional
How can we build the iOS standalone app?
data:image/s3,"s3://crabby-images/fc84e/fc84ec43dd664b35323bdb86594cf71fae0f62ad" alt=""
Step 1: register in Apple developer program
developer.apple.com/programs/enroll
must provide driver’s license
must pay $99 / year
to publish to app store, need current payment status
Step 2: add fields in app.json (app store uses)
bundleIdentifier
buildNumber
Step 3: expo build:ios
(terminal)
appleID, password
create distribution service
apple push notifications key
provisioning profile
Step 4: Download ipa file (monitor status)
for uploading in app store
How do we submit our ipa file to the app store?
Hint: file containing our app from expo build:ios
Step 1: appstoreconnect.apple/com
login with appleID
go to “my apps”
create new app
Step 2: install Transporter (from apple app store)
for uploading app to appstore
deliver it to app store
can invite users by email to test app, testers
must download testflight, then download app on phone
Step 3: prepare for submission
provide screen shots, url, etc.
takes 1 day or 2 or make changes
data:image/s3,"s3://crabby-images/9448f/9448f7de78e14a733f6c9cf020adfae6cbb38c14" alt=""
What is over the air updates?
data:image/s3,"s3://crabby-images/5afc5/5afc5abb3ca59840f20f096affbccd845f8ce4bb" alt=""
don’t have to re-submit to app store
App will check expo and automatically download updates
data:image/s3,"s3://crabby-images/45ad9/45ad934ce1d6072360e224624591c10d06a895fd" alt=""
What are the next steps?
think of next app idea
apply for a job as a React native developer
should have very strong foundation
be able to discover new things
if get a job or build an app, share a tweet with him
he wants to know he made an impact on our life
Salary:
Average $120,000 usd / year
entry level positions start at $80,000 usd /year
experienced $187,000 usd /year
data:image/s3,"s3://crabby-images/66069/6606912d2533272a5716d7ef0b4fe0fd14024008" alt=""