Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Update filename conventions #17

Merged
merged 4 commits into from
Jan 10, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
70 changes: 0 additions & 70 deletions docs/nuxt-client/1_CodeConventions.md

This file was deleted.

142 changes: 142 additions & 0 deletions docs/nuxt-client/1_ProjectStructure.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,142 @@
---
sidebar_position: 2

toc_min_heading_level: 2
toc_max_heading_level: 4
---

# Project Structure

## Filenames

Files should be consistently named like this:

| file content | style | filename | comment |
| ------------ | ---------- | ---------------------------------- | --------------------------------------------------------- |
| Components | PascalCase | `YourComponent.vue` | best practice |
| Pinia stores | PascalCase | `BoardCard.store.ts` | placed in data- modules |
| Composables | CamelCase | `confirmationDialog.composable.ts` | current convention |
| Utils | CamelCase | `yourArea.util.ts` | suggestion: move from yourUtil.ts => your.util.ts |
| other files | CamelCase | `yourFilename.ts` | |
| test files | | `{{ basename }}.unit.ts` | e.g. YourComponent**.unit.ts**, yourArea.util**.unit.ts** |

## Folders

Folders are written in **KebabCase** (e.g. `feature-board`).

## Building Blocks

The project's code is separated into building blocks.

### What is a building-block?

A **building-block** is a "container" were we place most of our applications logic and components. Each building-block is defined by an `index.ts (Barrel-File)` describing it's exported content (public API of a building-block) and a `type`.

Utilizing linting rules and the index.ts we can ensure that each building-block only exposes files which are meant to be used application-wide. This way we achieve a strong separation of concern across the whole application.

