-
-
Notifications
You must be signed in to change notification settings - Fork 44
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Setup foundation of centralizing user state #142
Merged
thadk
merged 17 commits into
codeforboston:develop
from
paavanb:paavan/user-state-context
Jan 15, 2020
Merged
Setup foundation of centralizing user state #142
thadk
merged 17 commits into
codeforboston:develop
from
paavanb:paavan/user-state-context
Jan 15, 2020
Conversation
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
12 tasks
thadk
approved these changes
Jan 15, 2020
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looks great! Thanks for explaining.
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
This PR sets up the root components necessary to help us centralize the management and persistence of user state.
Description
I used a context-based state-reducer approach to give descendent components access to two values:
UserState
andUserStateActions
.UserState
contains the current state of data related to the user, whileUserStateActions
is an object consisting of functions that allow components to modify the user state (e.g.actions.setBirthDate(value)
). Both of these are typed so that we don't make any mistakes while getting or setting these values.Then, I created a component called
UserStateManager
which is like a "double context provider". It's the single component responsible for managing/persisting the state and defining the actions, which are then provided via the two context providersUserStateContext
andUserStateActionsContext
. It sits at the top of the component tree insideLayout
. So far all it does is manageBirthDate
andRetireDate
.Usage
To use it, check out
Prescreen1a
, which I partially migrated for managingBirthDate
andRetireDate
. All I needed to do was create a wrapper component that uses theuseUserActions
anduseUserState
hooks to pass the two values as props into the main component. The main component never has to worry about managing internal state anymore, and can become more or less a controlled component.Computed Fields
The nice thing is that we can also define derived fields trivially. For example, I noticed that
RetireDate
is calculated usingBirthDate
, and we don't actually need to persist it to storage separately. Once all components are usingUserState
, we can just get rid ofsetRetireDate
and haveretireDate
defined off ofbirthDate
. Then, it will automatically update for all components any timebirthDate
changes!Migrating
When we want to add new fields to the user state, all we have to do is:
UserState
type definition. That will flag Typescript errors in theuser-state-manager
file, so we'll know where we need to add those fieldsUserStateActions
definition. That will also flag Typescript errors so we add the right function definitions in the manager fileAnd if we want to convert a new component to use the state/actions:
userState
anduserStateActions
as props to the main component, and export it from the file instead.Persistence
In order to handle persisting to storage, I brought in the super-useful
use-persisted-state
library, which allows us to completely black-box the persistence behavior and just pretend as if we have a normal state variable like when usinguseState
. We just tell it what key to store the value under, and we're done!Eslint
I also set up a dead-simple eslint configuration because it's kind of necessary to use hooks. All it does it use the
eslint-plugin-react-hooks
plugin in order to warn us if we break the Rules of Hooks or forget a dependency in the dependency array. Later, we can extend well-known eslint configurations (like airbnb's) to more aggressively lint the project.