We take developers from good to great
data:image/s3,"s3://crabby-images/70153/701535c334830d4339ec09b634f8b1b4c508b7ff" alt=""
WE ARE THE JSLEAGUE
data:image/s3,"s3://crabby-images/0ce61/0ce61fec9ee22a74d41d48a4ecf9a42519c51839" alt=""
One app, 3 Frameworks
30th of May
data:image/s3,"s3://crabby-images/012be/012be731af7ea46fbb18028e22de7ef7065705de" alt=""
bit.ly/voxxed-react
data:image/s3,"s3://crabby-images/5c302/5c302580185e5c7b2b9c033b47388720a45d2a9f" alt=""
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
React
Module 1
Fundamentals
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
Module 1
But first, a few considerations...
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
Module 1
Templating and state management
data:image/s3,"s3://crabby-images/20b61/20b616b5ce55642baef863a982675b4c65f3735c" alt=""
data:image/s3,"s3://crabby-images/52fd1/52fd1a2ac981acde9b0c2d056405bf4bc6d87fc5" alt=""
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
Module 1
Templating and state management
data:image/s3,"s3://crabby-images/643cf/643cf4b2228605eb98754525d19f89ed3b5bdca0" alt=""
Cannot stress this enough.
Everything is JS
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
Module 1
First stop: Components
Two types of components:
- "Class Components"
- "Functional Components"
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
Module 1
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
Module 1
Dumb Component
import React from 'react';
export default ({ counter, setCounter }) => (
<div>
<span>Count: {counter}</span>
<button onClick={() => setCounter(counter + 1)}>+1</button>
</div>
);
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
Module 1
Dumb Component
export default function() {
var _component = function() {
var _this;
for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {
args[_key] = arguments[_key];
}
return _this;
}
Object(babel_runtime_helpers.createClass(_component, [{
key: "render",
value: function render() {
var counter = this.props.counter;
var setCounter = this.props.setCounter;
return React.createElement(React.div, {
children: React.createElement(...),
});
}
}]);
return _component;
}
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
Module 1
Stateful Component
import React, { Component } from 'react';
import CounterComponent from './counter';
export default class MyComponent extends Component {
state = { counter: 0 }
updateCounter = value => this.setState({ counter: value });
render() {
const { counter } = this.state;
return (
<CounterComponent
counter={counter}
setCounter={this.updateCounter}
/>
);
}
}
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
Module 1
Stateful Component (don't)
import React, { Component } from 'react';
import CounterComponent from './counter';
export default class MyComponent extends Component {
constructor() {
super();
this.state = { counter: 0 };
this.updateCounter = this.updateCounter.bind(this);
}
updateCounter(value) { this.setState({ counter: value }); }
render() {
const { counter } = this.state;
return (
<CounterComponent
counter={counter}
setCounter={this.updateCounter}
/>
);
}
}
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
Module 1
State?
class MyComponent extends Component {
...
render() {
const { isEditing } = this.state;
const { data, updateData } = this.props;
...
}
}
class RootComponent extends Component {
...
render() {
return <MyComponent data={...} updateData={...} />
}
}
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
Module 1
Lifecycle
class MyComponent extends Component {
// on initial render
componentWillMount() //deprecated
componentDidMount()
// props changes
componentWillReceiveProps()
componentShouldUpdate()
componentWillUpdate()
// state changes
componentWillUpdate()
// component is no longer being rendered
componentWillUnmount()
...
}
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
Module 1
React Hooks
export default (props) => {
const [counter, setCounter] = useState(0);
useEffect(
() => document.head.title = "Ready to count",
[],
);
useEffect(
() => document.head.title = `Counted ${counter} times`,
[counter],
);
return (
<div>
<h2>Counter App</h2>
<div>Counted {counter} times!</div>
<button onClick={() => setCounter(counter + 1)}>Add one</button>
<button onClick={() => setCounter(counter + 5)}>Add five</button>
<button onClick={() => setCounter(counter - 1)}>Subtract one</button>
</div>
);
}
data:image/s3,"s3://crabby-images/b2ae6/b2ae68d14c3e95d5b3b594ad4ee76d778f655d0f" alt=""
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
React
Module 2
State / Data
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
Module 2
Two types of Data
- Properties (props)
- Component State (state)
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
Module 2
Class Component
import React, { Component } from 'react';
import CounterComponent from './counter';
export default class MyComponent extends Component {
state = { counter: 0 }
updateCounter = value => this.setState({ counter: value });
render() {
const { counter } = this.state;
const { firstName, lastName } = this.props;
return (
<CounterComponent
counter={counter}
name={`${firstName} ${lastName}`}
setCounter={this.updateCounter}
/>
);
}
}
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
Module 2
React Hooks
export default ({ name }) => {
const [counter, setCounter] = useState(0);
useEffect(
() => document.head.title = "Ready to count",
[],
);
useEffect(
() => document.head.title = `Counted ${counter} times`,
[counter],
);
return (
<div>
<h2>Counter App</h2>
<h1>Hi, my name is {name}</h1>
<div>Counted {counter} times!</div>
<button onClick={() => setCounter(counter + 1)}>Add one</button>
<button onClick={() => setCounter(counter + 5)}>Add five</button>
<button onClick={() => setCounter(counter - 1)}>Subtract one</button>
</div>
);
}
data:image/s3,"s3://crabby-images/b2ae6/b2ae68d14c3e95d5b3b594ad4ee76d778f655d0f" alt=""
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
React
Module 3
Global State
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
Module 3
data:image/s3,"s3://crabby-images/427c9/427c9b5afdda8b51d97d17ae799b9ccde1e38707" alt=""
Redux
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
Module 3
Mobx
data:image/s3,"s3://crabby-images/6535c/6535cc26c4487d866e185214746d9052fbb4417b" alt=""
Why?
Redux Example: bit.ly/voxxed-redux
Mobx Example: bit.ly/voxxed-mobx
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
Module 3
data:image/s3,"s3://crabby-images/b2ae6/b2ae68d14c3e95d5b3b594ad4ee76d778f655d0f" alt=""
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
React
Module 4
HTTP / Routing
HTTP?
Anything you like.
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
Module 4
Routing
Conditional.
Not Structural
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
Module 4
import { BrowserRouter as Router, Route, Switch, Link } from 'react-router-dom';
export default ({ isAdmin }) => (
<Router>
<Switch>
<Route exact path="/" component={HomeComponent} />
<Route path="/users" component={UsersLayout} />
</Switch>
<footer>
<Route
exact
path="/admin"
render={() => <Link to="/">Home</Link>}
/>
{isAdmin &&
<Route
path="/not-an-admin"
render={() => <Link to="/admin">Admin</Link>}
/>
}
</footer>
</Router>
);
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
Module 4
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
React
Module 5
Extras
Suspense
Lazy loading
+ Code Splitting
+ Fallback rendering
+ Error Boundary Handling
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
Module 5
import { BrowserRouter as Router, Route, Switch } from 'react-router-dom';
import React, { Suspense, lazy } from 'react';
const Home = lazy(() => import('./routes/Home'));
const About = lazy(() => import('./routes/About'));
const App = () => (
<Router>
<Suspense fallback={<div>Loading...</div>}>
<Switch>
<Route exact path="/" component={Home}/>
<Route path="/about" component={About}/>
</Switch>
</Suspense>
</Router>
);
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
Module 5
Thank you!
data:image/s3,"s3://crabby-images/2d0b6/2d0b602eb42d388bbe088e006864265678469402" alt=""
One App, 3 Frameworks -- React
By Sabin Marcu
One App, 3 Frameworks -- React
- 496