Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/juntossomosmais/frontend-guideline
📝 Front-end Guideline by Juntos Somos Mais
https://github.com/juntossomosmais/frontend-guideline
code-style coding-style css frontend frontend-guide frontend-guidelines good-practices hacktoberfest html javascript patterns patterns-design react typescript vue
Last synced: 4 days ago
JSON representation
📝 Front-end Guideline by Juntos Somos Mais
- Host: GitHub
- URL: https://github.com/juntossomosmais/frontend-guideline
- Owner: juntossomosmais
- License: mit
- Created: 2021-09-01T20:08:00.000Z (over 3 years ago)
- Default Branch: main
- Last Pushed: 2024-07-15T18:45:50.000Z (6 months ago)
- Last Synced: 2025-01-03T08:02:59.192Z (11 days ago)
- Topics: code-style, coding-style, css, frontend, frontend-guide, frontend-guidelines, good-practices, hacktoberfest, html, javascript, patterns, patterns-design, react, typescript, vue
- Homepage: https://juntossomosmais.github.io/frontend-guideline/
- Size: 512 KB
- Stars: 1,142
- Watchers: 44
- Forks: 50
- Open Issues: 1
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
- stars - juntossomosmais/frontend-guideline - 📝 Front-end Guideline by Juntos Somos Mais (Others)
README
![juntos-somos-devs](https://user-images.githubusercontent.com/3603793/131751022-fda4146c-9ada-4ad0-82fc-d8f0a73acd3f.png)
# Front-end - Guideline
[![GitHub contributors](https://img.shields.io/badge/ghpages-online-brightgreen.svg)](https://juntossomosmais.github.io/frontend-guideline/)
[![GitHub contributors](https://img.shields.io/github/contributors/juntossomosmais/frontend-guideline.svg)](https://github.com/juntossomosmais/frontend-guideline/graphs/contributors)> "Every line of code should appear to be written by a single person, no matter the number of contributors." - Chinese Proverb.
The following document describes generic rules of writing in development languages that we use on our Front-end projects, that HTML, CSS, JavaScript, React, and Vue
The idea of this repository is not to be a complete guideline, the target is just to help developers who participate in our projects to be able to inform the coding standards used.
As this is a live document, some rules may not have been applied in old projects and changes can occur at any time.
## We are hiring! 🔥
If you are looking opportunities as Front-end Developer we are hiring!
You can [check all our job opportunities](https://www.linkedin.com/company/juntos-somos-mais/jobs/) and apply if you like it 😁
This is our [Front-end Challenge](https://github.com/juntossomosmais/frontend-challenge)
## 📖 Summary
1. [General Code Patterns](#1-general-code-patterns)
2. [Architecture](#2-architecture)
3. [Git](#3-git)
4. [HTML](#4-html)
5. [CSS](#5-css)
6. [JavaScript](#6-javascript)
7. [React](#7-react)
8. [Vue](#8-vue)
9. [Storybook](#9-storybook)
10. [Testing](#10-testing)
11. [Typescript](#11-typescript)---
## 1. General Code Patterns
- 1.0 [Prettier](#10-prettier)
- 1.1 [Code Syntax](#11-code-syntax)
- 1.2 [Refactoring](#12-refactoring)
- 1.3 [Imports](#13-imports)### 1.0 Prettier
We use [Prettier](https://prettier.io/) to format our code, and we have a [shared rule to validade this](https://github.com/juntossomosmais/time-out-market/tree/main/packages/linters#prettier)
### 1.1 Code Syntax
Use soft tabs with two spaces. You need to configure your editor for this.
**✅ Good:**
```js
const obj = {
prop: 'value',
prop2: 'value2',
prop3: 'value3',
}
``````css
.foo {
color: red;
}
``````html
Hello World
```**❌ Bad:**
```js
const obj = {
prop: 'value',
prop2: 'value2',
prop3: 'value3',
}
``````css
.foo {
color: red;
}
``````html
Hello World
```### 1.2 Refactoring
Refactoring makes part of JSMLover's way of being, doing it every day and task by task. We have good practices and conditions to do that, though.
```js
if(!isWholeCodeCoveraged) return
```
- We can only refactor codes that have tests (and that tests!), which means 100% coverage! This way, we can improve or code safely.- Keep the current tests and make them pass!
Once the current code is tested and can be refactored. We must make sure that the new changes will not break the current tests.### 1.3 Imports
If the data to be imported belongs to the **same module/scope**, use **relative path**.
#### Relative Path Example
`HeaderButton.js importing style from header/styles.css`
```sh
┣ 📂 src/components \
┣ ┣ 📂 header \
┣ ┃ ┣ 📂 components
┣ ┃ ┣ ┣ 📂 Buttons
┣ ┃ ┣ ┣ ┣ 📜 HeaderButton.js
┣ ┃ ┣ ┣ ┣ 📜 RedirectButton.js
┣ ┃ ┣ ┣ ┣ 📜 EspecificButton.js
┣ ┃ ┣ ┣ 📂 Card
┣ ┃ ┣ ┣ 📂 Modal
┣ ┃ ┣ 📂 __tests__
┣ ┃ 📜 index.js
┣ ┃ 📜 styles.css
┣ ┃ 📜 index.stories.mdx
┣ ┃ 📜 index.spec.js
```use this:
```js
import { HeaderButtonClass } from '../../../styles'
```If the data to be imported belongs to **another module/scope**, use an **absolute path**.
#### Absolute Path Example
`HeaderPopup.js` importing an enum from `src/enum/errors.js`
```sh
┣ 📂 src \
┣ ┣ 📂 components \
┣ ┃ ┣ 📂 header \
┣ ┃ ┃ ┣ 📂 components
┣ ┃ ┃ ┃ ┣ 📂 Card
┣ ┃ ┃ ┃ ┣ 📂 Popup
┣ ┃ ┃ ┃ ┃ ┣ 📜 HeaderPopup.js
┣ ┃ ┃ ┃ ┃ ┣ 📜 RedirectPopup.js
┣ ┃ ┃ ┃ ┃ ┣ 📜 EspecificPopup.js
┣ ┃ ┃ ┣ 📂 __tests__
┣ ┃ ┣ 📜 index.js
┣ ┃ ┣ 📜 styles.scss
┣ ┃ ┣ 📜 index.stories.mdx
┣ ┃ ┣ 📜 index.spec.js
┣ ┃ 📂 enums \
┣ ┃ ┣ 📜 errors.js
┣ ┃ ┣ 📜 pages.js
┣ ┃ ┣ 📜 routes.js
┣ ┃ ┣ 📜 environments.js
┣ ┃ ┣ 📜 index.js
```use this:
```js
import { UploadError } from '~/enums/errors.js'
```## 2. Architecture
The proper architecture for projects, and how to create and name files and folders.
- 2.1 [File Name](#21-file-name)
- 2.2 [Folder Architecture](#22-folder-architecture)### 2.1 File Name
**✅ Good:**
- `UserProfile/UserProfile.vue`
- `UserProfile/index.js`
- `UserProfile/index.ts`
- `UserProfile/styles.js`
- `UserProfile/UserProfile.scss`
- `UserProfile/UserProfile.stories.mdx`**❌ Bad:**
- `UserProfile/component.vue`
- `src/UserProfile.js`
- `UserProfile/component.ts`
- `UserProfile/style.scss`
- `UserProfileStyles.js`
- `UserProfile/UserProfile.mdx`### 2.2 Folder Architecture
#### Global Components/Helpers
Global Components should only be components used in more than one place.
For example:
```sh
┣ 📂 src/components \
┣ ┣ 📂 component \
┣ ┃ ┣ 📜 index.js
┣ ┃ ┣ 📜 styles.js
┣ ┃ ┣ 📜 index.spec.js
┣ ┃ ┣ 📜 index.stories.mdx
```#### Scoped Components
We need to add inside `pages/**/components`, for example, all components that is need used just a context or scope, like a components that be used just a some place or specific page.
If we need to used the component again in another context or page it need to be moved to `src/components`.
For example:
```sh
┣ 📂 pages \
┣ ┣ 📂 Home \
┣ ┃ ┣ 📜 Home.js \
┣ ┃ ┣ 📜 Home.style.js \
┣ ┃ ┣ 📜 Home.spec.js \
┣ ┃ ┣ 📂 components \
┣ ┃ ┃ ┣ 📂 UserProfile \
┣ ┣ ┃ ┃ ┣ 📜 UserProfile.style.js \
┣ ┣ ┃ ┃ ┣ 📜 UserProfile.spec.js \
┣ ┣ ┃ ┃ ┣ 📜 UserProfile.stories.mdx \
```#### Scoped Files
We need to add inside `pages/**/{utils, helpers, context, hooks, etc...}` and use `camelCase` as **Naming Convention**.
For example:
```sh
┣ 📂 pages \
┣ ┣ 📂 Home \
┣ ┃ ┣ 📂 utils \
┣ ┃ ┣ ┣ 📜 someUtils.js
┣ ┃ ┣ 📂 helpers \
┣ ┣ ┣ ┣ 📜 someHelper.js
┣ ┃ ┣ 📂 hooks \
┣ ┣ ┣ ┣ 📜 useSomeHook.js
```If we need use these files again in another context or page it need to be moved to `src/{utils, helpers, context, hooks}`.
```sh
┣ 📂 src
┣ ┣ 📂 utils \
┣ ┃ ┣ 📜 someUtils.js
┣ ┣ 📂 helpers \
┣ ┃ ┣ 📜 someHelper.js
┣ ┣ 📂 hooks \
┣ ┃ ┣ 📜 useSomeHook.js
```**[⬆ back to summary](#-summary)**
---
## 3. Git
- 3.0 [Commitlint](#30-git-commitlint)
- 3.1 [Commit Messages](#31-commit-messages)### 3.0 Git Commitlint
We use [Commitlint](https://commitlint.js.org/#/) to validate our commit messages, and we have a [shared rule to validade this](https://github.com/juntossomosmais/time-out-market/tree/main/packages/linters#commitlint)
### 3.1 Commit Messages
In order to facilitate the contribution of anyone in a project, all commit messages must be in **English**.
We also use [conventional commit messages](https://www.conventionalcommits.org/en/v1.0.0/), that is, the commit message must be in the form of a sentence, with the first word being an action, and the rest of the sentence a describing text.
We must always commit in lower-case. We are using a [shared rule to validade this](https://github.com/juntossomosmais/time-out-market/blob/main/packages/linters/src/commitlint.config.js).
**✅ Good:**
```powershell
git commit -m "feat: allow provided config object to extend configs"
git commit -m "docs: correct spelling of CHANGELOG"
git commit -m "feat(lang): add the Portuguese language"
```**❌ Bad:**
```powershell
git commit -m "Add placeholder on input"
```**[⬆ back to summary](#-summary)**
---
## 4. HTML
We main reference for HTML good patterns is [W3C](https://www.w3.org/TR/html/) and [MDN](https://developer.mozilla.org/en-US/docs/Web/HTML/Element), behind these docs we could learn a lot with semantic and another good practices.
- 4.1 [HTML Component Scope](#41-html-component-scope)
### 4.1 HTML Component Scope
We don't guest the scope of HTML components inside page, so when we start a new component, we should use a semantic tag, like `section` or `article` for example, to be able to starting to use the heading tags by context.
**✅ Good:**
```html
Title
Paragraph
```
**❌ Bad:**
```html
Title
Paragraph
```## 5. CSS
The tips above could be used in any CSS framework or preprocessor, like SCSS, Styled Components and etc
- 5.0 [CSS Stylelint](#50-css-stylelint)
- 5.1 [CSS Code Syntax](#51-css-syntax)
- 5.2 [CSS Declaration Order](#52-css-declaration-order)
- 5.3 [CSS Class Names](#53-css-class-names)
- 5.4 [CSS Good Practices](#54-css-good-practices)
- 5.5 [CSS Media Queries](#55-css-media-queries)
- 5.6 [Spacing and size of image and components](#56-spacing-and-size-of-image-and-components)
- 5.6.1 [Dynamic values](#561-dynamic-values)
- 5.6.2 [Images and well defined components](#562-images-and-well-defined-components)
- 5.7 [Avoid using shorthand properties](#57-avoid-using-shorthand-properties)### 5.0 CSS Stylelint
We use [Stylelint](https://stylelint.io/) to validate our code, and we have a [shared rule to validade this](https://github.com/juntossomosmais/time-out-market/tree/main/packages/linters#stylelint)
### 5.1 CSS Syntax
Keep one declaration per line.
**✅ Good:**
```scss
.selector-1,
.selector-2,
.selector-3 {
...
}
```**❌ Bad:**
```scss
.selector-1, .selector-2, .selector-3 {
...
}
```Separate each ruleset by a blank line.
**✅ Good:**
```scss
.selector-1 {
...
}.selector-2 {
...
}
```**❌ Bad:**
```scss
.selector-1 {
...
}
.selector-2 {
...
}
```Use lowercase and avoid specifying units is zero-values.
**✅ Good:**
```scss
.selector-1 {
color: #aaa;
margin: 0;
}
```**❌ Bad:**
```scss
.selector-1 {
color: #aaaaaa;
margin: 0px;
}
```### 5.2 CSS Declaration Order
The declarations should be added in alphabetical order.
**✅ Good:**
```scss
.selector {
background: #fff;
border: #333 solid 1px;
color: #333;
display: flex;
height: 200px;
margin: 5px;
padding: 5px;
width: 200px;
}
```**❌ Bad:**
```scss
.selector {
padding: 5px;
height: 200px;
background: #fff;
margin: 5px;
width: 200px;
color: #333;
border: #333 solid 1px;
display: flex;
}
```### 5.3 CSS Class Names
Keep class lowercase and use dashes to separate the classname.
**✅ Good:**
```scss
.page-header { ... }
```**❌ Bad:**
```scss
.pageHeader { ... }
.page_header { ... }
```Is a good idea follows a [BEM naming convention](http://getbem.com/introduction/) to avoid conflicts with other components. If you are using CSS-in-JS like a Styled-Component, you can use BEM if you need to nesting elements inside parent.
The main pattern is use single dash to element name, double underline to element block and double dash to style modification.
**✅ Good:**
```scss
/* Good */
.page-header__title { ... }
.page-header--active { ... }.button--active { ... }
```**❌ Bad:**
```scss
.page-header-title { ... }
.page-header-active { ... }.active { ... }
.primary { ... }
```Dashes and underline serve as natural breaks in related class. Prefix class based on the closest parent or base class.
**✅ Good:**
```scss
.nav { ... }
.nav__item { ... }
.nav__link { ... }
```**❌ Bad:**
```scss
.item-nav { ... }
.link-nav { ... }
```Avoid giving too short names for class and always choose meaningful names that provide the class function.
**✅ Good:**
```scss
/* Good */
.button { ... }
.page-header { ... }
.progress-bar { ... }
```**❌ Bad:**
```scss
.s { ... }
.btn { ... }
.ph { ... }
.block { ... }
```### 5.4 CSS Good Practices
Avoid use values like colors, spacing and etc directly in the elements, use variables instead, and it can be CSS variables or some preprocessor variables, always check the context.
**✅ Good:**
```scss
.button {
color: var(--color-primary);
padding: var(--space-sm);
}
```**❌ Bad:**
```scss
.button {
color: #333;
padding: 16px;
}
```Never use IDs to style elements, always use classes instead.
**✅ Good:**
```scss
.header { ... }
.section { ... }
```**❌ Bad:**
```scss
#header { ... }
#section { ... }
```Do not style directly the elements, it will create a lot of conflicts, always use classes instead.
**✅ Good:**
```scss
.form-control { ... }
.header { ... }
.section { ... }
```**❌ Bad:**
```scss
input[type="text"] { ... }
header
section
```Avoid nesting elements, because it decrease performance and increase the specificity of the CSS, always use classes instead.
**✅ Good:**
```scss
.navbar { ... }
.nav { ... }
.nav__item { ... }
.nav__link { ... }
```**❌ Bad:**
```scss
.navbar ul { ... }
.navbar ul li { ... }
.navbar ul li a { ... }
```### 5.5 CSS Media Queries
Start the development with generic rules and add media queries inside scope using mobile first. Also is important
keep the media queries as close to their relevant rule sets whenever possible.**✅ Good:**
```scss
.navbar {
margin-bottom: var(--space);@media (min-width: 480px) {
padding: 10px;
}@media (min-width: 768px) {
position: absolute;
top: 0;
left: 0;
}@media (min-width: 992px) {
position: fixed;
}
}
```**❌ Bad:**
```scss
.navbar {
position: fixed;
top: 0;
left: 0;@media (max-width: 767px) {
position: static;
padding: var(--space-sm);
}
}
```### 5.6 Spacing and size of image and components
Is a commom problem to use width and height or all dynamic or all hardcoded, but each one has it own purpose. We should avoid using magic numbers at all times.
> _"Magic numbers are those numbers that appear in code without explanation, but that 'magically' make things work."_ Are numbers that dont have a why, but works.
### 5.6.1 Dynamic values
If you are using padding, margin, gap should use our [Atomium tokens](https://juntossomosmais.github.io/atomium/?path=/docs/getting-started-tokens--docs). Any space that override it values must be validated once our Design System is well defined around these values and our UX Teams guide must follow it.
Icons, width and height that are relative to our Design System or that have sizes based on calc upon our spacing variable must also use [Atomium tokens](https://juntossomosmais.github.io/atomium/?path=/docs/getting-started-tokens--docs) instead of magic numbers.**✅ Good:**
```scss
.logout__icon {
height: var(--spacing-xxlarge);
width: var(--spacing-xxlarge);
}.icon__button {
min-width: var(--spacing-giant);
}
```**❌ Bad:**
```scss
.logout__icon {
height: 25px;
width: 25px;
}.icon__button {
min-width: 34px;
}
```### 5.6.2 Images and well defined components
If you are using a image, or a component that has a design size and it sizes at maximum vary from desktop/mobile, use the value of it:**✅ Good:**
```scss
.shopfrom__banner {
height: 900px;
width: 480px;
@media (min-width: 991px) {
height: 740px;
width: 240px;
}
}
```**❌ Bad:**
```scss
.shopfrom__banner {
height: calc(4 * var(--spaceing-xxxlarge);
width: calc(2 * var(--spacing-giant);
}.shopfrom__banner {
height: 480px;
width: 170px;
@media (max-width: 746px) {
height: 740px;
width: 240px;
}
@media (max-width: 991px) {
height: 900px;
width: 320px;
}
@media (max-width: 1024px) {
height: 980px;
width: 300px;
}
}
```## 5.7 Avoid using shorthand properties
Shorthand properties are great for reducing CSS, but they can also make the code harder to read and override. It's better to use longhand properties to make the code more readable and maintainable.
**✅ Good:**
```scss
.element {
margin-left: auto;
margin-right: auto;
}
```**❌ Bad:**
```scss
.element {
margin: 0 auto;
}
```Shorthands can be used when you want to apply the same value to multiple properties.
**[⬆ back to summary](#-summary)**
## 6. JavaScript
- 6.0 [JavaScript Eslint](#60-javascript-eslint)
- 6.1 [Javascript Code Syntax](#61-javascript-code-syntax)
- 6.2 [Variables](#62-variables)
- 6.3 [Descriptive validations (if)](#63-descriptive-validations-if)
- 6.4 [Avoid multiple if's](#64-avoid-multiple-ifs)
- 6.5 [Code Comments](#65-code-comments)
- 6.6 [Avoid errors while destructuring](#66-avoid-errors-while-destructuring)
- 6.7 [Prefer early return](#67-prefer-early-return)### 6.0 JavaScript Eslint
We use [ESLint](https://eslint.org/) to validate our code, and we have a [shared rule to validade this](https://github.com/juntossomosmais/time-out-market/tree/main/packages/linters#eslint)
### 6.1 JavaScript Code Syntax
Never use semicolons.
**✅ Good:**
```js
const foo = 'bar'
const baz = 'qux'
const func = () => {}
```**❌ Bad:**
```js
const foo = 'bar';
const baz = 'qux';
const func = () => {};
```Always use single quotes or template literals
**✅ Good:**
```js
const string = 'foo'
const template = `foo`
```**❌ Bad:**
```js
const string = "foo"
const template = "foo"
```For strict equality checks `===` should be used in favor of `==`.
**✅ Good:**
```js
if (foo === 'foo') {
statement
}
```**❌ Bad:**
```js
if (foo == 'foo') {
statement
}
```Add empty lines between blocks of code.
**✅ Good:**
```js
const foo = () => {
// do something
}const bar = () => {
// do something
}
```Add empty lines between blocks of `if` statements.
**✅ Good:**
```js
if (foo) {
// do something
}if (bar) {
// do something
}
```**❌ Bad:**
```js
if (foo) {
// do something
}
if (bar) {
// do something
}
```Add empty lines between before return statements.
**✅ Good:**
```js
const foo = () => {
const bar = 'bar'return bar
}
```**❌ Bad:**
```js
const foo = () => {
const bar = 'bar'
return bar
}
```**❌ Bad:**
```js
const foo = () => {
// do something
}
const bar = () => {
// do something
}
```Remove empty lines between groups of `const`, `let` and `var` declarations, but use empty lines between the groups.
**✅ Good:**
```js
const foo = 'foo'
const bar = 'bar'let qux = 'qux'
let quux = 'quux'
```**❌ Bad:**
```js
const foo = 'foo'const bar = 'bar'
let qux = 'qux'
let quux = 'quux'
```### 6.2 Variables
Use meaningful, pronounceable, and in **English** variable names.
**✅ Good:**
```js
const currentDate = new Date().toLocaleDateString('pt-BR')
```**❌ Bad:**
```js
const xpto = new Date().toLocaleDateString('pt-BR')
```### 6.3 Descriptive validations (if)
Creating const to describe validations.
**✅ Good:**
```js
const hasFullUserName = user.firstName && user.lastnameif (hasFullUserName) {
//do awesome something
}
```**❌ Bad:**
```js
if (user.firstName && user.lastname) {
//do something
}
```### 6.4 Avoid multiple if's
Use an execution map instead a multiple if validations.
**✅ Good:**
```js
const messagingChannels = {
whatsapp: (message) => {
// send message to whatsapp
},
email: (message) => {
// send message to email
}
}const sendMessage = (message, channel) => {
const send = messagingChannels[channel];
return send && send(message);
}
```**❌ Bad:**
```js
const sendWhatsapp = (message) => {
// send message to whatsapp
}const sendEmail = (message) => {
// send message to email
}const sendMessage = (message, channel) => {
if (channel === 'whatsapp') {
sendWhatsapp(message)
} else if (channel === 'email') {
sendEmail(message)
}
}
```### 6.5 Code Comments
Avoid writing comments to explain the code. Use comments to answer “Why?” instead “How?”. Some cases you could write a code comment: warnings, complex expressions, or unusual decision clarification.
**✅ Good:**
```js
const TIME_IN_SECONDS = 60 * 40 // 40 minutes// [email protected]
const regex = /^[a-z0-9.]+@[a-z0-9]+\.[a-z]+\.([a-z]+)?$/iconst calculateProductsPrice = () => {
// do something
}
```**❌ Bad:**
```js
// This coolFunction calculates the prices of the products
const coolFunction = () => {
// do something
}
```
### 6.6 Avoid errors while destructuringIts a common mistake destructuring while the object is null or undefined, the destructuring will throw an error.
**✅ Good:**
```js
const { age } = { ...null } // undefined
const { age } = null || {} // undefined// other values won't throw an error
const { emptyString } = '';
const { nan } = NaN;
const { emptyObject } = {};function foo(bar = {}) {
const { age } = bar;
}
foo() // undefined
```**❌ Bad:**
```js
const { age } = null // will throw an typeError
const { age } = undefined // will throw an typeError
```Prefer early return over conditional wrapping to enhance code readability and reduce nesting.
**✅ Good:**
```js
function foo() {
if (!someValidation) return // or throw error// do something here
if (!anotherValidation) return
return 'bar'
}
```**❌ Bad:**
```js
function foo() {
if (someValidation) {
// do something hereif (anotherValidation) {
return 'bar'
}
}
}
```**[⬆ back to summary](#-summary)**
---
## 7. React
- 7.1 [Keys in lists](#71-keys-in-lists)
- 7.2 [useState functional updates](#72-usestate-functional-updates)
- 7.3 [useEffect dependencies array](#73-useeffect-dependencies-array)
- 7.4 [Readable components](#74-readable-components)
- 7.5 [Styled Component Naming Convention](#75-styled-component-naming-convention)
- 7.6 [Using Styled Component in React Components](#76-using-styled-component-in-react-components)
- 7.7 [Enums](#77-avoid-compare-directly-strings)
- 7.8 [Using spread operator](#78-using-spread-operator)
- 7.9 [Conditional Rendering](#79-conditional-rendering)
- 7.9.1 [Using short circuit](#791-using-short-circuit)
- 7.9.2 [Using ternary operator](#792-using-ternary-operator)
- 7.10 [Enforce Boolean Attribute Notation in JSX](#710-enforce-boolean-attribute-notation-in-jsx)### 7.1 Keys in lists
The best way to pick a key is to use a string that uniquely identifies a list item among its siblings.
It is not recommended to use indexes for keys if the order of items can change. This can negatively affect performance and can cause problems with the component's state.
**✅ Good:**
```js
array.map((item, index) => )
```**❌ Bad:**
```js
array.map((item, index) => )
```### 7.2 useState functional updates
If the new state is calculated using the previous state, you can pass a function to setState. Thus avoiding competition between states and preventing possible bugs.
**✅ Good:**
```js
const [number, setNumber] = useState(1)return (
{number}
setNumber((prevNumber) => prevNumber + 1)}>
Increase
setNumber((prevNumber) => prevNumber - 1)}>
Decrease
)
```**❌ Bad:**
```js
const [number, setNumber] = useState(1)return (
{number}
setNumber(number + 1)}>Increase
setNumber(number - 1}>Decrease
)
```### 7.3 useEffect dependencies array
Use the useEffect dependency array to trigger side effects, and make your code cleaner.
**✅ Good:**
```js
const [page, setPage] = useState(1)useEffect(() => {
requestListUser()
// calls useEffect when page state changes
}, [page])return (
setPage((prevState) => prevState + 1)}>
Next Page
)
```**❌ Bad:**
```js
const [page, setPage] = useState(1)useEffect(() => {
requestListUser()
}, [])const requestListUser = () => {
setPage((prevState) => prevState + 1)
// ...
// any code to return user list
}return (
requestListUser()}>Next Page
)
```### 7.4 Readable components
Avoid creating very large components.
If possible divided into sub-components, improving the understanding and reading of the code.**✅ Good:**
```js
const Screen = () => (
Filter
{data.map((item) => (
))}
)
```**❌ Bad:**
```js
const Screen = () => (
Titulo
Filter
{data.map((item) => (
{item.name}
))}
)
```### 7.5 Styled Component Naming Convention
Use PascalCase as a convention in styled-components
**✅ Good:**
```js
export const CustomText = styled.p`
color: 'red'
`
```**❌ Bad:**
```js
export const customText = styled.p`
color: 'red'
`
```### 7.6 Using Styled Component in React Components
Import Styled Component as `S`
**✅ Good:**
```tsx
import * as S from './styles'const MyComponent = () => (
text example
)
```**❌ Bad:**
```tsx
import * as Style from './styles'const MyComponent = () => (
text example
)import { CustomText } from './styles'
const MyComponent = () => (
text example
)
```### 7.7 Avoid compare directly strings
When know all possible values we can use enum to achieve better readability and control.
**✅ Good:**
```tsx
const FEEDBACK = {
CORRECT: 'correct',
INCORRECT: 'incorrect',
}const MyComponent = (type) => {
const text = type === FEEDBACK.CORRECT ? '😎' : '😢'
return (
{text}
)
}
```**❌ Bad:**
```tsx
const MyComponent = (type) => {
const text = type === 'correct' ? '😎' : '😢'return (
{text}
)
}
```### 7.8 Using spread operator
When creating a component wrapper we can spread the types from our original component. That way the wrapper extends all the props from the original component automatically. This is useful to avoid creating a custom interface for our wrapper with missing props from the original component.
**✅ Good:**
```tsx
import { MenuItem, TextField } from '@mui/material';
import { TextFieldProps } from '@mui/material';export type SelectOption = { value: string; label: string, id: string, };
export type SelectProps = TextFieldProps & {
options: SelectOption[];
};const Select = ({ options, ...props }: SelectProps) => {
return (
{options.map((option) => (
{option.label}
))}
);
};
```**❌ Bad:**
```tsx
import { MenuItem, TextField } from '@mui/material';export type SelectOption = { value: string; label: string, id: string, };
export type SelectProps = {
options: SelectOption[];
disabled: boolean;
onChange: () => void;
value: string;
onBlur: () => void;
};const Select = ({
options,
disabled,
onChange,
value,
onBlur,
} : SelectProps) => {
return (
{options.map((option) => (
{option.label}
))}
);
};
```### 7.9 Conditional Rendering
#### 7.9.1 Using short circuit
when we only need to validate a logical case and return a component, we can directly use the short circuit
**✅ Good:**
```tsx
import { useState } from 'react'
import Welcome from '../components/Welcome'const HomePage = () => {
const [showWelcome, setShowWelcome] = useState(true)
return showWelcome &&
};
```**❌ Bad:**
```tsx
import { useState } from 'react'
import Welcome from '../components/Welcome'const HomePage = () => {
const [showWelcome, setShowWelcome] = useState(true)
return showWelcome ? : <>>
};
```#### 7.9.2 Using ternary operator
when we need to validate two logical cases and return a component in both cases, we can use the ternary instead of the if...else
**✅ Good:**
```tsx
import { useState } from 'react'
import Welcome from '../components/Welcome'
import Dashboard from '../components/Dashboard'const HomePage = () => {
const [showWelcome, setShowWelcome] = useState(false)
return showWelcome ? :
};
```**❌ Bad:**
```tsx
import { useState } from "react"
import Welcome from "../components/Welcome"
import Dashboard from "../components/Dashboard"const HomePage = () => {
const [showWelcome, setShowWelcome] = useState(false)if (!showWelcome) {
return
}return
};
```### 7.10 Enforce Boolean Attribute Notation in JSX
Consistently pass the value for boolean attributes in JSX to ensure clarity and readability.
**✅ Good:**
```tsx
```
**❌ Bad:**
```tsx
```
**[⬆ back to summary](#-summary)**
---
## 8. Vue
- 8.1 [Keys in lists](#81-keys-in-lists)
- 8.2 [Use Computed for real time updates](#82-use-computed-for-real-time-updates)
- 8.3 [Multi-word component names](#83-multi-word-component-names)
- 8.4 [Prop definitions](#84-prop-definitions)
- 8.5 [Vue property decorator](#85-vue-property-decorator)### 8.1 Keys in lists
The best way to pick a key is to use a string that uniquely identifies a list item among its siblings.
It is not recommended to use indexes for keys if the order of items can change. This can negatively affect performance and can cause problems with the component's state.
**✅ Good:**
```html
```
**❌ Bad:**
```html
```
### 8.2 Use Computed for real time updates
If you need listen changes at data use computeds instead of methods
**✅ Good:**
```js
computed: {
fullName(){
return `${this.name} ${this.lastName}`
}
}
```**❌ Bad:**
```js
methods: {
fullName() {
this.fullName = `${this.name} ${this.lastName}`
}
}
```### 8.3 Multi-word component names
Component names should always be multi-word, except for root App components, and built-in components provided by Vue.
This prevents conflicts with existing and future HTML elements, since all HTML elements are a single word.
**✅ Good:**
```js
export default {
name: 'TodoItem',
// ...
}
```**❌ Bad:**
```js
export default {
name: 'Todo',
// ...
}
```### 8.4 Prop definitions
In committed code, prop definitions should always be as detailed as possible, specifying at least type(s).
**✅ Good:**
```js
export default {
status: {
type: String,
required: true
}
// ...
}
```**❌ Bad:**
```js
export default {
props: ['status']
// ...
}
```### 8.5 Vue property decorator
Vue prop decorator should not be used, use Vue.extend instead
**✅ Good:**
```js
import Vue from 'vue'
export default Vue.extend({
name: 'MyComponent',
})```
**❌ Bad:**
```js
import { Component, Vue } from 'vue-property-decorator'
@Component({})
export default class MyComponent extends Vue {
name: 'MyComponent'
}```
**[⬆ back to summary](#-summary)**
---
## 9. Storybook
- 9.1 [Story file](#91-story-file)
### 9.1 Story file
Create a file with the same name of your component, or index, and with the suffix `.stories.mdx`.
**✅ Good:**
- Button.stories.mdx
- Dialog/index.stories.mdx**❌ Bad:**
- Input.mdx
- Dialog/index.mdx**[⬆ back to summary](#-summary)**
---
## 10. Testing
- 10.1 [Write tests with "it"](#101-write-tests-with-it)
- 10.2 [Using test-id](#102-using-test-id)
- 10.3 [Selecing component](#103-selecting-component)### 10.1 Write tests with "it"
Write tests with the alias "it" instead "test" method.
**✅ Good:**
```js
describe('yourModule', () => {
it('should do this thing', () => {});
});
```**❌ Bad:**
```js
describe('yourModule', () => {
test('if it does this thing', () => {});
});
```### 10.2 Using test-id
To get components during tests we use `test-id` custom html attributes with unique id and our own convention deeply inpired by the css's BEM.
To define the `test-id` to a component use the follow structure: `[page-name||component-name]__[element-type]--[type]`**✅ Good:**
- forgot-password__input--email
- header__select--cnpjList
- login__button--forgot-password**❌ Bad:**
- forgot-email-input
- header__cnpjList
- button--forgot-password### 10.3 Selecting component
To select a component in order to test a behavior of to trigger any event we must use ou `test-id` attribute to select it.
**✅ Good:**
```js
describe('yourModule', () => {
it('should do trigger click event', () => {
const button = wrapper.find('[data-testid="login__button--forgot-password"]')
});
});
```**❌ Bad:**
```js
describe('yourModule', () => {
it('should do trigger click event', () => {
const button = wrapper.find('button.btn-primary')
});
});
```**[⬆ back to summary](#-summary)**
---
## 11. Typescript
- 11.1 [Do not use any type](#111-do-not-use-any-type)
- 11.2 [Naming convention](#112-naming-convention)
- 11.3 [Exporting types](#113-exporting-types)
- 11.4 [Types within a file](#114-types-within-a-file)
- 11.5 [Increase legible](#115-increase-legible)
- 11.6 [Type or Interface](#116-type-or-interface)### 11.1 Do not use any type
Avoid using `any` type. It's best to use the type that is more specific whenever possible. Prefer to use `unknown` when necessary.
**✅ Good:**
```ts
function foo(x: unknown) {}function foo(): unknown {}
```**❌ Bad:**
```ts
function foo(x: any) {}function foo(): any {}
```### 11.2 Naming convention
For convention, use PascalCase for type names.
**✅ Good:**
```ts
type MyBeautifulType = {
name: string
age: number
}
```**❌ Bad:**
```ts
type myBeautifulType = {
name: string
age: number
}
```The same to Enum keys.
**✅ Good:**
```ts
enum UserResponse {
NotSuccess = 0,
Success = 1,
}
```**❌ Bad:**
```ts
enum UserResponse {
NOT_SUCCESS = 0,
success = 1,
}
```### 11.3 Exporting types
Do not export types/functions unless you need to use it across multiple components.
### 11.4 Types within a file
Within a file, type definitions should come first.
**✅ Good:**
```ts
// imports...type MyBeautifulType = {
name: string
age: number
}// rest of the file...
```**❌ Bad:**
```ts
// imports...// part of the file...
type MyBeautifulType = {
name: string
age: number
}// rest of the file...
```### 11.5 Increase legible
Create a type for increase legible
**✅ Good:**
```ts
type PersonType = {
name: string
age: number
birthDate: string
};const Person = ({ name, age, birthDate }: PersonType) => {
// ...
};
```**❌ Bad:**
```ts
const Person = ({
name,
age,
birthDate,
}: {
name: string,
age: number,
birthDate: string,
}) => {
// ...
};
```### 11.6 Type or Interface
We use `type` when its usage is inside the same file and `interface` when it is exported.**✅ Good:**
```ts
type ProductType = {
name: string
code: number
value: string
};export interface OrderList {
orderNumber: number
seller: string
products: ProductType[]
}
```**❌ Bad:**
```ts
interface ProductType {
name: string
code: number
value: string
};export type OrderList = {
orderNumber: number
seller: string
products: ProductType[]
}
```We follow the principle the official [TypeScript doc](https://www.typescriptlang.org/play#example/types-vs-interfaces):
> _For publicly exposed types, it's a better call to make them an interface._