Scalable React Components architecture
13 min readIt's been a while since I've started working with React and React-Native in production. One of the greatest things about React is the flexibility the library gives to you. Meaning that you are free to decide how do you want to implement almost every detail of your project for example the architecture and structure.
However this freedom on the long term, could lead to a complex and messy codebase, specially if you don't follow a pattern. In this post I'll explain a simple way to organize and structure React Components.
A Component is a JavaScript function or class that returns a piece of UI.
We're going to create an EmojiList
component and then we are going to refactor it breaking it up into smaller isolated pieces applying the folder pattern. Here's how our component looks like:
EmojiList
As I mentioned before, we can start really simple and small, without following any pattern. This is our EmojiList
component contained in a single function.
If you open the CodeSandbox sidebar you'll see that our file tree looks like this:
.
├── components
│ ├── EmojiList.js
│ └── styles.js
└── index.js
There's nothing wrong with this approach. But on larger codebases that kind of component becomes hard to maintain, because there a lot of things in it: state, ui, data... Take a look at our component code below 👇
EmojiList.js
import React from 'react'
import styles from './styles'
class EmojiList extends React.Component {
state = {
searchInput: '',
emojis: [],
}
render() {
const emojis = this.state.emojis.filter((emoji) =>
emoji.code.includes(this.state.searchInput.toLowerCase())
)
return (
<ul style={styles.list}>
<input
style={styles.searchInput}
placeholder="Search by name"
type="text"
value={this.state.searchInput}
onChange={(event) =>
this.setState({ searchInput: event.target.value })
}
/>
{emojis.map((emoji, index) => (
<li key={index} style={styles.item}>
<div style={styles.icon}>{emoji.emoji}</div>
<div style={styles.content}>
<code style={styles.code}>{emoji.code}</code>
<p style={styles.description}>{emoji.description}</p>
</div>
</li>
))}
</ul>
)
}
}
export default EmojiList
A step to improve this code, would be to create separate components into the same file and then using them at the main component. However, you'll be sharing styles among other things and that could be confusing.
Refactor
Let's start refactoring the single component into multiple ones by breaking up the UI into a component hierarchy.
If we take a look at the image, it's easy to identify that we can break up our UI in three different components: 🛠
EmojiList
: Combines the smaller components and shares the state down.SearchInput
: Receives user input and displays the search bar.EmojiListItem
: Displays the List Item for each emoji, with the icon, name and description.
We're going to create a folder for each component, with two files, an index.js
that is going to hold all the code for the component and the styles.js
. That's one of the good things about this pattern. Every component defines his own UI and styles, isolating this piece of code from another components that doesn't need to know anything about them.
Notice that inside the EmojiList
folder, (that is a component), we add two nested components that only will be used within the EmojiList
component. Again, that's because these two components aren't going to be used out of that context. This helps reducing the visual clutter a lot.
.
├── EmojiList
│ ├── EmojiListItem
│ │ ├── index.js
│ │ └── styles.js
│ ├── SearchInput
│ │ ├── index.js
│ │ └── styles.js
│ ├── index.js
│ └── styles.js
└── index.js
Now let's isolate and separate the code into the three components from the smallest to the biggest one:
EmojiListItem/
This component renders every emoji item that will appear on the list.
import React from 'react'
import styles from './styles'
const EmojiListItem = (props) => (
<li style={styles.item}>
<div style={styles.icon}>{props.emoji}</div>
<div style={styles.content}>
<code style={styles.code}>{props.code}</code>
<p style={styles.description}>{props.description}</p>
</div>
</li>
)
export default EmojiListItem
SearchInput/
This component receives the user input and updates the state of the parent component.
import React from 'react'
import styles from './styles'
const SearchInput = (props) => (
<input
style={styles.searchInput}
placeholder="Search by name"
type="text"
value={props.value}
onChange={props.onChange}
/>
)
export default SearchInput
EmojiList/
This is the top level component, holds the state and data of our example and imports the other components to recreate the whole UI of our tiny application. Isolating components makes the render method more readable and easier to understand ✨.
import React from 'react'
import SearchInput from './SearchInput'
import EmojiListItem from './EmojiListItem'
import styles from './styles'
class EmojiList extends React.Component {
state = {
searchInput: '',
emojis: [],
}
render() {
const emojis = this.state.emojis.filter((emoji) =>
emoji.code.includes(this.state.searchInput.toLowerCase())
)
return (
<ul style={styles.list}>
<SearchInput
onChange={(event) =>
this.setState({ searchInput: event.target.value })
}
value={this.state.searchInput}
/>
{emojis.map((emoji, index) => (
<EmojiListItem
key={index}
code={emoji.code}
description={emoji.description}
emoji={emoji.emoji}
/>
))}
</ul>
)
}
}
export default EmojiList
That's basically the architecture that I use at the company I'm working on. I'm pretty satisfied with the experience of using this pattern. Our components turned out a lot easier to maintain and use. Anyway there are no silver bullets on Software Engineering, so figure what works best for you or your team!
Enjoyed the article? 😍