React projects can easily become a mess. Losing track of the location of files is very common and can lead to a significant loss of efficiency in the development process. So, how do you improve and organize your React project? By organizing all the documents for your project in a multi-level structure.

So, you’ll always know how to place — and find — each file. This is a game changer. Not only will it make you a more effective developer, but it will also make your projects easier to maintain.

In addition, your entire team will benefit by turning the project into an organized structure. This is because different people can work on different layers, avoiding overlap and potential overhead.

Let’s take a look at why you might want to use multiple layers in React.

How does a multilayer structure optimize React applications

Reuse code

Imagine that all your React projects share the same multi-level file structure to maximize your time. This shared file structure allows you to import any type of file or code snippet from one project to another, rather than adjusting components or files imported from other code bases to your target project, like a simple copy and paste operation. This encourages code reuse, making it easier and less time for you to build your application.

For example, suppose you have a project with several math problems. By using this file structure, you won’t be tempted to spread out all the necessary utility functions in your code base.

Instead, it motivates you to define a math-utils.js file where you can store all your math utilities. Furthermore, by externalizing elements that serve the same purpose in the same file, you make them easier to reuse. In fact, if you face similar problems in future projects, you can simply copy the math-utils.js file you defined earlier and paste it into your utils layer.

Avoid code duplication

The consequence of imposing layers is that every file, function, or piece of code belongs in a particular place. This means that before writing anything new, you should look at its what-if folder to avoid code duplication and reduce the size of your build.

A bundle with a lot of code is obviously bigger than it needs to be. This is bad for browsers because they have to download and render code.

So, avoiding code duplication will make your application cleaner, easier to build, and faster to render. For example, you might be told to follow this approach whenever you need to retrieve data returned from a particular API.

function FooComponent1(props) {     
      // ...

      // retrieving data
      fetch(`/api/v1/foo`)
      .then((resp) => resp.json())
      .then(function(data) {
          const results = data.results;
          // handling data ...
        })
        .catch(function(error) {
          console.log(error);
        });

      // component logic ...
}

export default FooComponent1
Copy the code
function FooComponent2(props) {
      // ... 

      // retrieving data
      fetch(`/api/v1/foo`)
      .then((resp) => resp.json())
      .then(function(data) {
          const results = data.results;
          // handling data in a different way than FooComponent1 ...
        })
        .catch(function(error) {
          console.log(error);
        });

      // different component logic from FooComponent1 ...
}

export default FooComponent2

Copy the code

As you can see, FooComponent1 and FooComponent2 are affected by code duplication issues. In particular, the endpoint of the API is duplicated, forcing you to update it every time you change it.

This is not a good habit. Instead, you should map the API to the API layer, as shown below.

