Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/axept/javascript

JavaScript Style Guide
https://github.com/axept/javascript

arrow-functions code-style codestyle es6 eslint javascript linting naming-conventions prettier style-guide styleguide

Last synced: 3 days ago
JSON representation

JavaScript Style Guide

Awesome Lists containing this project

README

        

# Axept JavaScript Style Guide

Follow the original [Airbnb JavaScript Style Guide](https://github.com/airbnb/javascript).

This document describes only those rules which we decided to extend, to override or to add.

Other Style Guides:

+ [React](https://github.com/axept/javascript/tree/master/react)

TODO:

+ [ ] Make it as a shareable cards (Twitter, FB and other Social Media)
+ [ ] Apply https://google.github.io/styleguide/javascriptguide.xml

## Contents

1. [Comparison Operators & Equality](#comparison-operators--equality)
2. [Comments](#comments)
3. [Semicolons](#semicolons)
4. [Lines](#lines)
5. [Naming Conventions](#naming-conventions)
6. [Imports](#imports)
7. [Packages](#packages)
8. [Type Casting & Coercion](#type-casting--coercion)
9. [Template Strings](#template-strings)
10. [Objects](#object)
11. [Classes & Constructors](#classes--constructors)
12. [Unsorted Yet](#unsorted-yet)

## Comparison Operators & Equality

Extends: [Section 15](https://github.com/airbnb/javascript#comparison-operators--equality)


- [1.1](#comparison--no-ternary-in-expression) Ternaries should not be as a part of expression, only whole expression.
> TODO: find or make ESLint rule for this

> Why? The following example demonstrate how it could be misunderstood.

```javascript
// bad
const query = { 'forward_url': '/' }
const targetURL = 'https://qwe.com/auth?' + (typeof query === 'string') ? query : JSON.stringify(query)

// Uncaught TypeError: split is not a function
// because type of targetURL is Object
const partsOfURL = targetURL.split('?')


// may be
const query = { 'forward_url': '/' }
const queryString = typeof query === 'string' ? query : JSON.stringify(query)
const targetURL = 'https://qwe.com/auth?' + queryString


// good
const query = { 'forward_url': '/' }
const queryString = (typeof query === 'string') ? query : JSON.stringify(query)
const targetURL = 'https://qwe.com/auth?' + queryString
```

**[⬆ back to top](#contents)**

## Comments

Extends: [Section 18](https://github.com/airbnb/javascript#comments)


- [2.1](#comments--no-obvious-comments) Avoid using obvious comments.

> Why? It makes code harder to read because you can think that here is something else than you see.

```javascript
// bad - it duplicates the function name

/**
* Update hosts
*/
function updateHosts(list) {
hosts = list
}


// good - nothing bad is better than something bad
function updateHosts(list) {
hosts = list
}


// bad - it duplicates the module path
import Drawer from 'react-native-drawer'

// Components
import DrawerMenu from './components/drawer-menu'


// good - see the previous "good" explaination
import Drawer from 'react-native-drawer'
import DrawerMenu from './components/drawer-menu'


// good
if (!root.isAbstract()) {
// No need to wrap child nodes of a known concrete type.
return nextChildren;
}
```

**[⬆ back to top](#contents)**

## Semicolons

Extends: [Section 21](https://github.com/airbnb/javascript#semicolons)


- [3.1](#semicolons--nope) Don't use semicolons, in the presence of excellent automation tools.
> ESLint rule: [semi](http://eslint.org/docs/rules/semi)

> Why? Since we always have linting tools we can detect dangerous situations automatically. But this style makes us really faster according to a few years experience.

**[⬆ back to top](#contents)**

## Lines


- [4.1](#lines--width) 80 characters is good. But feel free to use up to 100 characters in a line.
> ESLint rule: [max-len](http://eslint.org/docs/rules/max-len)

> Why? Almost all developers used to 80 characters. It's a very old practice. Now we have a wide screens and we used to seat a little bit more far from monitor. In this case we can use a bit more characters and code will be still readable.

Read [this](http://softwareengineering.stackexchange.com/questions/312889/how-to-follow-the-80-character-limit-best-practice-while-writing-source-code) and [this](http://stackoverflow.com/questions/578059/studies-on-optimal-code-width) for more details.

```javascript
// bad
if ((this.props.suggestions.length === 0) && (newProps.suggestions.length > 0) && (newProps.autosuggestValue.length > 0) {

return
}

// good
const hasSuggestions = (this.props.suggestions.length === 0)
const willHaveSuggestions = (newProps.suggestions.length > 0)
if (hasSuggestions && willHaveSuggestions && (newProps.autosuggestValue.length > 0)) {
return
}

// good
const hasSuggestions = (this.props.suggestions.length === 0)
const willHaveSuggestions = (newProps.suggestions.length > 0)
const willHaveAutosuggest = (newProps.autosuggestValue.length > 0)
if (hasSuggestions && willHaveSuggestions && willHaveAutosuggest) {
return
}
```


- [4.2](#lines--excess-line) Don't use excess empty line
> ESLint rule: [no-unexpected-multiline](http://eslint.org/docs/rules/no-unexpected-multiline), [padded-blocks](http://eslint.org/docs/rules/padded-blocks)

```javascript
// bad
const config = loadConfig()

if (config) {
applyConfig(config)
}


// good
const config = loadConfig()
if (config) {
applyConfig(config)
}
```


- [4.3](#lines--function-declaration) One empty line before and after function or class declaration.
> ESLint rule: [no-unexpected-multiline](http://eslint.org/docs/rules/no-unexpected-multiline)

> Why? No empty line needed after function or class declaration - if the following line is javascript brackets.

```javascript
// bad

const x = 0
function a() {
return 1
}
const b = () => 2


// bad

const x = 0
class A {
a() {
return 1
}
b() {
return 2
}
}


// good

const x = 0

function a() {
return 1
}

const b = () => 2


// good

const x = 0

class A {

a() {
return 1
}

b() {
return 2
}
}
```

**[⬆ back to top](#contents)**

## Naming Conventions

Extends: [Section 22](https://github.com/airbnb/javascript#naming-conventions)


- [5.1](#naming--no-reduced-names) Avoid using reduced names.

> Why? It makes code harder to understand. Names are for readability, not to appease a computer algorithm.

```javascript
// bad - what does it mean "ref"? referals? references? single reference?
const ref1 = fetch('/api/books')


// good
const reference1 = fetch('/api/books')
```

- [5.2](#naming--do-not-override-globals) Do not override globals.

```javascript
// bad
import fetch from './fetch-with-credentials'

// good
import fetchWithCredentials from './fetch-with-credentials'

// bad
function alert(message) {
console.error(message)
}

// good
function alertToConsole(message) {
console.error(message)
}
```


- [5.3](#naming--acronyms-and-initialisms) Acronyms and initialisms should always be all capitalized.

> Why? Names are for readability, not to appease a computer algorithm.

```javascript
// bad
import SmsContainer from './containers/SmsContainer'

// bad
const HttpRequests = [
// ...
]

// good
import SMSContainer from './containers/SMSContainer'

// good
const HTTPRequests = [
// ...
]

// best
import TextMessageContainer from './containers/TextMessageContainer'

// best
const requests = [
// ...
]
```

**[⬆ back to top](#contents)**

## Imports


- [6.1](#imports--order) Use the following modules import order:

+ React
+ Relay
+ Redux
+ Tools from node_modules
+ Tools from current repo
+ Other components
+ Any kind of static assets
+ Styles
+ Constants

**[⬆ back to top](#contents)**

## Packages


- [7.1](#packages--lodash-and-ramda) Try to avoid using Lodash and Ramda.

> Why? Underscore and Lodash has been made when ES6 has not been so popular. Ramda do not have performance benefits, but being opinionated custom approach increase unnecessary requirements for Developers skills. Besides of that Ramda did not stay "standard de-facto".

**[⬆ back to top](#contents)**

## Type Casting & Coercion

Extends: [Section 22](https://github.com/airbnb/javascript#type-casting--coercion)


- [8.1](#coercion--implicit-coercion) Do not use JavaScript "magic".
> ESLint rule: [no-implicit-coercion](http://eslint.org/docs/rules/no-implicit-coercion), [no-bitwise](http://eslint.org/docs/rules/no-bitwise)

> It makes source code more portable to other languages and available for reading by other developers.

```js
// bad
const b = !!a

// bad
const b = (~a.indexOf('k'))

// bad
const c = !!d.length

// good
const b = Boolean(a)

// good
const b = (a.indexOf('k') >= 0)

// good
const c = (d.length > 0)
```

**[⬆ back to top](#contents)**

## Template Strings

Extends: [Section 6](https://github.com/airbnb/javascript#es6-template-literals)


- [9.1](#template-strings--call-function) Don't call function inside template string.
> TODO: find or make ESLint rule for this

```javascript
// bad
const targetURL = `${getHost()}/search/?query=${query}`


// good
const targetHost = getHost()
const targetURL = `${targetHost}/search/?query=${query}`
```

**[⬆ back to top](#contents)**

## Objects

Extends: [Section 3](https://github.com/airbnb/javascript#objects)


- [10.1](#objects--expression-assignment) Do not mix expression in one-line objects as assignment.
> TODO: find or make ESLint rule for this

```js
// bad
this.setState({ hasErrors: true, route })


// bad
const state = { hasErrors: true, route }


// good
this.setState({ hasErrors, route })


// good
this.setState({ hasErrors: true })


// good
const state = { hasErrors, route }
```

**[⬆ back to top](#contents)**

## Classes & Constructors

Extends: [Section 9](https://github.com/airbnb/javascript#classes--constructors)


- [11.1](#classes--static) Static members should be declared in class body
> TODO: find or make ESLint rule for this

```javascript
// bad

Profile.propTypes = {
classes: PropTypes.object.isRequired,
handleClick: PropTypes.func,
}


// good

class Profile extends Component {

static propTypes = {
classes: PropTypes.object.isRequired,
handleClick: PropTypes.func,
}

// ...
}
```

**[⬆ back to top](#contents)**

## Unsorted Yet


- [12.1](#unsorted--brackets) Use brackets for expressions.
> TODO: find or make ESLint rule for this

```javascript
// bad
const useSrcKey = _srcKey !== undefined || this._srcKey !== undefined
const componentAsync = typeof src === 'function' ? src() : import(src)
const isProduction = process.env['NODE_ENV'] === 'production'

// good
const useSrcKey = (_srcKey !== undefined) || (this._srcKey !== undefined)
const componentAsync = (typeof src === 'function') ? src() : import(src)
const isProduction = (process.env['NODE_ENV'] === 'production')
```


- [12.2](#unsorted--dot-syntax) Use dot syntax only for internal objects.
> TODO: find or make ESLint rule for this

Seperate importable/exportable JSON data objects and internal runtime objects.

Also it means - do not mix accessing to values by dot and by string.

> [Google Closure Advanced Compilation](https://developers.google.com/closure/compiler/docs/api-tutorial3):
> Whenever possible, use dot-syntax property names rather than quoted strings. Use quoted string property names only when you don't want Closure Compiler to rename a property at all. For example, to export a property you must use a quoted string. However, for properties used only within your compiled code, use dot syntax.

```javascript
// bad
fetch('/api/users')
.then(response => response.json())
.then(json => {
if (!json.errors) {
json.data.users.forEach(user => {
const { id, name } = user
if (users[id].name !== name) {
setUserName(id, name)
}
})
}
})

// bad
const a = { b: 1 }
console.log(a['b'])

// bad
const a = { 'b': 1 }
console.log(a.b)

// good
fetch('/api/users')
.then(response => response.json())
.then(json => {
if (!json['errors']) {
json['data']['users'].forEach(user => {
const id = user['id']
const name = user['name']
if (users[id].name !== name) {
setUserName(id, name)
}
})
}
})

// good
const a = { b: 1 }
console.log(a.b)

// good
const a = { 'b': 1 }
console.log(a['b'])
```

It also means you can't rename stringifed keys because they are not in your code base only - in external API, data files, etc.


- [12.3](#unsorted--jsx-js) Don't mix JS and JSX in one line.
> TODO: find or make ESLint rule for this

```javascript
// bad
return


// bad
const body = items.map(it =>

  • {it.title}
  • )


    // good
    return (

    );


    // good
    const body = items.map(it => (
  • {it.title}

  • ))


    // good
    return (

    {items.map((it, index) => (

    ))}

    )
    ```


    - [12.4](#unsorted--jsx-text) Don't mix JSX and Text node in one line.
    > TODO: find or make ESLint rule for this

    ```javascript
    // bad
    return (
    Open in Safari
    )


    // good
    return (

    Open in Safari

    )


    // good - if you need a Text node for styling
    return (
     
    )
    ```

    **[⬆ back to top](#contents)**