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

Fix: Fat arrow method in namespaced class causes syntax error #1

Open
wants to merge 21 commits into
base: master
Choose a base branch
from
Open
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
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -8,3 +8,4 @@ examples/beautiful_code/parse.coffee
*.gem
build/
node_modules/
.idea/
213 changes: 213 additions & 0 deletions CLOSURE.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,213 @@
# Closure Mode

Original fork by @bolinfest <http://bolinfest.com/coffee/features.html> who added Closure Library support.

Forked from @smhanov <https://github.com/smhanov/coffee-script> who removed the need to use the Closure Library.

---

This fork's purpose is to allow developers to **type-check CoffeeScript/Backbone/Underscore projects** with **full backwards compatibility** and **minimal JSDoc annotations**.

Type checking uses the Google Closure Compiler with Advanced Optimizations mode. Type information is added using [Google Closure type-checking JSDoc annotations](https://developers.google.com/closure/compiler/docs/js-for-compiler). *Closure Mode* helps by adding annotations automatically by using CoffeeScript syntax.

I use vanilla CoffeeScript for compiling production/debug code, but use this fork with [jzbuild](https://github.com/vjpr/jzbuild) for occasional type-checking.

There is no intention to integrate this back into the master branch.

## Usage

### Add a helper function method called `__`

This function allows us to repurpose the *default parameter* CoffeeScript feature to specify type information while not breaking your code when it is compiled with vanilla CoffeeScript.

*Closure Mode* parses the *default parameter* to look for type information. If this fails, this method passes along the default parameter as normal. If there is type information, we return `undefined` unless there is a default parameter specified.

###* @param {*} a ###
###* @param {*} [b] ###
window.__ = (a, b) ->
return b if b?
if a.indexOf('{') is 0
and if a.substr(-1) is '}'
return undefined
else
return a

### Declaring custom types

When you have a custom object like `{foo: 'bar'}` used in a few places you should create an alias for it like so.

###* @typedef {{ foo: string }} ###
FooBar;

### Add constructors to Backbone classes

When passing options or other arguments to Backbone you need to provide a constructor.

class Foo extends Backbone.View
constructor: (options) ->

### Optional types

You must add type annotations to optional types like this:

###* @param {*} [name] ###
foo = (name) ->

*or...*

###* @param {*=} name ###

*or we pass it to our magic function `__()` that is recognized by *Closure Mode* returns undefined...*

foo = (name = __('{*=}') ->

*or we infer the type from the value...*

foo = (name = __('foo'))

### Inline types as default parameters

If a param **does not** have a default value

foo = (name = '{*=}') ->

otherwise...

foo = (name = __('{*=}', 'foo'))

### Namespacing

When using something like this at the top of your classes:

Foo ?= {}
Foo.Bar ?= {}

Modify it to:

do ->
Foo ?= {}
Foo.Bar ?= {}

### Skipping something

If something is continuously giving you warnings/errors use:

###* @suppress {checkTypes} ###

See [here](http://code.google.com/p/closure-compiler/wiki/Warnings) for different options.

## Features

### Automatic constructor annotations

`@constructor` and `@extends` annotations automatically added to `class` declarations

### Basic type inference

Use `--closure_infer`

Adds inline annotations to optional function parameters based on the type of the default parameter.

class Foo
constructor: (bar = '{string=}')

*Generates...*

/**
* @constructor
* @param {string=} bar
*/
Foo = function(bar) {
if bar == null {
bar = '{string=}'
}
}

## Current Google Closure mapping

Array

## Possible alternatives for the default parameter function

I like this one a lot:

foo = (name = +string) ->
foo = (name =+ string) ->
foo = (name =+ )
foo = (name =+ num) ->

Or:

Add `function(new:String,*)` for each inline param - this is unlikely

foo = (name = String, name = String.Opt)
foo = (name = String, name = String[s])

foo = (name = String, name = Opt(String))
foo = (name = String, name = Opt(String))

foo = (name = String, name = Array[String])
foo = (name = String, name = Object[String,Number]})

foo = (name =+ tString('hello'), name = tArray[(tString, tNumber)->])

foo = (name =+ _String('hello'), name = _Array[_String, _Number])

These are some more ideas:

foo = (name = '{string}'.type()) ->
foo = (name = '{string}'.default('foo')) ->

foo = (name = 'foo'.t('{?string=}')) ->

foo = (name = '{string}'._()) ->
foo = (name = '{string}'._('foo')) ->

foo = (name = string()) ->
foo = (name = 'foo'.string()) ->
foo = (name = (1).number()) ->
foo = (name = 'Foo.Bar'.type()) ->

foo = (name = (__ '{string}')) ->
foo = (name = (__ '{string}', 'foo')) ->

foo = (name = __['{string}']) ->

foo = (name = t('{string}')) ->

foo = name

Or:

Implement a special short-hand CoffeeScript JavaDoc:

# @name string
# @date Date
# =string
foo = (name = 1,date=+Date()
'hello'

## Planned Features

* Better inline annotations based on identifier not just
* Allow adding params by annotating the initialize method in Backbone
* Automatic @return

## Dev

* Hack, hack, hack
* `git checkout lib/*.js && bin/cake build && bin/cake build:parser && ./test-closure.sh`

For interactive testing try: `open closure/demo/index.html`

## Compile

To compile `extras/coffee-script.js` run `MINIFY=false bin/cake build:browser`

## Dev Tips

Use `rewriter.coffee` for modifying tokens before they are parsed by the lexer

Modify `grammar.coffee` to accomodate new tokens

Modify `compileNode()` method which actually writes out the code for each node
103 changes: 103 additions & 0 deletions closure/test/no-google-libs/closure.coffee
Original file line number Diff line number Diff line change
@@ -0,0 +1,103 @@

class test3

###*
@param {string} bar
###
constructor: (foo='test') ->

get: =>


class test

###*
@param {string} bar
###
constructor: (foo, @bar = 'test') ->
foo = 'foo'

class test2

###*
@param {string} bar
###
constructor: (foo='test') ->
test

###* hello ###
getBar: => @bar

###*
hello
###
getFoo: => @foo

###*
hello
###

getFoo2: => @foo

###*
@param {?string=} a
@return {Object}
###
aMethod: (a = '{?string=}', b='{{id: number, name: string}}', c='yo', d=T('{number=}'), e=T('{number=}', 1) ) =>

###*
Type inference
###
a: (a=T('a')) ->

b: (b=T(1)) ->

c: (c=T('{?string=}')) ->

cc: (c='{?string=}') ->

d: (d='yo!') ->

class toast

###*
@interface
###
constructor: (foo='test') ->

class toast2


# Block Comments

###
This is a here-comment.
Kind of like a heredoc.
###

test "something", ->
a

# Block Comments

###
This is a here-comment.
Kind of like a heredoc.
###

test "block comments in objects", ->
a = {}
b = {}
obj = {
a: a
###
comment
###
b: b
}

eq a, obj.a
eq b, obj.b

class a
constructor: (a) ->
9 changes: 9 additions & 0 deletions coffee-script.iml
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
<?xml version="1.0" encoding="UTF-8"?>
<module type="WEB_MODULE" version="4">
<component name="NewModuleRootManager" inherit-compiler-output="true">
<exclude-output />
<content url="file://$MODULE_DIR$" />
<orderEntry type="sourceFolder" forTests="false" />
</component>
</module>

Loading