export const FooAPI = { // ... // mapping the API of interest get: function() { return axiosInstance.request({ method: "GET", url: `/api/v1/foo` }); } / /... }Copy the code

Then use it where needed to avoid code duplication.

import React from "react"; import {FooAPI} from ".. /.. /api/foo"; / /... function FooComponent1(props) { // ... // retrieving data FooAPI .get() .then(function(response) { const results = response.data.results; // handling data ... }) .catch(function(error) { console.log(error); }); // component logic ... } export default FooComponent1Copy the code
import React from "react"; import {FooAPI} from ".. /.. /api/foo"; / /... function FooComponent2(props) { // ... // retrieving data FooAPI .get() .then(function(response) { const results = response.data.results; // handling data in a different way than FooComponent1 ... }) .catch(function(error) { console.log(error); }); // different component logic from FooComponent1 ... } export default FooComponent2Copy the code

For now, the API endpoint is only kept in one place, as it should always be.

Take advantage of team synergy

Using a well-known file structure means that all members of the development team are on the same starting line and use it in the same way.

Every time a team member maps a new API, or creates a new utils file or a new component, for example, any team member can use it immediately. And because every file is logically placed in a specific folder, everyone knows how (and where) to access it, communication overhead is reduced or eliminated altogether, simplifying the company’s development process.

A potential side effect of building a multi-tier structure

Creating an NPM module can be complex

There is a potential downside to having such an organized file architecture: files are distributed across many folders, which forces you to use relative imports intensively. This represents a well-known problem when creating NPM modules.

Relative imports are easy to break, especially when trying to encapsulate part of the code base so that it can be distributed as a separate module. On the one hand, this of course represents an additional challenge when trying to create an NPM package.

On the other hand, this can be easily avoided by turning relative imports into unbreakable absolute imports, as described here. Technology like this allows you to put this.

import UserComponent from ".. /.. /components/UserComponent"; import userDefaultImage from ".. /.. /.. /assets/images/user-default-image.png"; import {UserAPI} from ".. /.. /apis/user";Copy the code

It looks like this.

importUserComponentfrom "@components/UserComponent";
import userDefaultImage from "@assets/images/user-default-image.png";
import {UserAPI} from "@apis/user";

Copy the code

So, using many relative imports should not be seen as a real problem.

Now, let’s look at how to design an efficient multi-tier architecture for your project in React and JavaScript.

Creating a multi-tier architecture

First, let’s look at the end result of a multi-tier architecture. Keep in mind that each layer of our architecture should be enclosed in a specific folder. As expected, it is strongly recommended that the names of these folders be the same as the layers that the schema contains. This makes retrieving files intuitive and fast. You will also always know where to place a new file, which is a big benefit.

This is what the final structure looks like.

As you can see, each layer of the architecture is represented by a folder.

Now, let’s take a closer look at each of the layers covered by the architecture described.

1. The API layer

By leveraging a promise-based HTTP client such as Axios, you can define a function for each API your application relies on, encapsulating all the logic needed to call it. This way, you can separate where you define API requests from where they are actually used. Let’s look at how to create such a layer with a simple example.

const axiosInstance = axios.create({ baseURL: 'https://yourdomain.org' }); export const UserAPI = { getAll: function() { return axiosInstance.request({ method: "GET", url: `/api/v1/users` }); }, getById: function(userId) { return axiosInstance.request({ method: "GET", url: `/api/v1/users/${userId}` }); }, create: function(user) { return axiosInstance.request({ method: "POST", url: `/api/v1/users`, data: user }); }, update: function(userId, user) { return axiosInstance.request({ method: "PUT", url: `/api/v1/users/${userId}`, data: user, }); }},Copy the code

Now, you can import UserAPI objects and make calls where you need them, like this.

import React, {useEffect, useState} from "react"; import {UserAPI} from ".. /.. /api/user"; / /... function UserComponent(props) { const { userId } = props; / /... const [user, setUser] = useState(undefined) // ... useEffect(() => { UserAPI.getById(userId).then( function (response) { // response handling setUser(response.data.user) }  ).catch(function (error) { // error handling }); } []); / /... } export default UserComponent;Copy the code

By using [Promise] (https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Promise), You can collect all API request definitions in one place. But let’s not lose sight of the goal. This article is not intended to show you what you can achieve with the API layer, you can read further by following this and this.

Organize all files belonging to the API layer into the same folder.

Instead, focus on the fact that with this approach, you will be able to define a completely dedicated layer for API requests. This will keep all the logic for processing API requests in one place, rather than repeating the entire logic to call the API every time it is needed.

Layer 2. Assets

Your React project may rely on files that are not strictly part of your code base, such as multimedia files. It’s best to have a place to store them all, because you should clearly separate them from your code. Note that such a layer can organize subfolders by type or extension to keep them organized.

A sample of asset layers organized by type.

It’s also a great place to store all the translation files you need for an international application with an internationalized framework, such as [i18Next](https://www.i18next.com/).

Place all i18Next translation files in the translation folder.

3. The component layer

This is where you should put all the components. You should create a folder for each component and name it the same way as the component itself. Such a folder usually contains only two files: index.js and index.css. They represent where the React component is defined and where its style is included, respectively.

A running sample component layer.

Because your components can be large, you should consider organizing them in a nested structure. Finding a good standard to assign them to subfolders is not always an easy task.

Also, keep in mind that you can change it at any time, rearranging the storage structure of your components as you wish.

4. Constant layer

A common practice is to define all constants used in your application in a single constans.js file. In a small project, this approach might seem like the best solution. But inevitably, as your file gets bigger, it slowly becomes a messy file.

That’s why you should split your constants into multiple files. For example, in an internationalized application, you might consider storing all custom keys in a specific file. [i18next] (https://www.npmjs.com/package/i18next) used in all custom keys are stored in a specific file.

This way, all i18n-related constants will be in the same file, making them easier to manage. Similarly, you can follow the latter approach in other situations, such as the example below.

Split all application constants into four different files.

5. Redux layer

This is an optional layer that you should define only if your application uses Redux. As stated in the official documentation, Redux is a predictable state container for JavaScript applications. If you’ve ever used Redux, you know how many files it requires.

It is common to get lost in the multitude of Redux restorer and Redux actions, especially the latter. In addition, defining actions involves template code. This is why the use of a library, such as [redux – actions] (https://www.npmjs.com/package/redux-actions) to manage all of your actions. You will then be able to simply get an organized structure that arranges your files as follows.

The Redux layer of a sample consists of two folders and a file.

As shown above, you can put all your Redux reducers in the Reducers folder, your Redux actions in the Actions folder, and define a top-level index.js file that contains your storage and persistence logic. You can even organize the files contained in the above two folders in subfolders if desired.

6. The routing layer

Your project may have stored all the routes covered by your application in a routes.js file. Define a file that contains all your routes, gather them in one place, and separate them from actual use.

Again, this approach may be sufficient for small projects. The problem is, you can’t predict how big your application will grow. This is why you should define a layer designed to contain all your routing files and logic. The trick to easily splitting your routes.js file into many files is to create one for each layer of routing paths used by your application.

A routing layer sample consisting of four files.

7. Practical layer

This is a place to store all the custom utilities that your entire code base depends on. While you may store your functions in a single utilis.js file, you may be forced to split it into multiple files as your project grows in size.

That’s why there’s a dedicated layering, which is probably unavoidable. Rather than wait for this to happen, get ahead and create a new layer, as shown below.

Split an initial utils.js file into two different files.

8. The view layer

This is the top layer where you will store all the pages your application contains. Note that this is the only layer where you might need to import files from any of the other layers mentioned earlier.

If your project has a large number of pages, you should organize them into subfolders. A good criterion for achieving this goal is to copy routing structures, as in PHP projects. All share [https://yourdomain.dom/users/] (https://yourdomain.dom/users/), for example, the basic path of the page should be placed on the Users folder.

A sample of the standard view layer structure presented in the example.

The remaining files in the multi-level structure

Keep in mind that this structure may not cover all the files that make up your code base. That’s why there are always some extra files that you may not know where to put.

By following such an organized structure, you can reduce them to a few places. A good rule of thumb is that if you have fewer than three files, you should leave them in the top folder of your project, or wherever you think they belong. Otherwise, you can always design new layers to suit your needs.

conclusion

In this article, we looked at how (and why you should) organize a React project to make it more efficient. By placing your files in folders that represent multiple layers of architecture, you can turn your project into a more organized structure.

With this approach, each file has its own location. This will enable you and your team to set architectural standards, make your entire code base more robust and maintainable, and simplify your development process.

Thanks for reading! I hope you found this article helpful. If you have any questions, comments or suggestions, please feel free to contact me.

The postOptimize React apps using a multi-layer structureappeared first onLogRocket Blog.