Skip to content

Commit

Permalink
feat: rewrite the overview (#685)
Browse files Browse the repository at this point in the history
  • Loading branch information
illright authored Jun 27, 2024
1 parent 1b79e12 commit 53b06c1
Show file tree
Hide file tree
Showing 4 changed files with 276 additions and 204 deletions.
102 changes: 0 additions & 102 deletions i18n/en/docusaurus-plugin-content-docs/current/get-started/overview.md

This file was deleted.

Original file line number Diff line number Diff line change
@@ -0,0 +1,138 @@
---
sidebar_position: 1
---

# Overview

**Feature-Sliced Design** (FSD) is an architectural methodology for scaffolding front-end applications. Simply put, it's a compilation of rules and conventions on organizing code. The main purpose of this methodology is to make the project more understandable and stable in the face of ever-changing business requirements.

Apart from a set of conventions, FSD is also a toolchain. We have a [linter][ext-steiger] to check your project's architecture, [folder generators][ext-tools] through a CLI or IDEs, as well as a rich library of [examples][examples].

## Is it right for me? {#is-it-right-for-me}

FSD can be implemented in projects and teams of any size. It is right for your project if:

- You're doing **frontend** (UI on web, mobile, desktop, etc.)
- You're building an **application**, not a library

And that's it! There are no restrictions on what programming language, UI framework, or state manager you use. You can also adopt FSD incrementally, use it in monorepos, and scale to great lengths by breaking your app into packages and implementing FSD individually within them.

If you already have an architecture and you're considering a switch to FSD, make sure that the current architecture is **causing trouble** in your team. For example, if your project has grown too large and inter-connected to efficiently implement new features, or if you're expecting a lot of new members to join the team. If the current architecture works, maybe it's not worth changing. But if you do decide to migrate, see the [Migration][migration] section for guidance.

## Basic example {#basic-example}

Here is a simple project that implements FSD:

- `πŸ“ app`
- `πŸ“ pages`
- `πŸ“ shared`

These top-level folders are called _layers_. Let's look deeper:

- `πŸ“‚ app`
- `πŸ“ routes`
- `πŸ“ analytics`
- `πŸ“‚ pages`
- `πŸ“ home`
- `πŸ“‚ article-reader`
- `πŸ“ ui`
- `πŸ“ api`
- `πŸ“ settings`
- `πŸ“‚ shared`
- `πŸ“ ui`
- `πŸ“ api`

Folders inside `πŸ“‚ pages` are called _slices_. They divide the layer by domain (in this case, by pages).

Folders inside `πŸ“‚ app`, `πŸ“‚ shared`, and `πŸ“‚ pages/article-reader` are called _segments_, and they divide slices (or layers) by technical purpose, i.e. what the code is for.

## Concepts {#concepts}

Layers, slices, and segments form a hierarchy like this:

<figure>
![Hierarchy of FSD concepts, described below](/img/visual_schema.jpg)

<figcaption style={{ fontStyle: "italic", fontSize: "0.9em" }}>
<p>Pictured above: three pillars, labeled left to right as "Layers", "Slices", and "Segments" respectively.</p>
<p>The "Layers" pillar contains seven divisions arranged top to bottom and labeled "app", "processes", "pages", "widgets", "features", "entities", and "shared". The "processes" division is crossed out. The "entities" division is connected to the second pillar "Slices" in a way that conveys that the second pillar is the content of "entities".</p>
<p>The "Slices" pillar contains three divisions arranged top to bottom and labeled "user", "post", and "comment". The "post" division is connected to the third pillar "Segments" in the same way such that it's the content of "post".</p>
<p>The "Segments" pillar contains three divisions, arranged top to bottom and labeled "ui", "model", and "api".</p>
</figcaption>
</figure>

### Layers {#layers}

Layers are standardized across all FSD projects. You don't have to use all of the layers, but their names are important. There are currently seven of them (from top to bottom):

1. App\* β€” everything that makes the app run β€” routing, entrypoints, global styles, providers.
2. Processes (deprecated) β€” complex inter-page scenarios.
3. Pages β€” full pages or large parts of a page in nested routing.
4. Widgets β€” large self-contained chunks of functionality or UI, usually delivering an entire use case.
5. Features β€” _reused_ implementations of entire product features, i.e. actions that bring business value to the user.
6. Entities β€” business entities that the project works with, like `user` or `product`.
7. Shared\* β€” reusable functionality, especially when it's detached from the specifics of the project/business, though not necessarily.

_\* β€” these layers, App and Shared, unlike the other layers, don't have slices, and are made up of segments directly._

The trick with layers is that modules on one layer can only know about and import from modules from the layers strictly below.

### Slices {#slices}

Next up are slices, which partition the code by business domain. You're free to choose any names for them, and create as many as you wish. Slices make your codebase easier to navigate by keeping logically related modules close together.

Slices cannot use other slices on the same layer, and that helps with high cohesion and low coupling.

### Segments {#segments}

Slices, as well as layers App and Shared, consist of segments, and segments group your code by its purpose. Segment names are not constrained by the standard, but there are several conventional names for the most common purposes:

- `ui` β€” everything related to UI display: UI components, date formatters, styles, etc.
- `api` β€” backend interactions: request functions, data types, mappers, etc.
- `model` β€” the data model: schemas, interfaces, stores, and business logic.
- `lib` β€” library code that other modules on this slice need.
- `config` β€” configuration files and feature flags.

Usually these segments are enough for most layers, you would only create your own segments in Shared or App, but this is not a rule.

## Advantages {#advantages}

- **Uniformity**
Since the structure is standardized, projects become more uniform, which makes onboarding new members easier for the team.

- **Stability in face of changes and refactoring**
A module on one layer cannot use other modules on the same layer, or the layers above.
This allows you to make isolated modifications without unforeseen consequences to the rest of the app.

- **Controlled reuse of logic**
Depending on the layer, you can make code very reusable or very local.
This keeps a balance between following the **DRY** principle and practicality.

- **Orientation to business and users needs**
The app is split into business domains and usage of the business language is encouraged in naming, so that you can do useful product work without fully understanding all other unrelated parts of the project.

## Incremental adoption {#incremental-adoption}

If you have an existing codebase that you want to migrate to FSD, we suggest the following strategy. We found it useful in our own migration experience.

1. Start by slowly shaping up the App and Shared layers module-by-module to create a foundation.

2. Distribute all of the existing UI across Widgets and Pages using broad strokes, even if they have dependencies that violate the rules of FSD.

3. Start gradually resolving import violations and also extracting Entities and possibly even Features.

It's advised to refrain from adding new large entities while refactoring or refactoring only certain parts of the project.

## Next steps {#next-steps}

- **Want to get a good grasp of how to think in FSD?** Check out the [Tutorial][tutorial].
- **Prefer to learn from examples?** We have a lot in the [Examples][examples] section.
- **Have questions?** Drop by our [Telegram chat][ext-telegram] and get help from the community.

[tutorial]: /docs/get-started/tutorial
[examples]: /examples
[migration]: /docs/guides/migration/from-legacy
[ext-steiger]: https://github.com/feature-sliced/steiger
[ext-tools]: https://github.com/feature-sliced/awesome?tab=readme-ov-file#tools
[ext-telegram]: https://t.me/feature_sliced

Loading

0 comments on commit 53b06c1

Please sign in to comment.