The React newbie village
Component thinking – Create a Todo app
In this section we will learn:
- How to split an app into components
- Concepts: Presentation /dumb components and container components
Component thinking simply means breaking the app into components one by one. Take the example of building a simple to-do application that breaks down many components and lets do it together.
List of todos
We start with the to-do list and design the underlying data structure before we think about presenting the data.
First to-do items have description fields, then think about how to complete a to-do item. The to-do list should contain a list of items because we want to execute some of the items on the list at some point in time. After the item is completed, should it be deleted or marked as completed? We choose the latter, and in order to make ourselves feel good about completing it, colleagues can also check in on what they’ve done before. Information about our data structure. It would probably look something like this:
[{title: 'clean'.done: false
}, {
title: 'do dishes'.done: false}]Copy the code
The draft data structure is designed. Try rendering it in React.
Apply colours to a drawing list
Try it first:
{todos.map(todo= > (
<div>
<input type="checkbox" checked={todo.done} /> {todo.title}
</div>
)}
Copy the code
The to-do list is rendered, but the value of the to-do list cannot be changed. We need to build a real React component class and add event change functionality.
import React, { Component } from 'react';
import styled from 'styled-components';
const todos = [
{
title: 'clean'.done: false}, {title: 'do the dishes'.done: true,}];const Todos = styled.div` padding: 30px; `;
const Todo = styled.div` box-shadow: 0 0 5px gray; padding: 30px; margin-bottom: 10px; `;
class App extends Component {
render() {
return (
<Todos>
<h2>Todos</h2>
{todos.map(todo => (
<Todo>
<input type="checkbox" checked={todo.done} /> {todo.title}
</Todo>
))}
</Todos>); }}export default App;
Copy the code
For styled components, NPM install –save Styled components is required
We have created a complete component above, but we have not added support for changing the backlog. So let’s do this. We need to consider:
- Add a listener for the check box
onChange
The event - Modify the to-do list by calling the event bound event method
Bind the onChange event to the listener using the arrow function:
<input type="checkbox" checked={todo.done} onChange={() => this.handleChange(todo)} />
Copy the code
Then figure out a way to change your to-do list. We could change the toDOS list directly, but it’s better to create a toDOS state in the React component, like this:
state = {
todos
}
Copy the code
Final code:
import React, { Component } from 'react';
import styled from 'styled-components';
const todos = [
{
title: 'clean'.done: false.id: 1}, {title: 'do the dishes'.done: true.id: 2,}];const Todos = styled.div` padding: 30px; `;
const Todo = styled.div` box-shadow: 0 0 5px gray; padding: 30px; margin-bottom: 10px; `;
class App extends Component {
state = {
todos,
};
handleChecked = (todo) = > {
const newTodos = this.state.todos.map(t= > {
if (t.id === todo.id) {
return { ...t, done: !t.done };
}
return t;
});
this.setState({
todos: newTodos,
});
}
render() {
return (
<Todos>
<h2>Todos</h2>
{this.state.todos.map(todo => (
<Todo key={todo.id}>
<input type="checkbox" onChange={()= > this.handleChecked(todo)} checked={todo.done} />
{todo.title}
</Todo>
))}
</Todos>); }}export default App;
Copy the code
HandleChecked () is the onChange binding method, click the Modify check box:
handleChecked = (todo) = > {
const newTodos = this.state.todos.map(t= > {
if (t.id === todo.id) {
return { ...t, done: !t.done };
}
return t;
});
this.setState({
todos: newTodos,
});
}
Copy the code
Iterate to find the clicked to-do item and modify it using the object expansion operator:
return { ...t, done: !t.done }
Copy the code
Note that todo now consists of three properties:
- title
- done
- id
The ID attribute is used to identify which TODO to modify.
Create a TODOS component
Now that we have the basic functionality of the App, it looks confusing to put all the functional logic together. Now we refactor the code by creating a Todos component, which will be split into the following files:
- App.js – App entry file
- Todos.js – List logic file
Move the list logic file to todos.js
// Todos.js
import React, { Component } from 'react';
import styled from 'styled-components';
import PropTypes from 'prop-types';
const TodosContainer = styled.div` padding: 30px; `;
const Todo = styled.div` box-shadow: 0 0 5px gray; padding: 30px; margin-bottom: 10px; `;
class Todos extends Component {
static propTypes = {
todos: PropTypes.array.isRequired,
}
state = {
todos: this.props.todos,
};
handleChecked = (todo) = > {
const newTodos = this.state.todos.map(t= > {
if (t.id === todo.id) {
return { ...t, done: !t.done };
}
return t;
});
this.setState({
todos: newTodos,
});
}
render() {
return (
<TodosContainer>
<h2>Todos</h2>
{this.state.todos.map(todo => (
<Todo key={todo.id}>
<input type="checkbox" onChange={()= > this.handleChecked(todo)} checked={todo.done} /> {todo.title}
</Todo>
))}
</TodosContainer>); }}export default Todos;
Copy the code
Since React 15.5.0, PropTypes requires users to introduce NPM install –save prop-types themselves
The App entry file is as follows:
import React, { Component } from 'react';
import Todos from './Todos';
const todos = [
{
title: 'clean'.done: false.id: 1}, {title: 'do the dishes'.done: true.id: 2,}];class App extends Component {
render() {
return (
<Todos todos={todos} />); }}export default App;
Copy the code
Split toDOS components
We see a lot of content in the todos.js component. We can continue to break it up on a liability basis. While it works fine, we usually break it up into smaller components that are easier to manage and expand. In addition, smaller components make components easier to call. How do YOU split todos.js components?
- The Todos component, which renders a list of Todo components, handles all events
- Todo component, responsible for rendering Todo and sending up any changes
Refactor the code to add todo.js:
// Todo.js
import React from 'react';
import styled from 'styled-components';
import PropTypes from 'prop-types';
const TodoContainer = styled.div` box-shadow: 0 0 5px gray; padding: 30px; margin-bottom: 10px; `;
const Todo = ({ todo, handleChecked }) = > (
<TodoContainer key={todo.id}>
<input type="checkbox" onChange={()= > handleChecked(todo)} checked={todo.done} />
{todo.title}
</TodoContainer>
);
Todo.propTypes = {
todo: PropTypes.shape({
title: PropTypes.string,
done: PropTypes.bool,
id: PropTypes.number,
}),
handleChecked: PropTypes.func,
};
export default Todo;
Copy the code
Put Todo in a separate component. This component does not inherit React and is a simple functional component, often referred to as a demo or dumb component, that is context-blind and relies only on input. The result of the todo depends on the methods passed in · handleChecked() and props.
Change todos.js to:
// Todos.js
import React, { Component } from 'react';
import styled from 'styled-components';
import PropTypes from 'prop-types';
import Todo from './Todo';
const TodosContainer = styled.div` padding: 30px; `;
class Todos extends Component {
static propTypes = {
todos: PropTypes.array.isRequired,
}
state = {
todos: this.props.todos,
};
handleChecked = (todo) = > {
const newTodos = this.state.todos.map(t= > {
if (t.id === todo.id) {
return { ...t, done: !t.done };
}
return t;
});
this.setState({
todos: newTodos,
});
}
render() {
return (
<TodosContainer>
<h2>Todos</h2>
{this.state.todos.map(todo => (
<Todo todo={todo} key={todo.id} handleChecked={this.handleChecked} />
))}
</TodosContainer>); }}export default Todos;
Copy the code
Introducing Todo components in todos.js:
import Todo from './Todo';
Copy the code
Invoke the Todo component
<Todo todo={todo} key={todo.id} handleChecked={this.handleChecked} />
Copy the code
In the code above, rendering is done in a toDO component called a presentation component by passing in the data todo and the method handleChecked() to the todo component.
Another type of component is called container component, which is the component where the data and logic reside and which “contains” the content of the application.
In our App, the Todo component is the presentation component and the Todos component is the container component. An App built with a component mindset should contain most presentation components and a few container components.
conclusion
Create a to-do App.
The App is divided into several components dedicated to a particular function. Smaller components are easier to maintain and call later, and also easier.
Two terms, show/dumb component and container component, were introduced. Explained their concepts, their differences are executed in the incoming content.