forked from vercel/next.js
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Refactor with redux observable (vercel#13615)
Related to [11014](vercel#11014) 1. Moved the reducer into the store and created new store file 2. The example was using a server that was no longer available, now it uses JSON placeholder instead. 3. Moved from getInitialProps to getStaticProps 4. Refactored all the classes to functional components, using the new redux hooks API. 5. Upgraded all the packages and using custom redux wrapper instead of next-redux-wrapper, which I have removed from the example. 6. Upgraded all the other packages. Please, let me know if I should change anything.
- Loading branch information
1 parent
79b7d8b
commit f18a5a8
Showing
15 changed files
with
201 additions
and
225 deletions.
There are no files selected for viewing
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
44 changes: 0 additions & 44 deletions
44
examples/with-redux-observable/components/CharacterInfo.js
This file was deleted.
Oops, something went wrong.
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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,50 @@ | ||
import { useSelector } from 'react-redux' | ||
|
||
const useUser = () => { | ||
return useSelector((state) => ({ | ||
character: state.character, | ||
error: state.error, | ||
isFetchedOnServer: state.isFetchedOnServer, | ||
})) | ||
} | ||
|
||
const UserInfo = () => { | ||
const { character, isFetchedOnServer, error } = useUser() | ||
const { name, id, username, email, phone, website } = character | ||
|
||
return ( | ||
<div className="UserInfo"> | ||
{error ? ( | ||
<p>We encountered and error.</p> | ||
) : ( | ||
<article> | ||
<h3>Name: {name}</h3> | ||
<p>Id: {id}</p> | ||
<p>Username: {username}</p> | ||
<p>Email: {email}</p> | ||
<p>Phone: {phone}</p> | ||
<p>Website: {website}</p> | ||
</article> | ||
)} | ||
<p> | ||
(was user fetched on server? - <b>{isFetchedOnServer.toString()})</b> | ||
</p> | ||
<p> Please note there are no more than 10 users in the API!</p> | ||
<style jsx>{` | ||
article { | ||
background-color: #528ce0; | ||
border-radius: 15px; | ||
padding: 15px; | ||
width: 250px; | ||
margin: 15px 0; | ||
color: white; | ||
} | ||
button { | ||
margin-right: 10px; | ||
} | ||
`}</style> | ||
</div> | ||
) | ||
} | ||
|
||
export default UserInfo |
Binary file not shown.
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
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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,25 +1,12 @@ | ||
import { Provider } from 'react-redux' | ||
import App from 'next/app' | ||
import withRedux from 'next-redux-wrapper' | ||
import makeStore from '../redux' | ||
import { useStore } from '../store/store' | ||
|
||
class MyApp extends App { | ||
static async getInitialProps({ Component, ctx }) { | ||
const pageProps = Component.getInitialProps | ||
? await Component.getInitialProps(ctx) | ||
: {} | ||
export default function App({ Component, pageProps }) { | ||
const store = useStore(pageProps.initialReduxState) | ||
|
||
return { pageProps } | ||
} | ||
|
||
render() { | ||
const { Component, pageProps, store } = this.props | ||
return ( | ||
<Provider store={store}> | ||
<Component {...pageProps} /> | ||
</Provider> | ||
) | ||
} | ||
return ( | ||
<Provider store={store}> | ||
<Component {...pageProps} /> | ||
</Provider> | ||
) | ||
} | ||
|
||
export default withRedux(makeStore)(MyApp) |
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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,49 +1,30 @@ | ||
import { Component } from 'react' | ||
import { useEffect } from 'react' | ||
import { useDispatch } from 'react-redux' | ||
import Link from 'next/link' | ||
import { of, Subject } from 'rxjs' | ||
import { StateObservable } from 'redux-observable' | ||
import { connect } from 'react-redux' | ||
import CharacterInfo from '../components/CharacterInfo' | ||
import { rootEpic } from '../redux/epics' | ||
import * as actions from '../redux/actions' | ||
import UserInfo from '../components/UserInfo' | ||
import { stopFetchingUsers, startFetchingUsers } from '../store/actions' | ||
|
||
class Counter extends Component { | ||
static async getInitialProps({ store, isServer }) { | ||
const state$ = new StateObservable(new Subject(), store.getState()) | ||
const resultAction = await rootEpic( | ||
of(actions.fetchCharacter(isServer)), | ||
state$ | ||
).toPromise() // we need to convert Observable to Promise | ||
store.dispatch(resultAction) | ||
const Counter = () => { | ||
const dispatch = useDispatch() | ||
|
||
return { isServer } | ||
} | ||
useEffect(() => { | ||
dispatch(startFetchingUsers()) | ||
return () => { | ||
dispatch(stopFetchingUsers()) | ||
} | ||
}, [dispatch]) | ||
|
||
componentDidMount() { | ||
this.props.startFetchingCharacters() | ||
} | ||
|
||
componentWillUnmount() { | ||
this.props.stopFetchingCharacters() | ||
} | ||
|
||
render() { | ||
return ( | ||
<div> | ||
<h1>Index Page</h1> | ||
<CharacterInfo /> | ||
<br /> | ||
<nav> | ||
<Link href="/other"> | ||
<a>Navigate to "/other"</a> | ||
</Link> | ||
</nav> | ||
</div> | ||
) | ||
} | ||
return ( | ||
<div> | ||
<h1>Index Page</h1> | ||
<UserInfo /> | ||
<br /> | ||
<nav> | ||
<Link href="/other"> | ||
<a>Navigate to "/other"</a> | ||
</Link> | ||
</nav> | ||
</div> | ||
) | ||
} | ||
|
||
export default connect(null, { | ||
startFetchingCharacters: actions.startFetchingCharacters, | ||
stopFetchingCharacters: actions.stopFetchingCharacters, | ||
})(Counter) | ||
export default Counter |
This file was deleted.
Oops, something went wrong.
This file was deleted.
Oops, something went wrong.
This file was deleted.
Oops, something went wrong.
This file was deleted.
Oops, something went wrong.
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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,5 @@ | ||
export const FETCH_USER = 'FETCH_USER' | ||
export const FETCH_USER_SUCCESS = 'FETCH_USER_SUCCESS' | ||
export const FETCH_USER_FAILURE = 'FETCH_USER_FAILURE' | ||
export const START_FETCHING_USERS = 'START_FETCHING_USERS' | ||
export const STOP_FETCHING_USERS = 'STOP_FETCHING_USERS' |
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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,21 @@ | ||
import * as types from './actionTypes' | ||
|
||
export const startFetchingUsers = () => ({ | ||
type: types.START_FETCHING_USERS, | ||
}) | ||
export const stopFetchingUsers = () => ({ | ||
type: types.STOP_FETCHING_USERS, | ||
}) | ||
export const fetchUser = (isServer = false) => ({ | ||
type: types.FETCH_USER, | ||
payload: { isServer }, | ||
}) | ||
export const fetchUserSuccess = (response, isServer) => ({ | ||
type: types.FETCH_USER_SUCCESS, | ||
payload: { response, isServer }, | ||
}) | ||
|
||
export const fetchUserFailure = (error, isServer) => ({ | ||
type: types.FETCH_USER_FAILURE, | ||
payload: { error, isServer }, | ||
}) |
Oops, something went wrong.