Skip to content

sovolve/coffeescript-style-guide

 
 

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

19 Commits
 
 

Repository files navigation

Table of Contents

Source Code Layout

  • Use UTF-8 as the source file encoding.

  • Use two spaces per indentation level.

    # good
    someMethod = ->
      doSomething()
    
    # bad - four spaces
    someMethod = ->
        doSomething()
  • Use Unix-style line endings. (*BSD/Solaris/Linux/OSX users are covered by default, Windows users have to be extra careful.)

    • If you're using Git you might want to add the following configuration setting to protect your project from Windows line endings creeping in:

      $ git config --global core.autocrlf true

  • Use spaces around operators, after commas and semicolons.

    sum = 1 + 2
    nums = [1, 2, 3]
  • No spaces after (, [ or before ], ).

    some(arg).other
    [1, 2, 3].length
  • No spaces after : when used in an object.

    # bad
    song =
      name : 'Misty'
      duration : 192
      year : 1999
    
    # good
    song =
      name: 'Misty'
      duration: 192
      year: 1999
  • Use empty lines between methods and to break up a method into logical paragraphs.

    someMethod = ->
      data = initialize(options)
    
      data.manipulate
    
      data.result
    
    someOtherMethod = ->
      result
  • Keep lines fewer than 100 characters.

  • Remove trailing whitespace.

Syntax

  • Omit the parentheses on method creation when the method doesn't accept any arguments. This also goes for anonymous functions.

    someMethod = ->
      # body omitted
    
    someMethodWithArguments = (arg1, arg2) ->
      # body omitted
  • Avoid the use of third party (underscore, jQuery) iterators when a for loop can be used. The iterators add unnecessary overhead.

    arr = [1, 2, 3]
    
    # bad
    _.each arr, (elem) ->
      console.log elem
    
    # good
    console.log elem for elem in arr
  • Never use then for multi-line if/unless.

    # bad
    if someCondition then
      # body omitted
    
    # good
    if someCondition
      # body omitted
  • Favor modifier if/unless usage when you have a single-line body. Another good alternative is the usage of control flow and/or.

    # bad
    if someCondition
      doSomething
    
    # good
    doSomething if someCondition
    
    # another good option
    someCondition and doSomething
  • Favor unless over if for negative conditions (or control flow or).

    # bad
    doSomething if !someCondition
    
    # good
    doSomething unless someCondition
    
    # another good option
    someCondition or doSomething
  • Never use unless with else. Rewrite these with the positive case first.

    # bad
    unless success?
      console.log 'failure'
    else
      console.log 'success'
    
    # good
    if success?
      console.log 'success'
    else
      console.log 'failure'
  • Don't use parentheses around the condition of an if/unless/while, unless the condition contains an assignment (see "Using the return value of =" below).

    # bad
    if (x > 10)
      # body omitted
    
    # good
    if x > 10
      # body omitted
    
    # ok
    if (x = self.nextValue)
      # body omitted
  • Favor modifier for usage when you have a single-line body.

    # bad
    for elem in elements
      doSomething
    
    # good
    doSomething for elem in elements
  • Omit parentheses around parameters for the first function call on a line. Use parentheses around the arguments of all other method invocations.

    class Person
      constructor: (name, age) ->
        @checkAge age
        alert Math.max(age, 10)
    
    
    x = Math.sin y
    array.delete e, Person.maxAge(a)
  • Avoid return where not required.

    # bad
    someMethod = (someArr) ->
      return someArr.size
    
    # good
    someMethod = (someArr) ->
      someArr.size
  • Avoid shadowing methods with local variables unless they are both equivalent

    class Foo
      constructor: (options) ->
        @options = options
    
      # bad
      doSomething = (options = {}) ->
        unless options['when'] == 'later'
          output(self.options['message'])
    
      # good
      doSomething = (params = {}) ->
        unless params['when'] == 'later'
          output(options['message'])
  • Use spaces around the = operator when assigning default values to method parameters:

    # bad
    someMethod = (arg1='default', arg2=null, arg3=[]) ->
      # do something...
    
    # good
    someMethod = (arg1 = 'default', arg2 = null, arg3 = []) ->
      # do something...
  • Using the return value of = (an assignment) is ok, but surround the assignment with parenthesis.

    # good - shows intended use of assignment
    if (v = str.indexOf('foo')) ...
    
    # bad
    if v = str.indexOf('foo') ...
    
    # also good - shows intended use of assignment and has correct precedence.
    if (v = @nextValue) == 'hello' ...
  • Use or= or ?= freely to initialize variables.

    # set name to Bozhidar, only if it's null or false
    name or= 'Bozhidar'
    # set name to Bozhidar, only if it's null
    name ?= 'Bozhidar'
  • Don't use or= to initialize boolean variables. (Consider what would happen if the current value happened to be false.)

    # bad - would set enabled to true even if it was false
    enabled or= true
    
    # good
    enabled ?= true

Naming

