diff --git a/beta/src/pages/learn/escape-hatches.md b/beta/src/pages/learn/escape-hatches.md
index 66a50027..b6e86e92 100644
--- a/beta/src/pages/learn/escape-hatches.md
+++ b/beta/src/pages/learn/escape-hatches.md
@@ -2,4 +2,861 @@
title: Escape Hatches
---
-This chapter is incomplete and is still being written!
+
+
+Some of your components may need to control and synchronize with systems outside of React. For example, you might need to focus an input using the browser API, play and pause a video player implemented without React, or connect and listen to messages from a remote server. In this chapter, you'll learn the escape hatches that let you "step outside" React and connect to external systems. Most of your application logic and data flow should not rely on these features.
+
+
+
+
+
+* [How to "remember" information without re-rendering](/learn/referencing-values-with-refs)
+* [How to access DOM elements managed by React](/learn/manipulating-the-dom-with-refs)
+* [How to synchronize components with external systems](/learn/synchronizing-with-effects)
+* [How to remove unnecessary Effects from your components](/learn/you-might-not-need-an-effect)
+* [How an Effect's lifecycle is different from a component's](/learn/lifecycle-of-reactive-effects)
+* [How to prevent some values from re-triggering Effects](/learn/separating-events-from-effects)
+* [How to make your Effect re-run less often](/learn/removing-effect-dependencies)
+* [How to share logic between components](/learn/reusing-logic-with-custom-hooks)
+
+
+
+## Referencing values with refs {/*referencing-values-with-refs*/}
+
+When you want a component to "remember" some information, but you don't want that information to [trigger new renders](/learn/render-and-commit), you can use a *ref*:
+
+```js
+const ref = useRef(0);
+```
+
+Like state, refs are retained by React between re-renders. However, setting state re-renders a component. Changing a ref does not! You can access the current value of that ref through the `ref.current` property.
+
+
+
+```js
+import { useRef } from 'react';
+
+export default function Counter() {
+ let ref = useRef(0);
+
+ function handleClick() {
+ ref.current = ref.current + 1;
+ alert('You clicked ' + ref.current + ' times!');
+ }
+
+ return (
+
+ );
+}
+```
+
+
+
+A ref is like a secret pocket of your component that React doesn't track. For example, you can use refs to store [timeout IDs](https://developer.mozilla.org/en-US/docs/Web/API/setTimeout#return_value), [DOM elements](https://developer.mozilla.org/en-US/docs/Web/API/Element), and other objects that don't impact the component's rendering output.
+
+
+
+Read **[Referencing Values with Refs](/learn/referencing-values-with-refs)** to learn how to use refs to remember information.
+
+
+
+## Manipulating the DOM with refs {/*manipulating-the-dom-with-refs*/}
+
+React automatically updates the DOM to match your render output, so your components won't often need to manipulate it. However, sometimes you might need access to the DOM elements managed by React—for example, to focus a node, scroll to it, or measure its size and position. There is no built-in way to do those things in React, so you will need a ref to the DOM node. For example, clicking the button will focus the input using a ref:
+
+
+
+```js
+import { useRef } from 'react';
+
+export default function Form() {
+ const inputRef = useRef(null);
+
+ function handleClick() {
+ inputRef.current.focus();
+ }
+
+ return (
+ <>
+
+
+ >
+ );
+}
+```
+
+
+
+
+
+Read **[Manipulating the DOM with Refs](/learn/manipulating-the-dom-with-refs)** to learn how to access DOM elements managed by React.
+
+
+
+## Synchronizing with Effects {/*synchronizing-with-effects*/}
+
+Some components need to synchronize with external systems. For example, you might want to control a non-React component based on the React state, set up a server connection, or send an analytics log when a component appears on the screen. Unlike event handlers, which let you handle particular events, *Effects* let you run some code after rendering. Use them to synchronize your component with some system outside of React.
+
+Press Play/Pause multiple times and see how the video player stays synchronized to the `isPlaying` prop value:
+
+
+
+```js
+import { useState, useRef, useEffect } from 'react';
+
+function VideoPlayer({ src, isPlaying }) {
+ const ref = useRef(null);
+
+ useEffect(() => {
+ if (isPlaying) {
+ ref.current.play();
+ } else {
+ ref.current.pause();
+ }
+ });
+
+ return ;
+}
+
+export default function App() {
+ const [isPlaying, setIsPlaying] = useState(false);
+ return (
+ <>
+
+
+ >
+ );
+}
+```
+
+```css
+button { display: block; margin-bottom: 20px; }
+video { width: 250px; }
+```
+
+
+
+Many Effects also need to "clean up" after themselves. For example, if your Effect sets up a connection to a chat server, it should return a *cleanup function* that tells React how to disconnect your component from that server:
+
+
+
+```js
+import { useState, useEffect } from 'react';
+import { createConnection } from './chat.js';
+
+export default function ChatRoom() {
+ useEffect(() => {
+ const connection = createConnection();
+ connection.connect();
+ return () => connection.disconnect();
+ }, []);
+ return
Welcome to the chat!
;
+}
+```
+
+```js chat.js
+export function createConnection() {
+ // A real implementation would actually connect to the server
+ return {
+ connect() {
+ console.log('✅ Connecting...');
+ },
+ disconnect() {
+ console.log('❌ Disconnected.');
+ }
+ };
+}
+```
+
+```css
+input { display: block; margin-bottom: 20px; }
+```
+
+
+
+In development, React will immediately run and clean up your Effect one extra time. This is why you see `"✅ Connecting..."` printed twice. This ensures that you don't forget to implement the cleanup function.
+
+
+
+Read **[Synchronizing with Effects](/learn/synchronizing-with-effects)** to learn how to synchronize components with external systems.
+
+
+
+## You Might Not Need An Effect {/*you-might-not-need-an-effect*/}
+
+Effects are an escape hatch from the React paradigm. They let you "step outside" of React and synchronize your components with some external system. If there is no external system involved (for example, if you want to update a component's state when some props or state change), you shouldn't need an Effect. Removing unnecessary Effects will make your code easier to follow, faster to run, and less error-prone.
+
+There are two common cases in which you don't need Effects:
+- **You don't need Effects to transform data for rendering.**
+- **You don't need Effects to handle user events.**
+
+For example, you don't need an Effect to adjust some state based on other state:
+
+```js {5-9}
+function Form() {
+ const [firstName, setFirstName] = useState('Taylor');
+ const [lastName, setLastName] = useState('Swift');
+
+ // 🔴 Avoid: redundant state and unnecessary Effect
+ const [fullName, setFullName] = useState('');
+ useEffect(() => {
+ setFullName(firstName + ' ' + lastName);
+ }, [firstName, lastName]);
+ // ...
+}
+```
+
+Instead, calculate as much as you can while rendering:
+
+```js {4-5}
+function Form() {
+ const [firstName, setFirstName] = useState('Taylor');
+ const [lastName, setLastName] = useState('Swift');
+ // ✅ Good: calculated during rendering
+ const fullName = firstName + ' ' + lastName;
+ // ...
+}
+```
+
+However, you *do* need Effects to synchronize with external systems.
+
+
+
+Read **[You Might Not Need an Effect](/learn/you-might-not-need-an-effect)** to learn how to remove unnecessary Effects.
+
+
+
+## Lifecycle of reactive effects {/*lifecycle-of-reactive-effects*/}
+
+Effects have a different lifecycle from components. Components may mount, update, or unmount. An Effect can only do two things: to start synchronizing something, and later to stop synchronizing it. This cycle can happen multiple times if your Effect depends on props and state that change over time.
+
+This Effect depends on the value of the `roomId` prop. Props are *reactive values,* which means they can change on a re-render. Notice that the Effect *re-synchronizes* (and re-connects to the server) after you update the `roomId`:
+
+
+
+```js
+import { useState, useEffect } from 'react';
+import { createConnection } from './chat.js';
+
+const serverUrl = 'https://localhost:1234';
+
+function ChatRoom({ roomId }) {
+ useEffect(() => {
+ const connection = createConnection(serverUrl, roomId);
+ connection.connect();
+ return () => connection.disconnect();
+ }, [roomId]);
+
+ return
Welcome to the {roomId} room!
;
+}
+
+export default function App() {
+ const [roomId, setRoomId] = useState('general');
+ return (
+ <>
+
+
+
+ >
+ );
+}
+```
+
+```js chat.js
+export function createConnection(serverUrl, roomId) {
+ // A real implementation would actually connect to the server
+ return {
+ connect() {
+ console.log('✅ Connecting to "' + roomId + '" room at ' + serverUrl + '...');
+ },
+ disconnect() {
+ console.log('❌ Disconnected from "' + roomId + '" room at ' + serverUrl);
+ }
+ };
+}
+```
+
+```css
+input { display: block; margin-bottom: 20px; }
+button { margin-left: 10px; }
+```
+
+
+
+React provides a linter rule to check that you've specified your Effect's dependencies correctly. If you forget to specify `roomId` in the list of dependencies in the above example, the linter will find that bug automatically.
+
+
+
+Read **[Lifecycle of Reactive Events](/learn/lifecycle-of-reactive-effects)** to learn how an Effect's lifecycle is different from a component's.
+
+
+
+## Separating events from Effects {/*separating-events-from-effects*/}
+
+Event handlers only re-run when you perform the same interaction again. Unlike event handlers, Effects re-synchronize if some value they read, like a prop or a state variable, is different from what it was on last render. Sometimes, you want a mix of both behaviors: an Effect that re-runs in response to some values but not others.
+
+All code inside Effects is *reactive.* It will run again if some reactive value it reads has changed due to a re-render. For example, this Effect will re-connect to the chat if either `roomId` or `theme` have changed after interaction:
+
+
+
+```json package.json
+{
+ "dependencies": {
+ "react": "latest",
+ "react-dom": "latest",
+ "react-scripts": "latest",
+ "toastify-js": "1.12.0"
+ },
+ "scripts": {
+ "start": "react-scripts start",
+ "build": "react-scripts build",
+ "test": "react-scripts test --env=jsdom",
+ "eject": "react-scripts eject"
+ }
+}
+```
+
+```js
+import { useState, useEffect } from 'react';
+import { createConnection, sendMessage } from './chat.js';
+import { showNotification } from './notifications.js';
+
+const serverUrl = 'https://localhost:1234';
+
+function ChatRoom({ roomId, theme }) {
+ useEffect(() => {
+ const connection = createConnection(serverUrl, roomId);
+ connection.on('connected', () => {
+ showNotification('Connected!', theme);
+ });
+ connection.connect();
+ return () => connection.disconnect();
+ }, [roomId, theme]);
+
+ return
Welcome to the {roomId} room!
+}
+
+export default function App() {
+ const [roomId, setRoomId] = useState('general');
+ const [isDark, setIsDark] = useState(false);
+ return (
+ <>
+
+
+
+
+ >
+ );
+}
+```
+
+```js chat.js
+export function createConnection(serverUrl, roomId) {
+ // A real implementation would actually connect to the server
+ let connectedCallback;
+ let timeout;
+ return {
+ connect() {
+ timeout = setTimeout(() => {
+ if (connectedCallback) {
+ connectedCallback();
+ }
+ }, 100);
+ },
+ on(event, callback) {
+ if (connectedCallback) {
+ throw Error('Cannot add the handler twice.');
+ }
+ if (event !== 'connected') {
+ throw Error('Only "connected" event is supported.');
+ }
+ connectedCallback = callback;
+ },
+ disconnect() {
+ clearTimeout(timeout);
+ }
+ };
+}
+```
+
+```js notifications.js
+import Toastify from 'toastify-js';
+import 'toastify-js/src/toastify.css';
+
+export function showNotification(message, theme) {
+ Toastify({
+ text: message,
+ duration: 2000,
+ gravity: 'top',
+ position: 'right',
+ style: {
+ background: theme === 'dark' ? 'black' : 'white',
+ color: theme === 'dark' ? 'white' : 'black',
+ },
+ }).showToast();
+}
+```
+
+```css
+label { display: block; margin-top: 10px; }
+```
+
+
+
+This is not ideal. You want to re-connect to the chat only if the `roomId` has changed. Switching the `theme` shouldn't re-connect to the chat! Move the code reading `theme` out of your Effect into an *Event function*:
+
+
+
+```json package.json hidden
+{
+ "dependencies": {
+ "react": "latest",
+ "react-dom": "latest",
+ "react-scripts": "latest",
+ "toastify-js": "1.12.0"
+ },
+ "scripts": {
+ "start": "react-scripts start",
+ "build": "react-scripts build",
+ "test": "react-scripts test --env=jsdom",
+ "eject": "react-scripts eject"
+ }
+}
+```
+
+```js
+import { useState, useEffect } from 'react';
+import { useEvent } from './useEvent.js';
+import { createConnection, sendMessage } from './chat.js';
+import { showNotification } from './notifications.js';
+
+const serverUrl = 'https://localhost:1234';
+
+function ChatRoom({ roomId, theme }) {
+ const onConnected = useEvent(() => {
+ showNotification('Connected!', theme);
+ });
+
+ useEffect(() => {
+ const connection = createConnection(serverUrl, roomId);
+ connection.on('connected', () => {
+ onConnected();
+ });
+ connection.connect();
+ return () => connection.disconnect();
+ }, [roomId, onConnected]); // TODO: Linter will allow [roomId] in the future
+
+ return
Welcome to the {roomId} room!
+}
+
+export default function App() {
+ const [roomId, setRoomId] = useState('general');
+ const [isDark, setIsDark] = useState(false);
+ return (
+ <>
+
+
+
+
+ >
+ );
+}
+```
+
+```js chat.js
+export function createConnection(serverUrl, roomId) {
+ // A real implementation would actually connect to the server
+ let connectedCallback;
+ let timeout;
+ return {
+ connect() {
+ timeout = setTimeout(() => {
+ if (connectedCallback) {
+ connectedCallback();
+ }
+ }, 100);
+ },
+ on(event, callback) {
+ if (connectedCallback) {
+ throw Error('Cannot add the handler twice.');
+ }
+ if (event !== 'connected') {
+ throw Error('Only "connected" event is supported.');
+ }
+ connectedCallback = callback;
+ },
+ disconnect() {
+ clearTimeout(timeout);
+ }
+ };
+}
+```
+
+```js notifications.js hidden
+import Toastify from 'toastify-js';
+import 'toastify-js/src/toastify.css';
+
+export function showNotification(message, theme) {
+ Toastify({
+ text: message,
+ duration: 2000,
+ gravity: 'top',
+ position: 'right',
+ style: {
+ background: theme === 'dark' ? 'black' : 'white',
+ color: theme === 'dark' ? 'white' : 'black',
+ },
+ }).showToast();
+}
+```
+
+```js useEvent.js
+import { useRef, useInsertionEffect, useCallback } from 'react';
+
+// The useEvent API has not yet been added to React,
+// so this is a temporary shim to make this sandbox work.
+// You're not expected to write code like this yourself.
+
+export function useEvent(fn) {
+ const ref = useRef(null);
+ useInsertionEffect(() => {
+ ref.current = fn;
+ }, [fn]);
+ return useCallback((...args) => {
+ const f = ref.current;
+ return f(...args);
+ }, []);
+}
+```
+
+```css
+label { display: block; margin-top: 10px; }
+```
+
+
+
+Code inside Event functions isn't reactive, so changing the `theme` no longer makes your Effect re-connect.
+
+
+
+Read **[Separating Events from Effects](/learn/separating-events-from-effects)** to learn how to prevent some values from re-triggering Effects.
+
+
+
+## Removing effect dependencies {/*removing-effect-dependencies*/}
+
+When you write an Effect, the linter will verify that you've included every reactive value (like props and state) that the Effect reads in the list of your Effect's dependencies. This ensures that your Effect remains synchronized with the latest props and state of your component. Unnecessary dependencies may cause your Effect to run too often, or even create an infinite loop. The way you remove them depends on the case.
+
+For example, this Effect depends on the `options` object which gets re-created every time you edit the input:
+
+
+
+```js
+import { useState, useEffect } from 'react';
+import { createConnection } from './chat.js';
+
+const serverUrl = 'https://localhost:1234';
+
+function ChatRoom({ roomId }) {
+ const [message, setMessage] = useState('');
+
+ const options = {
+ serverUrl: serverUrl,
+ roomId: roomId
+ };
+
+ useEffect(() => {
+ const connection = createConnection(options);
+ connection.connect();
+ return () => connection.disconnect();
+ }, [options]);
+
+ return (
+ <>
+
Welcome to the {roomId} room!
+ setMessage(e.target.value)} />
+ >
+ );
+}
+
+export default function App() {
+ const [roomId, setRoomId] = useState('general');
+ return (
+ <>
+
+
+
+ >
+ );
+}
+```
+
+```js chat.js
+export function createConnection({ serverUrl, roomId }) {
+ // A real implementation would actually connect to the server
+ return {
+ connect() {
+ console.log('✅ Connecting to "' + roomId + '" room at ' + serverUrl + '...');
+ },
+ disconnect() {
+ console.log('❌ Disconnected from "' + roomId + '" room at ' + serverUrl);
+ }
+ };
+}
+```
+
+```css
+input { display: block; margin-bottom: 20px; }
+button { margin-left: 10px; }
+```
+
+
+
+You don't want the chat to re-connect every time you start typing a message in that chat. To fix this problem, move creation of the `options` object inside the Effect so that the Effect only depends on the `roomId` string:
+
+
+
+```js
+import { useState, useEffect } from 'react';
+import { createConnection } from './chat.js';
+
+const serverUrl = 'https://localhost:1234';
+
+function ChatRoom({ roomId }) {
+ const [message, setMessage] = useState('');
+
+ useEffect(() => {
+ const options = {
+ serverUrl: serverUrl,
+ roomId: roomId
+ };
+ const connection = createConnection(options);
+ connection.connect();
+ return () => connection.disconnect();
+ }, [roomId]);
+
+ return (
+ <>
+
Welcome to the {roomId} room!
+ setMessage(e.target.value)} />
+ >
+ );
+}
+
+export default function App() {
+ const [roomId, setRoomId] = useState('general');
+ return (
+ <>
+
+
+
+ >
+ );
+}
+```
+
+```js chat.js
+export function createConnection({ serverUrl, roomId }) {
+ // A real implementation would actually connect to the server
+ return {
+ connect() {
+ console.log('✅ Connecting to "' + roomId + '" room at ' + serverUrl + '...');
+ },
+ disconnect() {
+ console.log('❌ Disconnected from "' + roomId + '" room at ' + serverUrl);
+ }
+ };
+}
+```
+
+```css
+input { display: block; margin-bottom: 20px; }
+button { margin-left: 10px; }
+```
+
+
+
+Notice that you didn't start by editing the dependency list to remove the `options` dependency. That would be wrong. Instead, you changed the surrounding code so that the dependency became *unnecessary.* You can think of the dependency list as a list of all the reactive values used by your Effect's code. You don't intentionally choose what to put on that list. The list describes your code. To change the dependency list, change the code.
+
+
+
+Read **[Removing Effect Dependencies](/learn/removing-effect-dependencies)** to learn how to make your Effect re-run less often.
+
+
+
+## Reusing logic with custom Hooks {/*reusing-logic-with-custom-hooks*/}
+
+React comes with built-in Hooks like `useState`, `useContext`, and `useEffect`. Sometimes, you’ll wish that there was a Hook for some more specific purpose: for example, to fetch data, to keep track of whether the user is online, or to connect to a chat room. To do this, you can create your own Hooks for your application's needs.
+
+In this example, the `usePointerPosition` custom Hook tracks the cursor position, while `useDelayedValue` custom Hook returns a value that's "lagging behind" the value you passed by a certain number of milliseconds. Move the cursor over the sandbox preview area to see a moving trail of dots following the cursor:
+
+
+
+```js
+import { useState, useEffect } from 'react';
+import { usePointerPosition } from './usePointerPosition.js';
+import { useDelayedValue } from './useDelayedValue.js';
+
+export default function Canvas() {
+ const pos1 = usePointerPosition();
+ const pos2 = useDelayedValue(pos1, 100);
+ const pos3 = useDelayedValue(pos2, 200);
+ const pos4 = useDelayedValue(pos3, 100);
+ const pos5 = useDelayedValue(pos3, 50);
+ return (
+ <>
+
+
+
+
+
+ >
+ );
+}
+
+function Dot({ position, opacity }) {
+ return (
+
+ );
+}
+```
+
+```js usePointerPosition.js
+import { useState, useEffect } from 'react';
+
+export function usePointerPosition() {
+ const [position, setPosition] = useState({ x: 0, y: 0 });
+ useEffect(() => {
+ function handleMove(e) {
+ setPosition({ x: e.clientX, y: e.clientY });
+ }
+ window.addEventListener('pointermove', handleMove);
+ return () => window.removeEventListener('pointermove', handleMove);
+ }, []);
+ return position;
+}
+```
+
+```js useDelayedValue.js
+import { useState, useEffect } from 'react';
+
+export function useDelayedValue(value, delay) {
+ const [delayedValue, setDelayedValue] = useState(value);
+
+ useEffect(() => {
+ setTimeout(() => {
+ setDelayedValue(value);
+ }, delay);
+ }, [value, delay]);
+
+ return delayedValue;
+}
+```
+
+```css
+body { min-height: 300px; }
+```
+
+
+
+You can create custom Hooks, compose them together, pass data between them, and reuse them between components. As your app grows, you will write fewer Effects by hand because you'll be able to reuse custom Hooks you already wrote. There are also many excellent custom Hooks maintained by the React community.
+
+
+
+Read **[Reusing Logic with Custom Hooks](/learn/reusing-logic-with-custom-hooks)** to learn how to share logic between components.
+
+
+
+## What's next? {/*whats-next*/}
+
+Head over to [Referencing Values with Refs](/learn/referencing-values-with-refs) to start reading this chapter page by page!
diff --git a/beta/src/pages/learn/reusing-logic-with-custom-hooks.md b/beta/src/pages/learn/reusing-logic-with-custom-hooks.md
index 4a7cdac1..b5ac3c36 100644
--- a/beta/src/pages/learn/reusing-logic-with-custom-hooks.md
+++ b/beta/src/pages/learn/reusing-logic-with-custom-hooks.md
@@ -11,7 +11,7 @@ React comes with several built-in Hooks like `useState`, `useContext`, and `useE
- What custom Hooks are, and how to write your own
-- How to use reuse logic between components
+- How to reuse logic between components
- How to name and structure your custom Hooks
- When and why to extract custom Hooks
@@ -2464,6 +2464,7 @@ export default function Canvas() {
>
);
}
+
function Dot({ position, opacity }) {
return (