In this tutorial we are going to use discuss on react best practices and best way to write clean code. In general, learning how to write cleaner React code will make you a more valuable and overall happier React developer. Lets quickly jump to react code.
How to Write Cleaner React Code
1. Make use of JSX shorthands
In the example below, we're using the prop showTitle to display the title of our app within a Navbar component.
// src/App.js export default function App() { return ( <main> <Navbar showTitle={true} /> </main> ); } function Navbar({ showTitle }) { return ( <div> {showTitle && <h1>My Special App</h1>} </div> ) }
Do we need to explicitly set show title to the Boolean true? We don't! A quick shorthand to remember is that any prop provided on a component has a default value of true.
So if we add the prop showTitle on Navbar, our title element will be shown:
// src/App.js export default function App() { return ( <main> <Navbar showTitle /> </main> ); } function Navbar({ showTitle }) { return ( <div> {showTitle && <h1>My Special App</h1>} // title shown! </div> ) }
Another useful shorthand to remember involves passing string props. When you're passing a prop value that's a string, you don't need to wrap it in curly braces.
If we are setting the title of our Navbar, with the title prop, we can just include its value in double quotes:
// src/App.js export default function App() { return ( <main> <Navbar title="My Special App" /> </main> ); } function Navbar({ title }) { return ( <div> <h1>{title}</h1> </div> ) }
2. Reduce prop drilling with React context
In a typical React application, data is passed top-down (parent to child) via props, but such usage can be cumbersome for certain types of props (e.g. locale preference, UI theme) that are required by many components within an application. Context provides a way to share values like these between components without having to explicitly pass a prop through every level of the tree.
Another essential pattern to employ for your React projects, especially if you have common properties that you want to reuse across your components, and you find yourself writing lots of duplicate props, is to use React Context.
For example, if we wanted to share user data across multiple components, instead of multiple repeat props (a pattern called props drilling), we could use the context feature that's built into the React library.
In our case, if we wanted to reuse user data across our Navbar and FeaturedPosts components, all we would need to do is wrap our entire app in a provider component.
Next, we can pass the user data down on the value prop and consume that context in our individual components with the help of the useContext hook:
// src/App.js import React from "react"; const UserContext = React.createContext(); export default function App() { const user = { name: "Reed" }; return ( <UserContext.Provider value={user}> <main> <Navbar title="My Special App" /> <FeaturedPosts /> </main> </UserContext.Provider> ); } // src/components/Navbar.js function Navbar({ title }) { const user = React.useContext(UserContext); return ( <div> <h1>{title}</h1> {user && <a href="/logout">Logout</a>} </div> ); } // src/components/FeaturedPosts.js function FeaturedPosts() { const posts = useFetchPosts(); const user = React.useContext(UserContext); if (user) return null; return ( <ul> {posts.map((post) => ( <li key={post.id}>{post.title}</li> ))} </ul> ); }
3. Format inline styles for less bloated code
A common pattern for React developers is to write inline styles in their JSX but once again, it makes our code harder to read and harder to write additional JSX:
// src/App.js export default function App() { return ( <main style={{ textAlign: 'center' }}> <Navbar title="My Special App" /> </main> ); } function Navbar({ title }) { return ( <div style={{ marginTop: '20px' }}> <h1 style={{ fontWeight: 'bold' }}>{title}</h1> </div> ) }
We want to apply this concept of separation of concerns to our JSX styles by moving our inline styles into a CSS stylesheet, which we can import into whatever component we like.
An alternative way to rewrite your inline styles is by organizing them into objects. You can see what such a pattern would look like below:
// src/App.js export default function App() { const styles = { main: { textAlign: "center" } }; return ( <main style={styles.main}> <Navbar title="My Special App" /> </main> ); } function Navbar({ title }) { const styles = { div: { marginTop: "20px" }, h1: { fontWeight: "bold" } }; return ( <div style={styles.div}> <h1 style={styles.h1}>{title}</h1> </div> ); }
4. Move shared functionality into React hooks
Taking a look at our FeaturedPosts component, let's say instead of displaying static posts data, we want to fetch our post data from an API.
We might do so with the fetch API. You can see the result below for that:
// src/components/FeaturedPosts.js import React from 'react'; export default function FeaturedPosts() { const [posts, setPosts] = React.useState([]); React.useEffect(() => { fetch('https://jsonplaceholder.typicode.com/posts') .then(res => res.json()) .then(data => setPosts(data)); }, []); return ( <ul> {posts.map((post) => ( <li key={post.id}>{post.title}</li> ))} </ul> ); }
However, what if we wanted to perform this request for data across multiple components?
Let's say in addition to a FeaturedPosts component we wanted to create a component called just Posts with the same data. We would have to copy the logic that we used to fetch our data and paste it within that component as well.
To avoid having to do that, why don't we just use a new React hook which we could call useFetchPosts:
// src/hooks/useFetchPosts.js import React from 'react'; export default function useFetchPosts() { const [posts, setPosts] = React.useState([]); React.useEffect(() => { fetch('https://jsonplaceholder.typicode.com/posts') .then(res => res.json()) .then(data => setPosts(data)); }, []); return posts; }
Once we've created this hook in a dedicated 'hooks' folder we can reuse it in whichever components we like, including our FeaturedPosts component:
// src/components/FeaturedPosts.js import useFetchPosts from '../hooks/useFetchPosts.js'; export default function FeaturedPosts() { const posts = useFetchPosts() return ( <ul> {posts.map((post) => ( <li key={post.id}>{post.title}</li> ))} </ul> ); }
5. Move unrelated code into a separate component
The easiest and most important way to write cleaner React code is to get good at abstracting our code into separate React component.
Let's look at the example below. What is our code doing?
Our app is displaying a Navbar component. We are iterating over an array of posts with .map() and displaying their title on the page.
// src/App.js export default function App() { const posts = [ { id: 1, title: "How to Build YouTube with React" }, { id: 2, title: "How to Write Your First React Hook" } ]; return ( <main> <Navbar title="My Special App" /> <ul> {posts.map(post => ( <li key={post.id}> {post.title} </li> ))} </ul> </main> ); } function Navbar({ title }) { return ( <div> <h1>{title}</h1> </div> ); }
Let's break down react code into separate component and take a look at the result:
// src/App.js export default function App() { return ( <main> <Navbar title="My Special App" /> <FeaturedPosts /> </main> ); } function Navbar({ title }) { return ( <div> <h1>{title}</h1> </div> ); } function FeaturedPosts() { const posts = [ { id: 1, title: "How to Build YouTube with React" }, { id: 2, title: "How to Write Your First React Hook" } ]; return ( <ul> {posts.map((post) => ( <li key={post.id}>{post.title}</li> ))} </ul> ); }
6. Create separate files for each component
In our previous example, we are including all of our components within a single file, the app.js file.
Similar to how we abstract code into separate components to make our app more readable, to make our application files more readable, we can put each component that we have into a separate file.
This, again, helps us separate concerns in our application, meaning that each file is responsible for just one component and there's no confusion where a component comes from if we want to reuse it across our app:
// src/App.js import Navbar from './components/Navbar.js'; import FeaturedPosts from './components/FeaturedPosts.js'; export default function App() { return ( <main> <Navbar title="My Special App" /> <FeaturedPosts /> </main> ); }
// src/components/Navbar.js export default function Navbar({ title }) { return ( <div> <h1>{title}</h1> </div> ); }
// src/components/FeaturedPosts.js export default function FeaturedPosts() { const posts = [ { id: 1, title: "How to Build YouTube with React" }, { id: 2, title: "How to Write Your First React Hook" } ]; return ( <ul> {posts.map((post) => ( <li key={post.id}>{post.title}</li> ))} </ul> ); }
Additionally, by including each individual component within its own file, we avoid one file becoming too bloated. We could easily see our app.js file becoming very large if we wanted to add all of our components into that file.
7. Remove as much JavaScript from your JSX as possible
Another very helpful, but often neglected way to clean up our components is to remove as much JavaScript from our JSX as possible.
Let's take a look at the example below:
// src/components/FeaturedPosts.js import useFetchPosts from '../hooks/useFetchPosts.js'; export default function FeaturedPosts() { const posts = useFetchPosts() return ( <ul> {posts.map((post) => ( <li onClick={event => { console.log(event.target, 'clicked!'); }} key={post.id}>{post.title}</li> ))} </ul> ); }
We're trying to handle a click event on one of our posts. You can see that our JSX becomes much harder to read. What can we do to fix this? We can extract the inline function, connected to the onClick into a separate handler, which we can give a an appropriate name like handlePostClick.
Once we do, our JSX becomes readable once again:
// src/components/FeaturedPosts.js import useFetchPosts from '../hooks/useFetchPosts.js'; export default function FeaturedPosts() { const posts = useFetchPosts() function handlePostClick(event) { console.log(event.target, 'clicked!'); } return ( <ul> {posts.map((post) => ( <li onClick={handlePostClick} key={post.id}>{post.title}</li> ))} </ul> ); }
This is all about writing cleaner react code using jsx. Thank you for reading this article, and if you have any problem, have a another better useful solution about this article, please write message in the comment section.
No comments:
Post a Comment