Our linting rule is based on the following concept: [Enforce Project Boundaries | Nx](https://nx.dev/core-features/enforce-project-boundaries)

_Note: in above documentation **libraries** are equivalent to building-blocks and **tags** represent the types defined below._

**near future**: All newer modules, that already follow our naming convention (see link above), will move from "components/" into a central "modules/" folder with one subfolder for each type of module (`page/`, `data/`, `feature/`, `ui/`, `util/` ).

### Types of building-blocks

There are different types of building blocks each with a different purpose.

| type | example | comment |
| ------- | ----------------------------------------------------- | -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
| Page | modules / **`page`** / dashboard | Contains a subpage of the application. Orchestrates Feature and UI building-blocks. |
| Feature | modules / **`feature`** / calendar | Complex features with **stateful / smart components**. Usually specialized to fulfill specific roles in the App. Can also contain presentational components that are specialized for this feature. |
| UI | modules / **`ui`** / forms | **Stateless / presentational components** which get their data via props and emit events. Usually less specialized. |
| Data | modules / **`data`** / auth | State and API-access. Does not contain any visual components. They are the data-sources of all smart components. |
| Util | <nobr> modules / **`util`** / form-validators </nobr> | Contains shared low-level code. |

_**Hint**: currently the modules are all placed under /components/feature-... etc. and will be refactored to the above with this ticket: [BC-5513](https://ticketsystem.dbildungscloud.de/browse/BC-5513)_

#### Matrix of allowed imports

Each type is only allowed to import modules of some of the other types.

| &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; Allowed to Import &#8594; <br/> It is &#8595; | page | feature | data | ui | util |
| :------------------------------------------------------------------------------------------------------------------------------------------------------------ | ---- | ------- | ---- | --- | ---- |
| page | | ✔ | ✔ | ✔ | ✔ |
| feature | | ✔ | ✔ | ✔ | ✔ |
| data | | | ✔ | | ✔ |
| ui | | | | ✔ | ✔ |
| util | | | | | ✔ |

#### Type: Page

A page building-block represents a **subpage** of the application. It contains the layout component and orchestrates feature and ui building blocks to create a subpage. It can not be imported into any other type of building-block. It is **only imported by the vue-router** and should be **lazy-loaded** if possible.

#### Type: Feature

A feature building-block contains a set of files that represent a business use case in an application.

Most of the components of features are **stateful / smart components** that interact with data sources. This type also contains most of the UI logic, form validation code, etc.

#### Type: UI

A ui building-block mainly contains **Stateless / presentational components** which are used all across the application. They don't have access to stores and do not use features in their templates. All data needed for components in ui building-blocks comes from props.

#### Type: Data

A data building-block contains **stores and api-services**. It does not contain any view components. They serve as data-sources for feature and page building blocks.

#### Type: Util

A utility building-block contains **low level code** used by many building-blocks. Often there is no framework-specific code and the building-block is simply a collection of types, utilities, pure functions, factories or composables.

Placed in folder **util**

### Life of a feature module

#### Starting a new Feature module

1. create a new subfolder inside of `modules/feature/`-folder and give it a speaking name
2. place any files of different purposes (for ui, data and/or util) inside of it.
3. provide a barrel-file **index.ts** that defines it's exposed API - the functionality that other modules are allowed to use. This should only export the bare minimum.

#### Another module needs access to the data of your feature

1. do not export your data-functionality from your feature
2. refactor your feature-module and extract the data part of it into a new data-module
3. use this new data-module in the original feature and the other module that also needs access

_Note: if you know upfront, which parts will definitly be used outside your feature, implement the data-module upfront_

#### Another module needs access to some of your UI-Components that form your feature

1. do not export these components simply from your feature
2. if the components are simple / dumb and do not need data-access: consider extracting them into a separate **ui-module** _(example: ui/preview-image)_
3. if the components are coupled with some data-functionality but this does not belong to the core of your feature: consider extracting those parts into a separate **feature-module** _(example: feature/alert-list)_

_Note: if you are sure that a part of your feature, will be reused in the project - consider extracting it from your ticket - before implementation - and implement it first._

### How to pick the correct type for my Task

`To render this graph in VS-Code markdown preview install this extension: bierner.markdown-mermaid`

```mermaid
flowchart TD
A[Your Task] --> B[Imagine the different requirements of the Task]
B --> C{Do I need a new subpage}
C -->|Yes| D(type: page)
C -->|No| E{Do I need UI?}
E --> |No| F{Do I need State?}
E --> |Yes| G{Do I need State?}
F -->|Yes| H(type: data)
F -->|No| I(type: util)
G -->|Yes| J(type: feature)
G -->|No| K(type: ui)
H --> L[Are all requirements of your task placed in a building-block?]
I --> L
J --> L
K --> L
D --> L
L -->|Yes| M[Happy Coding!]
L -->|No| O[You need an additional building-block]
O --> B
M -.-> P[Evaluate your choices as a part of your review and refactor when neccessary]
```
49 changes: 49 additions & 0 deletions docs/nuxt-client/2_CodeConventions.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,49 @@
---
sidebar_position: 3
---

# Code Conventions

<!-- vscode-markdown-toc -->

- [Code Conventions](#code-conventions)
- [data-testids](#data-testids)
- [ts-ignore-comments](#ts-ignore-comments)
- [composables](#composables)

<!-- vscode-markdown-toc-config
numbering=false
autoSave=true
/vscode-markdown-toc-config -->
<!-- /vscode-markdown-toc -->

## data-testids

Please use `<div ... data-testid="some-example" ...>` in your HTML-code if you want to define a data-testid.

- do not use uppercase-characters
- only use one dash - right after data

We also recommend to use **ref**s instead of data-testids. But if you do that, you need to be careful when removing them... as they could be used in the component-code AND in tests:

- [VueJs - template refs](https://vuejs.org/guide/essentials/template-refs.html)
- [VueTestUtils - ref](https://v1.test-utils.vuejs.org/api/#ref)

Also look here: _Frontend Arc Group: Meeting Notes 2022-11-04_

## ts-ignore-comments

Everybody should try to avoid `// @ts-ignore` and try his/her best to define the types of variables in TypeScript files.

Also look here: _Frontend Arc Group: Meeting Notes 2022-10-28_

## composables

Composables are a great way to make our code more reusable and to extract code from components. If you want to write a composable, consider using one of these well documented and well tested ones:
[VueUse - Collection of Vue Composition Utilities](https://vueuse.org/)

If you write a composable:

- it should have the extension `.composable.ts`
- should be placed in your feature folder (see section "directory structure" above), if it is only used inside of your feature
- should be placed in the global folder `/ src / composables`, if it is used in multiple features
Loading