The previous tutorial explained how to build your first React Js application from scratch.
React Js simplifies creating complex UIs by breaking them down into smaller, reusable components. This article will explore the React component hierarchy and composition to help beginners understand how to structure their React applications effectively.
A React component is a self-contained piece of code. A component is responsible for rendering a part of a user interface. Components can be written as JavaScript functions or classes. They take inputs, called props, similar to Javascript functions that take parameters. Components return React elements that describe the UI.
export const TextInput = () => {
return (
<div className="form-field">
<label htmlFor="name">Name</label>
<input type="text" id="name" name="name" required />
</div>
)
}
export const Button = () => {
return (
<button type="submit">Submit</button>
)
}
The above two components take no props and return HTML representing user input with a label and a button. But the label Name, input attributes, and button value are static. I have no way to change them. That's where the props come in handy.
React applications are built using a tree-like structure of components. A parent component can contain one or more child components, which can have their children forming a hierarchy.
import { Button } from "./Button";
import { TextInput } from "./TextInput";
export const SignupForm = () => {
return (
<div className="signup-form">
<h1>Signup Form</h1>
<form>
<TextInput/>
<TextInput/>
<Button/>
</form>
</div>
)
}
My Signup form uses the
In React,
Here are the modified version of TextInput and Button components that accept props to decide how it should render.
export const TextInput = (props) => {
return (
<div className="form-field">
<label htmlFor="name">{props.label}</label>
<input type={props.inputType} id={props.id} name={props.name} required />
</div>
)
}
export const Button = (props) => {
return (
<button type="submit">{props.value}</button>
)
}
When a component receives data through props, it becomes a controlled component. The component is controlled by the parent component that passes the data.
When a child component accepts props, the parent component must provide all the required props to its child components. Now I can re-write my signup form with child components that accept props.
import { Button } from "./Button";
import { TextInput } from "./TextInput";
export const SignupForm = () => {
return (
<div className="signup-form">
<h1>Signup Form</h1>
<form>
<TextInput label="User Name" id="name" name="name" inputType="text"/>
<TextInput label="Password" id="password" name="password" inputType="password"/>
<Button value="Submit"/>
</form>
</div>
)
}
In the above example, the parent component
React promotes the concept of composition. Composition means building more complex components by combining simpler ones. This approach enables code reusability and makes managing and maintaining the application easier.
The signup form I discussed above is a good example of composition. I built the signup form by combining
export const SignupForm = () => {
return (
<div className="signup-form">
<h1>Signup Form</h1>
<form>
<TextInput type="text"/>
<TextInput type="password"/>
<Input type="check"/>
<TermsAndConditions/>
<Button value="Submit"/>
</form>
</div>
)
}
React components can be divided into two categories.
Container components manage data and state, while presentational components focus on rendering the UI based on the received props. That is more of a design pattern approach.
const DisplayContainer = (props) => {
/*
Do some processing/calculations on
cardsToDisplay variable
*/
return <Display cards={cardsToDisplay} />
}