CoffeeScript
CoffeeScript is a programming language that compiles to JavaScript. It adds syntactic sugar inspired by Ruby, Python and Haskell in an effort to enhance JavaScript's brevity and readability. Specific additional features include list comprehension and destructuring assignment.
CoffeeScript support is included in Ruby on Rails version 3.1 and Play Framework. In 2011, Brendan Eich referenced CoffeeScript as an influence on his thoughts about the future of JavaScript.
History
On December 13, 2009, Jeremy Ashkenas made the first Git commit of CoffeeScript with the comment: "initial commit of the mystery language." The compiler was written in Ruby. On December 24, he made the first tagged and documented release, 0.1.0. On February 21, 2010, he committed version 0.5, which replaced the Ruby compiler with a self-hosting version in pure CoffeeScript. By that time the project had attracted several other contributors on GitHub, and was receiving over 300 page hits per day.On December 24, 2010, Ashkenas announced the release of stable 1.0.0 to Hacker News, the site where the project was announced for the first time.
On September 18, 2017, version 2.0.0 was introduced, which "aims to bring CoffeeScript into the modern JavaScript era, closing gaps in compatibility with JavaScript while preserving the clean syntax that is CoffeeScript’s hallmark."
Syntax
Almost everything is an expression in CoffeeScript, for exampleif
, switch
and for
expressions return a value. As in Perl, these control statements also have postfix versions; for example, if
can also be written in consequent if condition
form.Many unnecessary parentheses and braces can be omitted; for example, blocks of code can be denoted by indentation instead of braces, function calls are implicit, and object literals are often detected automatically.
To compute the body mass index, one may do :
const mass = 72
const height = 1.78
const BMI = mass / height ** 2
if
With CoffeeScript the interval is directly described:
mass = 72
height = 1.78
BMI = mass / height**2
alert 'You are healthy!' if 18.5 < BMI < 25
To compute the greatest common divisor of two integers with the euclidean algorithm, in JavaScript one usually needs a while loop:
gcd = =>
Whereas in CoffeeScript one can use
until
and destructuring assignment instead:gcd = ->
= until y is 0
x
Any for loop can be replaced by a list comprehension; so that to compute the squares of the positive odd numbers smaller than ten, one can do:
alert n*n for n in when n%2 is 1
Alternatively, there is:
alert n*n for n in by 2
A linear search can be implemented with a one-liner using the when keyword:
names =
linearSearch = -> alert for name in names when name is searchName
The
for... in
syntax allows looping over arrays while the for... of
syntax allows looping over objects.The
?
keyword quickly checks if a variable is null
or undefined
:personCheck = ->
if not person? then alert else alert
person = null
personCheck
person = "Ivan"
personCheck
This would alert "No person" if the variable is
null
or undefined
and "Have person" if there is something there.A common JavaScript snippet using the jQuery library is:
$.ready
Or even just:
$
In CoffeeScript, the
function
keyword is replaced by the ->
symbol, and indentation is used instead of curly braces, as in other off-side rule languages such as Python and Haskell. Also, parentheses can usually be omitted, using indentation level instead to denote a function or block. Thus, the CoffeeScript equivalent of the snippet above is:$.ready ->
# Initialization code goes here
Or just:
$ ->
# Initialization code goes here
Ruby-style string interpolation is included in CoffeeScript. Double-quoted strings allow for interpolated values, using #, and single-quoted strings are literal.
author = "Wittgenstein"
quote = "A picture is a fact. -- #"
sentence = "# is a decent approximation of π"
CoffeeScript has been criticized for its unusual scoping
rules. In particular, it completely disallows variable shadowing which makes reasoning about code more difficult and
error-prone in some basic programming patterns established
by and taken for granted since procedural programming
principles were defined.
For example, with the following code snippet in JavaScript
one does not have to look outside the
-block to know forsure that no possible
foo
variable in the outer scope can beincidentally overridden:
//...
function baz
//...
Development and distribution
The CoffeeScript compiler has been self-hosting since version 0.5 and is available as a Node.js utility; however, the core compiler does not rely on Node.js and can be run in any JavaScript environment. One alternative to the Node.js utility is the Coffee Maven Plugin, a plugin for the Apache Maven build system. The plugin uses the Rhino JavaScript engine written in Java.The official site at CoffeeScript.org has a "Try CoffeeScript" button in the menu bar; clicking it opens a modal window in which users can enter CoffeeScript, see the JavaScript output, and run it directly in the browser. The js2coffee site provides bi-directional translation.
Latest additions
- Source maps allow users to de-bug their CoffeeScript code directly, supporting CoffeeScript tracebacks on run time errors.
- CoffeeScript supports a form of Literate Programming, using the
.coffee.md
or.litcoffee
file extension. This allows CoffeeScript source code to be written in Markdown. The compiler will treat any indented blocks as code, and ignore the rest as comments.Extensions
await
and defer
. These additions simplify asynchronous control flow, making the code to look more like a procedural programming language, eliminating the call-back chain. It can be used on the server side and in the browser.Adoption
On September 13, 2012, Dropbox announced that their browser-side code base has been rewritten from JavaScript to CoffeeScript, however it has been migrated to TypeScript in 2017.GitHub's internal style guide once said "write new JS in CoffeeScript", and while it no longer does, all the advice in the style guide references how to write good CoffeeScript, and their Atom text editor is also written in the language.