My favourite React (Native) project architecture
15 January 2022I have seen and used lots of different architectures for a React (Native) project, but I prefer the modular one which evolved a little in my hands, as of January 2022. Here is a quick overview.
Based on experience
Throughout the years, I have seen many architectures and had various React projects, built multiple apps, big and small. Most of them used pretty standard, e.g.Β based on routes, file types or features. But from the experience, the modular architecture has served me very well.
Overview
π‘Note: my approach works for both React and React Native projects, but there are little differences.
π‘Note: I will be using the βstructureβ and βarchitectureβ words interchangeably for simplicity.
Of course, nowadays, the architecture and file structure is rather dictated by the stack and technologies used. For example, Gatsby.js or Next.js propose their own way of doing a React project. Redux gives distinction between Container (smart) and Presentational (dumb) components. That is totally fine too, but with this, the structure is not universal enough, especially if you want to migrate to React Hooks, for example.
For a bare React or React Native project, the modular architecture works well because the project becomes very testable, composable, without any extra complexity. So there is no learning curve for it. You can get it, just opening the repository. It is a good one because it also becomes easy to maintain and develop new features. And advanced techniques can play well too - e.g.Β module splitting. For instance, if the user is not logged in, there is no need to load and parse all available modules which are not reachable without logging in.
My vision of the architecture is a bit customised and adapted for even greater universality and integrity. And what is important is that it is based on continuous iterations.
Letβs take a look at such a project, it is a React Native project, but it works nice for web projects too, taking into account the platform differences, of course. There are some considerations underneath. Look (1)
, (2)
etc. for footnotes.
I have been using TypeScript for all projects by default, so the file extensions are in accordance.
- The
__generated__/
folder can be used for Relay or GraphQL generated types, for example. - The
jest
can also be used for__fixtures__/
or test helpers. Mock implementations should ideally be placed here in__mocks__/
, in the same folder at the node_modules level. - The
components/
folder can optionally have theui/
folder which should contain all the universal components and some other folders for more complex components, for exampleForm
which is also universal but consists of smaller components. This is very similar to atoms/molecules/organisms principle. - There are many ways to structure Redux specific files, for example, based on modules or based actions/reducers. Ideally this folder structure should follow principles, mentioned in this article, but also refer to Redux docs for details.
- GraphQL allows defining multiple schemas and use them in the app, that is why remote/local separation is helpful. The remote schema is used for backend API and the local one can complement the remote one and can be used for typing and saving search filters or combined types, for example. So next time when implementing a type/model which, letβs say, consists of 2 partial backend types, consider using the local schema for that. Together with GraphQL codegen, it is a very powerful tool. A few examples:
6. The utils/
folder can be structured freely to fit your needs. If a helper is a big one, consider a dedicated file for it. If the helpers take just a few lines, they can go all in one relevant file.
Considerations
This also shows common naming for files. Ideally, each folder should contain the entry file, usually called index.ts
. Other filesβ names in the folder should start with the module name. It helps to read the code even better. When importing, strive to preserve the full name.
If you find yourself using a module-specific component in different places/modules throughout the app, it should be made universal and moved to common components then.
Each module might not have the entry file because all screens are typically of equal importance, that is totally fine.
If the project has some screens which are shared and used in different modules, for example LoadingScreen
, consider creating a module called common/
.
For my latest projects, I avoid using Redux altogether because it increases complexity exponentially. React Hooks or Context are a very concise and expressive way, so Container components become redundant.
Conclusion
It goes without saying, the structure might differ depending on specific needs of the app. Feel free to extend or adapt it even further to fit your app needs. If in doubt, refer to one of the following JavaScript style guides or librariesβ best practices.
- https://github.com/airbnb/javascript - from Airbnb
- https://github.com/airbnb/javascript/tree/master/react - from Airbnb for React
- https://standardjs.com
When you introduce a convention or a style guide rule, it is more important to follow it consistently than the rule itself. Each team member should stick to it after the rule has been introduced. Later refactoring would be a dream in this case.