The only real difficulties in programming are cache invalidation and naming things.
-- Phil Karlton

  • Use camelCase for variables and properties.
  • Use capitalized CamelCase for constructors and modules. (Keep acronyms like HTTP, RFC, XML uppercase.)
  • Use SCREAMING_SNAKE_CASE for constants.
  • The names of predicate methods (methods that return a boolean value) should start with is or has. (i.e. isEmpty, hasDefaultValue).

Comments

Good code is its own best documentation. As you're about to add a comment, ask yourself, "How can I improve the code so that this comment isn't needed?" Improve the code and then document it to make it even clearer.
-- Steve McConnell

  • Remember that If the file has an initial comment block it will not be removed when the file is compressed. No sensitive information should be put in the initial block.

  • Any publically released code should contain a license block at the start of the file.

  • Gem spec style information should be included in a block at the start of the file (after the license if needed). Version and homepage information are important for publically released code. Internally, dependencies can be helpful in determining weather a JavaScript/CoffeeScript library is being used elsewhere.

    # version: 1.1.0
    # homepage: https://github.com/orgsync/some_lib
    # dependency: jQuery, ~> 1.7.1
  • Write self-documenting code and ignore the rest of this section. Seriously!

  • Comments longer than a word are capitalized and use punctuation. Use one space after periods.

  • Avoid superfluous comments.

    # bad
    counter += 1 # increments counter by one
  • Keep existing comments up-to-date. An outdated comment is worse than no comment at all.

Good code is like a good joke - it needs no explanation.
-- Russ Olsen

  • Avoid writing comments to explain bad code. Refactor the code to make it self-explanatory. (Do or do not - there is no try. --Yoda)

Annotations

  • Annotations should usually be written on the line immediately above the relevant code.

  • The annotation keyword is followed by a colon and a space, then a note describing the problem.

  • If multiple lines are required to describe the problem, subsequent lines should be indented two spaces after the #.

    bar = ->
      # FIXME: This has crashed occasionally since v3.2.1. It may
      #   be related to the BarBazUtil upgrade.
      baz('quux')
  • In cases where the problem is so obvious that any documentation would be redundant, annotations may be left at the end of the offending line with no note. This usage should be the exception and not the rule.

    bar = ->
      @sleep(100) # OPTIMIZE
  • Use TODO to note missing features or functionality that should be added at a later date.

  • Use FIXME to note broken code that needs to be fixed.

  • Use OPTIMIZE to note slow or inefficient code that may cause performance problems.

  • Use HACK to note code smells where questionable coding practices were used and should be refactored away.

  • Use REVIEW to note anything that should be looked at to confirm it is working as intended. For example: REVIEW: Are we sure this is how the client does X currently?

  • Use other custom annotation keywords if it feels appropriate, but be sure to document them in your project's README or similar.

Classes

  • When designing class hierarchies make sure that they conform to the Liskov Substitution Principle.

  • Try to make your classes as SOLID as possible.

  • Prefer duck-typing over inheritance.

    # bad
    class Animal
      # abstract method
      speak: ->
    
    # extend superclass
    class Duck extends Animal
      speak: ->
        alert 'Quack! Quack'
    
    # extend superclass
    class Dog extends Animal
      speak: ->
        alert 'Bau! Bau!'
    
    # good
    class Duck
      speak: ->
        alert 'Quack! Quack'
    
    class Dog
      speak: ->
        alert 'Bau! Bau!'
  • Denote private methods and attributes with _.

Strings

  • Prefer string interpolation instead of string concatenation:

    # bad
    emailWithName = user.name + ' <' + user.email + '>'
    
    # good
    emailWithName = "#{user.name} <#{user.email}>"
  • Consider padding string interpolation code with space. It more clearly sets the code apart from the string.

    "#{ user.last_name }, #{ user.first_name }"
  • Prefer single-quoted strings when you don't need string interpolation or special symbols such as \t, \n, ', etc.

    # bad
    name = "Bozhidar"
    
    # good
    name = 'Bozhidar'

Regular Expressions

  • Don't use regular expressions if you just need plain text search in string: 'text'.match('ex')

  • Use non capturing groups when you don't use captured result of parenthesis.

    /(first|second)/   # bad
    /(?:first|second)/ # good
  • Use /// modifier for complex regexps. This makes them more readable and you can add some useful comments. Just be careful as spaces are ignored.

    regexp = ///
      start         # some text
      \s            # white space char
      (group)       # first group
      (?:alt1|alt2) # some alternation
      end
    ///

Misc

  • Avoid hashes as optional parameters. Does the method do too much?
  • Avoid methods longer than 10 LOC (lines of code). Ideally, most methods will be shorter than 5 LOC. Empty lines do not contribute to the relevant LOC.
  • Avoid parameter lists longer than three or four parameters.
  • Code in a functional way, avoiding mutation when that makes sense.
  • Avoid needless metaprogramming.
  • Do not mutate arguments unless that is the purpose of the method.
  • Avoid more than three levels of block nesting.
  • Be consistent. In an ideal world, be consistent with these guidelines.
  • Use common sense.

Referencing DOM

  • A class should never be used for both scripting and CSS. Classes used in scripting to find DOM elements must use the prefix js-.

About

Sovolve rules for writing good CoffeeScript, strongly inspired from https://github.com/orgsync/coffeescript-style-guide

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published