
Epic React - Fundamentals
React Basics: Understanding the Principles of DOM Operations
Before diving deep into React, let’s review the most basic way of DOM operations. This will help us better understand why React is so popular and what problems it solves.
<script type="module">
// 1. Get the root element
const rootElement = document.getElementById('root')
// 2. Create new DOM element
const element = document.createElement('div')
// 3. Set element attributes
element.className = 'container'
// 4. Set element content
element.textContent = 'Hello World'
// 5. Add new element to DOM tree
rootElement.append(element)
</script>
Problems with JavaScript DOM API
- Verbose code: Even creating simple UI elements requires multiple lines of code
- Difficult to maintain: Direct DOM manipulation code is scattered and hard to track
- Performance concerns: Frequent DOM operations affect performance
This is exactly what modern frontend frameworks like React aim to solve.
Raw React: Deep Understanding of React’s Basic Concepts
Before starting to use React’s advanced features, let’s understand how React works at its core. By comparing native DOM operations with Raw React API, we can better understand React’s design philosophy.
Basic React Implementation
<script type="module">
import { createElement } from '/react.js'
import { createRoot } from '/react-dom/client.js'
// Get root element
const rootElement = document.getElementById('root')
// Create element using React
const element = createElement(
'div',
{ className: 'container' },
'Hello World'
)
// Render element
createRoot(rootElement).render(element)
</script>
This code demonstrates the most primitive way of using React. Compared to direct DOM manipulation, React provides a more declarative API.
Package Management and Dependencies
- In actual development, we need to manage React and its related dependencies. This requires:
- npm (Node Package Manager): The most mainstream package manager in the JavaScript ecosystem
- Package Registry: A centralized code repository where developers can publish and share reusable code packages
- ESM Service: Such as esm.sh, which provides React packages in CDN form for direct use of ES modules in browsers
React Core Concepts: Props and Children
Props
Props are the basic way of passing data in React, making data flow between components predictable and controllable.
Children
Children is a special prop in React used to define element’s child content. There are three ways to define children:
- As a standalone prop
createElement('div', { className: 'container', children: 'Hello World' })
- As multiple arguments
const elementProps = { id: 'element-id' }
const elementType = 'h1'
const reactElement1 = createElement(
elementType,
elementProps,
'Hello',
' ',
'world!'
)
- As an array
const children = ['Hello', ' ', 'world!']
const reactElement2 = createElement(elementType, elementProps, children)
Why Learn Raw React?
Although we rarely use these APIs directly in actual development, understanding them helps:
- Deeply understand how React works
- Better understand the essence of JSX (JSX eventually compiles into these raw API calls)
- Better debug complex problems when encountered
JSX: Modern Syntax for React Development
JSX is the most commonly used syntax sugar in React development, allowing us to write React components using HTML-like syntax. Let’s dive deep into the core concepts of JSX.
Babel and JSX Compilation
JSX code needs to be compiled through Babel to run in browsers. In development environment, we need:
<script type="text/babel" data-type="module">
import * as React from '/react.js'
import { createRoot } from '/react-dom/client.js'
const rootElement = document.getElementById('root')
const element = <div className="container">Hello World</div>
createRoot(rootElement).render(element)
</script>
Important notes:
- Set script tag’s type to text/babel
- Add data-type=“module” to support ES modules
- Babel will compile JSX into React.createElement calls
- Use className instead of class in JSX to define CSS classes
Interpolation
JSX supports inserting JavaScript expressions within curly braces {}:
const children = 'Hello World'
const className = 'container'
const element = <div className={className}>{children}</div>
This approach allows us to dynamically build UI content.
Converting HTML to JSX
When converting from HTML to JSX, note some syntax differences:
<div class="container">
<p>Here's Sam's favorite food:</p>
<ul class="sams-food">
<li>Green eggs</li>
<li>Ham</li>
</ul>
</div>
// class becomes className
<div className="container">
<p>Here's Sam's favorite food:</p>
<ul className="sams-food">
<li>Green eggs</li>
<li>Ham</li>
</ul>
</div>
React Fragment
React Fragment is a special component that allows us to group multiple elements together without adding extra nodes to the DOM:
// Using Fragment long syntax
<React.Fragment>
<p>Here's Sam's favorite food:</p>
<ul className="sams-food">
<li>Green eggs</li>
<li>Ham</li>
</ul>
</React.Fragment>
// Using short syntax
<>
<p>Here's Sam's favorite food:</p>
<ul className="sams-food">
<li>Green eggs</li>
<li>Ham</li>
</ul>
</>
Benefits of Fragment:
- Avoids adding unnecessary DOM nodes
- Maintains clear DOM structure
- Doesn’t affect CSS styles and layout
- Particularly suitable for scenarios requiring specific DOM structure (when combining multiple elements)
These features make JSX a powerful and flexible tool, maintaining HTML’s intuitiveness while providing all of JavaScript’s capabilities. Understanding these concepts is crucial for mastering React development.
React Components: Evolution from Functions to Components
In React, components are the basic building blocks of user interfaces. Let’s understand the concept of components and best practices through several examples.
Functional Approach
The most basic way is to use components as regular functions:
function message({ children }) {
return <div className="message">{children}</div>
}
const element = (
<div className="container">
{message({ children: 'Hello World' })}
{message({ children: 'Goodbye World' })}
</div>
)
While this approach works, it doesn’t utilize all features of React components.
Using React.createElement
function message({ children }) {
return <div className="message">{children}</div>
}
const element = (
<div className="container">
{React.createElement(message, { children: 'Hello World' })}
{React.createElement(message, { children: 'Goodbye World' })}
</div>
)
This approach is closer to React’s internal workings but is less commonly used in actual development.
Standard React Component Writing
Finally, this is the recommended way to write React components:
function Message({ children }) {
return <div className="message">{children}</div>
}
const element = (
<div className="container">
<Message>Hello World</Message>
<Message>Goodbye World</Message>
</div>
)
This approach has several important characteristics:
- Capitalized Component Names: React uses capitalization to distinguish custom components from native DOM elements
- JSX Syntax: Uses HTML-like syntax for more intuitive code
- children Property: Can contain child content like regular HTML elements
- Declarative Style: Code is clearer, intentions more explicit
Why This Is Best Practice?
- Readability: Component usage is similar to HTML, reducing learning curve
- Maintainability: Component structure and purpose are clear at a glance
- Reusability: Components can be easily reused in different places
- Encapsulation: Component’s internal implementation details are hidden from outside
Usage Tips
- Always use capitalized names for custom components
- Prefer JSX syntax over React.createElement
- Keep components single-responsibility
- Use props reasonably for data passing
Through this standard component writing approach, we can fully utilize React’s component features to build maintainable, scalable applications.
Typescript
typeof
The typeof operator allows you to use it in type context to reference the type of a variable or property
const user = { name: 'kody', isCute: true }
type User = typeof user
// type User = { name: string; isCute: boolean; }
keyof
The keyof operator allows you to obtain
type UserKeys = keyof User
// type UserKeys = "name" | "isCute"
Advanced TypeScript Type Operations in React
TypeScript provides powerful type support for React development. Let’s dive into some commonly used type operators and techniques.
typeof Operator
typeof
allows us to extract types from existing values:
const user = { name: 'kody', isCute: true }
type User = typeof user
// Results in type: { name: string; isCute: boolean; }
This is particularly useful when deriving types from actual data structures.
keyof Operator
keyof
is used to get all keys of a type as a union type:
type UserKeys = keyof User
// Results in type: "name" | "isCute"
Combining keyof typeof
This combination is particularly useful when creating types from objects:
const operations = {
'+': (left: number, right: number): number => left + right,
'-': (left: number, right: number): number => left - right,
'*': (left: number, right: number): number => left * right,
'/': (left: number, right: number): number => left / right,
}
type Operator = keyof typeof operations
// Results in type: '+' | '-' | '*' | '/'
Note: The order of typeof keyof
is meaningless because keyof
must act on types, not values.
Default Properties (Default Props)
TypeScript makes it easy to define function parameters with default values:
function add(a: number = 0, b: number = 0): number {
return a + b
}
// Returns 0 when no parameters are passed
Record Utility Type
Record
is used to create object types with specific key and value types:
type OperationFn = (left: number, right: number) => number
type Operator = '+' | '-' | '/' | '*'
const operations: Record<Operator, OperationFn> = {
'+': (left, right) => left + right,
'-': (left, right) => left - right,
'*': (left, right) => left * right,
'/': (left, right) => left / right,
}
Satisfies Operator
satisfies
is a new operator introduced in TypeScript 4.9 that allows us to validate expression types without affecting inference results:
type ValidCandies = 'twix' | 'snickers' | 'm&ms'
// Using type annotation
const candy1: ValidCandies = 'twix'
// candy1's type is ValidCandies
// Using satisfies
const candy2 = 'twix' satisfies ValidCandies
// candy2's type is literal type 'twix'
Advantages of satisfies
:
- Maintains more precise type inference
- Provides type checking without widening types
- Particularly useful in scenarios requiring type safety while maintaining literal types
Practical Application Tips
- Using typeof:
- When you need to create types from existing objects
- Avoid type definition repetition
- Using keyof:
- When you need to restrict property access
- Create stricter type constraints
- Using Record:
- When creating mapping objects
- Ensure object property completeness
- Using satisfies:
- When you need type checking but want to maintain literal types
- Validate implementation without affecting type inference
These TypeScript features can help us create safer, more maintainable code in React development. Proper use of these features can improve code quality and reduce runtime errors.
Detailed Guide to React Form Implementation
Basic Form Implementation
The simplest form implementation is as follows:
function App() {
return (
<form>
<div>
<label htmlFor="usernameInput">Username:</label>
<input id="usernameInput" name="username" />
</div>
<button type="submit">Submit</button>
</form>
)
}```
This implementation has an issue: form submission triggers page refresh.
## Form Submission Address Configuration
By setting the `action` attribute, we can specify the target address for form submission:
```jsx
<form action="api/onboarding">
// api/onboarding is just for testing
If action
is not set, form data will be submitted to the current URL by default.
Common Form Input Types
HTML5 provides various input types for better user experience:
// Password input
<div>
<label htmlFor="passwordInput">Password:</label>
<input id="passwordInput" name="password" type="password" />
</div>
// Age input (number type)
<div>
<label htmlFor="ageInput">Age:</label>
<input id="ageInput" name="age" type="number" min="0" max="200" />
</div>
// Image upload
<div>
<label htmlFor="photoInput">Photo:</label>
<input id="photoInput" name="photo" type="file" accept="image/*" />
</div>
// Color picker
<div>
<label htmlFor="colorInput">Favorite Color:</label>
<input id="colorInput" name="color" type="color" />
</div>
// Date selection
<div>
<label htmlFor="startDateInput">Start Date:</label>
<input id="startDateInput" name="startDate" type="date" />
</div>
Handling Form Submission Issues
The basic implementation has two main problems:
- Sensitive information (like passwords) appears in the URL
- The page completely refreshes, causing client-side code to reload
Solutions:
- Use POST method to submit data Change the browser’s default GET to POST, which will transmit data through the request body rather than URL parameters, more suitable for handling sensitive information:
<form action="api/onboarding" method="POST"></form>
- Handle POST requests on the server side:
export async function action({ request }: { request: Request }) {
const data = await request.formData()
return respondWithDataTable(data)
}
}```
We use `request.formData()` here because the request isn't a plain string, but in `application/x-www-form-urlencoded` format.
However, there's still an issue: when uploading photos, only the filename is returned, not the photo itself. This is because the browser's default `application/x-www-form-urlencoded` encoding is suitable for simple form data but not for large files like images. Large files can't be represented by string URLs, so we need to add new attributes to the form to let the browser accept the file itself:
```tsx
<form
action="api/onboarding"
method="POST"
encType="multipart/form-data"
></form>
The only way to solve the page refresh issue is to handle form submission through JavaScript rather than browser default behavior. In real work, we usually use frameworks like Remix to solve this problem, but the basic idea is to add an onSubmit handler to the form to take over the submission process and prevent browser refresh:
<form
action="api/onboarding"
method="POST"
encType="multipart/form-data"
onSubmit={event => {
event.preventDefault()
// ...
}}
></form>
This prevents page refresh, but we still need to manually get the form data. This is where the FormData API comes in:
const form = event.currentTarget
const formData = new FormData(form)
// Browser console.log(formData) doesn't show well
// You can view the data this way:
console.log(Object.fromEntries(formData))
// This converts formData to a plain object for easy viewing
// Note: formData may contain multiple values for the same key
// So this method is not recommended in production
Complete code
function App() {
return (
<form
action="api/onboarding"
method="POST"
encType="multipart/form-data"
onSubmit={event => {
event.preventDefault()
const formData = new FormData(event.currentTarget)
console.log(Object.fromEntries(formData))
}}
>
// your inputs
</form>
)
}
React also provides a more elegant built-in solution: the form’s action attribute can accept a function, which will receive the formData object as a parameter. Note that this is a React-specific feature, native HTML doesn’t support using functions as action values:
function App() {
function logFormData(formData: FormData) {
console.log(Object.fromEntries(formData))
}
return (
<form action={logFormData}>
// your inputs
</form>
Epic React Fundamental Summary
This article summarizes the key knowledge points from the Epic React Forms Workshop, but does not include practice-intensive content such as styling, inputs, errors, and arrays. For these topics, it is recommended to refer to the original workshop for learning.