React Notes
Kent C. Dodds | Notes from JS to know for React blogs
One of the things I love most about React compared to other frameworks that I've used is how exposed you are to JavaScript when you're using it. There's no template DSL (JSX compiles to sensible JavaScript), the component API has only gotten simpler with the addition of React Hooks, and the framework offers you very little abstraction outside the core UI concerns it's intended to solve.
Because of this, learning JavaScript features is really advisable for you to be effective building applications with React. So here are a few JavaScript features I'd recommend you spend some time learning so you can be as effective as possible working with React.
Before we get into some syntax stuff, another thing that's really useful to understand for React is the concept of a function "closure". There's a great write-up of this concept here: mdn.io/closure.
Ok, let's get to the JS features you'll want to know for React.
Template Literals
Template literals are like regular strings with super-powers:
const greeting = "Hello";
const subject = "World";
console.log(`${greeting} ${subject}!`); // Hello World!
// this is the same as:
console.log(greeting + " " + subject + "!");
// in React:
function Box({ className, ...props }) {
return <div className={`box ${className}`} {...props} />;
}
Shorthand property names
This is so common and useful that I do this without thinking now.
const a = "hello";
const b = 42;
const c = { d: [true, false] };
console.log({ a, b, c });
// this is the same as:
console.log({ a: a, b: b, c: c });
// in React:
function Counter({ initialCount, step }) {
const [count, setCount] = useCounter({ initialCount, step });
return <button onClick={setCount}>{count}</button>;
}
MDN: Object initializer New notations in ECMAScript 2015
Arrow functions
Arrow functions are another way to write functions in JavaScript, but they do
have a few semantic differences. Luckily for us in React land, we don't have to
worry about this
as much if we're using hooks in our project (rather than
classes), but the arrow function allows for terser anonymous functions and
implicit returns, so you'll see and want to use arrow functions plenty.
const getFive = () => 5;
const addFive = (a) => a + 5;
const divide = (a, b) => a / b;
// this is the same as:
function getFive() {
return 5;
}
function addFive(a) {
return a + 5;
}
function divide(a, b) {
return a / b;
}
// in React:
function TeddyBearList({ teddyBears }) {
return (
<ul>
{teddyBears.map((teddyBear) => (
<li key={teddyBear.id}>
<span>{teddyBear.name}</span>
</li>
))}
</ul>
);
}
Destructuring
Destructuring is probably my favorite JavaScript feature. I destructure objects and arrays all the time (and if you're using useState
you probably are too. I love how declarative it is.
// const obj = {x: 3.6, y: 7.8}
// makeCalculation(obj)
function makeCalculation({ x, y: d, z = 4 }) {
return Math.floor((x + d + z) / 3);
}
// this is the same as
function makeCalculation(obj) {
const { x, y: d, z = 4 } = obj;
return Math.floor((x + d + z) / 3);
}
// which is the same as
function makeCalculation(obj) {
const x = obj.x;
const d = obj.y;
const z = obj.z === undefined ? 4 : obj.z;
return Math.floor((x + d + z) / 3);
}
// in React:
function UserGitHubImg({ username = "ghost", ...props }) {
return <img src={`https://github.com/${username}.png`} {...props} />;
}
function nestedArrayAndObject() {
// refactor this to a single line of destructuring...
const info = {
title: "Once Upon a Time",
protagonist: {
name: "Emma Swan",
enemies: [
{ name: "Regina Mills", title: "Evil Queen" },
{ name: "Cora Mills", title: "Queen of Hearts" },
{ name: "Peter Pan", title: `The boy who wouldn't grow up` },
{ name: "Zelena", title: "The Wicked Witch" },
],
},
};
// const {} = info // <-- replace the next few `const` lines with this
const title = info.title;
const protagonistName = info.protagonist.name;
const enemy = info.protagonist.enemies[3];
const enemyTitle = enemy.title;
const enemyName = enemy.name;
return `${enemyName} (${enemyTitle}) is an enemy to ${protagonistName} in "${title}"`;
}
Parameter defaults
This is another feature I use all the time. It's a really powerful way to declaratively express default values for your functions.
// add(1)
// add(1, 2)
function add(a, b = 0) {
return a + b;
}
// is the same as
const add = (a, b = 0) => a + b;
// is the same as
function add(a, b) {
b = b === undefined ? 0 : b;
return a + b;
}
// in React:
function useLocalStorageState({
key,
initialValue,
serialize = (v) => v,
deserialize = (v) => v,
}) {
const [state, setState] = React.useState(
() => deserialize(window.localStorage.getItem(key)) || initialValue
);
const serializedState = serialize(state);
React.useEffect(() => {
window.localStorage.setItem(key, serializedState);
}, [key, serializedState]);
return [state, setState];
}
Rest/Spread
The ...
syntax can be thought of as kind of a "collection" syntax where it
operates on a collection of values. I use it all the time and strongly recommend
you learn how and where it can be used as well. It actually takes different
meanings in different contexts, so learning the nuances there will help you.
const arr = [5, 6, 8, 4, 9];
Math.max(...arr);
// is the same as
Math.max.apply(null, arr);
const obj1 = {
a: "a from obj1",
b: "b from obj1",
c: "c from obj1",
d: {
e: "e from obj1",
f: "f from obj1",
},
};
const obj2 = {
b: "b from obj2",
c: "c from obj2",
d: {
g: "g from obj2",
h: "h from obj2",
},
};
console.log({ ...obj1, ...obj2 });
// is the same as
console.log(Object.assign({}, obj1, obj2));
function add(first, ...rest) {
return rest.reduce((sum, next) => sum + next, first);
}
// is the same as
function add() {
const first = arguments[0];
const rest = Array.from(arguments).slice(1);
return rest.reduce((sum, next) => sum + next, first);
}
// in React:
function Box({ className, ...restOfTheProps }) {
const defaultProps = {
className: `box ${className}`,
children: "Empty box",
};
return <div {...defaultProps} {...restOfTheProps} />;
}
ESModules
If you're building an app with modern tools, chances are it supports modules, it's a good idea to learn how the syntax works because any application of even trivial size will likely need to make use of modules for code reuse and organization.
export default function add(a, b) {
return a + b;
}
/*
* import add from './add'
* console.assert(add(3, 2) === 5)
*/
export const foo = "bar";
/*
* import {foo} from './foo'
* console.assert(foo === 'bar')
*/
export function subtract(a, b) {
return a - b;
}
export const now = new Date();
/*
* import {subtract, now} from './stuff'
* console.assert(subtract(4, 2) === 2)
* console.assert(now instanceof Date)
*/
// dynamic imports
import("./some-module").then(
(allModuleExports) => {
// the allModuleExports object will be the same object you get if you had
// used: import * as allModuleExports from './some-module'
// the only difference is this will be loaded asynchronously which can
// have performance benefits in some cases
},
(error) => {
// handle the error
// this will happen if there's an error loading or running the module
}
);
// in React:
import React, { Suspense, Fragment } from "react";
// dynamic import of a React component
const BigComponent = React.lazy(() => import("./big-component"));
// big-component.js would need to "export default BigComponent" for this to work
Ternaries
I love ternaries. They're beautifully declarative. Especially in JSX.
const message = bottle.fullOfSoda
? "The bottle has soda!"
: "The bottle may not have soda :-(";
// is the same as
let message;
if (bottle.fullOfSoda) {
message = "The bottle has soda!";
} else {
message = "The bottle may not have soda :-(";
}
// in React:
function TeddyBearList({ teddyBears }) {
return (
<React.Fragment>
{teddyBears.length ? (
<ul>
{teddyBears.map((teddyBear) => (
<li key={teddyBear.id}>
<span>{teddyBear.name}</span>
</li>
))}
</ul>
) : (
<div>There are no teddy bears. The sadness.</div>
)}
</React.Fragment>
);
}
MDN: Conditional (ternary) operator
Array Methods
Arrays are fantastic and I use array methods all the time! I probably use the following methods the most frequently:
- find
- some
- every
- includes
- map
- filter
- reduce
Here are some examples:
const dogs = [
{
id: "dog-1",
name: "Poodle",
temperament: [
"Intelligent",
"Active",
"Alert",
"Faithful",
"Trainable",
"Instinctual",
],
},
{
id: "dog-2",
name: "Bernese Mountain Dog",
temperament: ["Affectionate", "Intelligent", "Loyal", "Faithful"],
},
{
id: "dog-3",
name: "Labrador Retriever",
temperament: [
"Intelligent",
"Even Tempered",
"Kind",
"Agile",
"Outgoing",
"Trusting",
"Gentle",
],
},
];
dogs.find((dog) => dog.name === "Bernese Mountain Dog");
// {id: 'dog-2', name: 'Bernese Mountain Dog', ...etc}
dogs.some((dog) => dog.temperament.includes("Aggressive"));
// false
dogs.some((dog) => dog.temperament.includes("Trusting"));
// true
dogs.every((dog) => dog.temperament.includes("Trusting"));
// false
dogs.every((dog) => dog.temperament.includes("Intelligent"));
// true
dogs.map((dog) => dog.name);
// ['Poodle', 'Bernese Mountain Dog', 'Labrador Retriever']
dogs.filter((dog) => dog.temperament.includes("Faithful"));
// [{id: 'dog-1', ..etc}, {id: 'dog-2', ...etc}]
dogs.reduce((allTemperaments, dog) => {
return [...allTemperaments, ...dog.temperament];
}, []);
// [ 'Intelligent', 'Active', 'Alert', ...etc ]
// in React:
function RepositoryList({ repositories, owner }) {
return (
<ul>
{repositories
.filter((repo) => repo.owner === owner)
.map((repo) => (
<li key={repo.id}>{repo.name}</li>
))}
</ul>
);
}
Nullish coalescing operator
If a value is null
or undefined
, then you may want to fallback to some
default value:
// here's what we often did for this:
x = x || "some default";
// but this was problematic for numbers or booleans where "0" or "false" are valid values
// So, if we wanted to support this:
add(null, 3);
// here's what we had to do before:
function add(a, b) {
a = a == null ? 0 : a;
b = b == null ? 0 : b;
return a + b;
}
// here's what we can do now
function add(a, b) {
a = a ?? 0;
b = b ?? 0;
return a + b;
}
// in React:
function DisplayContactName({ contact }) {
return <div>{contact.name ?? "Unknown"}</div>;
}
MDN: Nullish coalescing operator
Optional chaining
Also known as the "Elvis Operator," this allows you to safely access properties and call functions that may or may not exist. Before optional chaining, we used a hacky-workaround that relied on falsy/truthy-ness.
// what we did before optional chaining:
const streetName = user && user.address && user.address.street.name;
// what we can do now:
const streetName = user?.address?.street?.name;
// this will run even if options is undefined (in which case, onSuccess would be undefined as well)
// however, it will still fail if options was never declared,
// since optional chaining cannot be used on a non-existent root object.
// optional chaining does not replace checks like if (typeof options == "undefined")
const onSuccess = options?.onSuccess;
// this will run without error even if onSuccess is undefined (in which case, no function will be called)
onSuccess?.({ data: "yay" });
// and we can combine those things into a single line:
options?.onSuccess?.({ data: "yay" });
// and if you are 100% certain that onSuccess is a function if options exists
// then you don't need the extra ?. before calling it. Only use ?. in situations
// where the thing on the left might not exist.
options?.onSuccess({ data: "yay" });
// in React:
function UserProfile({ user }) {
return (
<div>
<h1>{user.name}</h1>
<strong>{user.bio?.short ?? "No bio provided"}</strong>
</div>
);
}
A word of caution around this is that if you find yourself doing ?.
a lot in
your code, you might want to consider the place where those values originate and
make sure they consistently return the values they should.
Promises and async/await
This one's a big subject and it can take a bit of practice and time working with them to get good at them. Promises are everywhere in the JavaScript ecosystem and thanks to how entrenched React is in that ecosystem, they're everywhere there as well (in fact, React itself uses promises internally).
Promises help you manage asynchronous code and are returned from many DOM APIs as well as third party libraries. Async/await syntax is a special syntax for dealing with promises. The two go hand-in-hand.
function promises() {
const successfulPromise = timeout(100).then((result) => `success: ${result}`);
const failingPromise = timeout(200, true).then(null, (error) =>
Promise.reject(`failure: ${error}`)
);
const recoveredPromise = timeout(300, true).then(null, (error) =>
Promise.resolve(`failed and recovered: ${error}`)
);
successfulPromise.then(log, logError);
failingPromise.then(log, logError);
recoveredPromise.then(log, logError);
}
function asyncAwaits() {
async function successfulAsyncAwait() {
const result = await timeout(100);
return `success: ${result}`;
}
async function failedAsyncAwait() {
const result = await timeout(200, true);
return `failed: ${result}`; // this would not be executed
}
async function recoveredAsyncAwait() {
try {
const result = await timeout(300, true);
return `failed: ${result}`; // this would not be executed
} catch (error) {
return `failed and recovered: ${error}`;
}
}
successfulAsyncAwait().then(log, logError);
failedAsyncAwait().then(log, logError);
recoveredAsyncAwait().then(log, logError);
}
function log(...args) {
console.log(...args);
}
function logError(...args) {
console.error(...args);
}
// This is the mothership of all things asynchronous
function timeout(duration = 0, shouldReject = false) {
return new Promise((resolve, reject) => {
setTimeout(() => {
if (shouldReject) {
reject(`rejected after ${duration}ms`);
} else {
resolve(`resolved after ${duration}ms`);
}
}, duration);
});
}
// in React:
function GetGreetingForSubject({ subject }) {
const [isLoading, setIsLoading] = React.useState(false);
const [error, setError] = React.useState(null);
const [greeting, setGreeting] = React.useState(null);
React.useEffect(() => {
async function fetchGreeting() {
try {
const response = await window.fetch("https://example.com/api/greeting");
const data = await response.json();
setGreeting(data.greeting);
} catch (error) {
setError(error);
} finally {
setIsLoading(false);
}
}
setIsLoading(true);
fetchGreeting();
}, []);
return isLoading ? (
"loading..."
) : error ? (
"ERROR!"
) : greeting ? (
<div>
{greeting} {subject}
</div>
) : null;
}
Conclusion
There are of course many language features which are useful when building React apps, but these are some of my favorites that I find myself using again and again. I hope you find this useful.
If you'd like to dive into any of these further, I do have a JavaScript workshop which I gave and recorded while I was working at PayPal which you may find helpful: ES6 and Beyond Workshop at PayPal
Good luck!
Wes Bos | Advanced React
I'm taking notes on https://github.dev/patrickjohnstevens/docublog/tree/main/PJSDocBlog so I'm coding while I'm taking notes. It's cool because I can use it just like CMS but my DB is GitHub. I've even tried it on my phone and I can also create a file and commit it to GitHub and it auto builds on Vercel. It's a nice workflow if I know I don't need to use the terminal. This might be the future. Plus, I am already using my VSCode Desktop App for the course I'm watching.
The primary tech stack for this project/course is React(UI Templates)/Next.js(framework)/Apollo Client(GraphQL)/Styled Components(Modular CSS) for the frontend and Keystone.js(CMS)/node.js(Backend Server Runtime Env)/MongoDB(Database) for the backend.
Keeping the Markdown features on Docusaurous link handy: https://docusaurus.io/docs/markdown-features#quotes
Idea
I wonder how easy it would be for me to not use Keystone.js and instead us Headless Wordpress? Why do this? Simple, I will be able to learn more about Headless Wordpress and be able to give a talk about it at the Omaha Wordpress meetup.
Next.js rendering
- Server
- Client
- Static - Prerendering. Docusaurous does a lot of this kind of thing.
This is HTML and it should render like the rest of the MDX/MD text because that gets converted to HTML anyways.
Random Notes, Thoughts, and Ideas
- Use React Suspense to load the data for the page. This will allow the page to load faster and then the data will load in after the page is loaded. This is a good way to improve the user experience. The technique is simple, you wrap your component under Suspense. The Suspense accepts a fallback component to display until the component data is fetched. The fallback component is rendered as long as the Promise is not resolved.