2010-12-29 14:06:57 -05:00
|
|
|
# Function Literals
|
|
|
|
# -----------------
|
|
|
|
|
2011-01-03 04:17:00 -05:00
|
|
|
# TODO: add indexing and method invocation tests: (->)[0], (->).call()
|
|
|
|
|
2010-12-29 14:06:57 -05:00
|
|
|
# * Function Definition
|
|
|
|
# * Bound Function Definition
|
|
|
|
# * Parameter List Features
|
|
|
|
# * Splat Parameters
|
|
|
|
# * Context (@) Parameters
|
|
|
|
# * Parameter Destructuring
|
|
|
|
# * Default Parameters
|
|
|
|
|
|
|
|
#### Function Definition
|
|
|
|
|
|
|
|
x = 1
|
|
|
|
y = {}
|
|
|
|
y.x = -> 3
|
|
|
|
ok x is 1
|
|
|
|
ok typeof(y.x) is 'function'
|
|
|
|
ok y.x instanceof Function
|
|
|
|
ok y.x() is 3
|
|
|
|
|
|
|
|
# The empty function should not cause a syntax error.
|
|
|
|
->
|
|
|
|
() ->
|
|
|
|
|
|
|
|
# Multiple nested function declarations mixed with implicit calls should not
|
|
|
|
# cause a syntax error.
|
|
|
|
(one) -> (two) -> three four, (five) -> six seven, eight, (nine) ->
|
|
|
|
|
|
|
|
# with multiple single-line functions on the same line.
|
|
|
|
func = (x) -> (x) -> (x) -> x
|
|
|
|
ok func(1)(2)(3) is 3
|
|
|
|
|
|
|
|
# Make incorrect indentation safe.
|
|
|
|
func = ->
|
|
|
|
obj = {
|
|
|
|
key: 10
|
|
|
|
}
|
|
|
|
obj.key - 5
|
|
|
|
eq func(), 5
|
|
|
|
|
2010-12-30 22:48:31 -05:00
|
|
|
# Ensure that functions with the same name don't clash with helper functions.
|
|
|
|
del = -> 5
|
|
|
|
ok del() is 5
|
|
|
|
|
2010-12-29 14:06:57 -05:00
|
|
|
|
|
|
|
#### Bound Function Definition
|
|
|
|
|
|
|
|
obj =
|
|
|
|
bound: ->
|
|
|
|
(=> this)()
|
|
|
|
unbound: ->
|
|
|
|
(-> this)()
|
|
|
|
nested: ->
|
|
|
|
(=>
|
|
|
|
(=>
|
|
|
|
(=> this)()
|
|
|
|
)()
|
|
|
|
)()
|
|
|
|
eq obj, obj.bound()
|
|
|
|
ok obj isnt obj.unbound()
|
|
|
|
eq obj, obj.nested()
|
|
|
|
|
|
|
|
|
|
|
|
#### Parameter List Features
|
|
|
|
|
|
|
|
test "splats", ->
|
|
|
|
arrayEq [0, 1, 2], (((splat...) -> splat) 0, 1, 2)
|
|
|
|
arrayEq [2, 3], (((_, _, splat...) -> splat) 0, 1, 2, 3)
|
|
|
|
arrayEq [0, 1], (((splat..., _, _) -> splat) 0, 1, 2, 3)
|
|
|
|
arrayEq [2], (((_, _, splat..., _) -> splat) 0, 1, 2, 3)
|
|
|
|
|
|
|
|
test "@-parameters: automatically assign an argument's value to a property of the context", ->
|
|
|
|
nonce = {}
|
|
|
|
|
|
|
|
((@prop) ->).call context = {}, nonce
|
|
|
|
eq nonce, context.prop
|
|
|
|
|
|
|
|
# allow splats along side the special argument
|
|
|
|
((splat..., @prop) ->).apply context = {}, [0, 0, nonce]
|
|
|
|
eq nonce, context.prop
|
|
|
|
|
|
|
|
# allow the argument itself to be a splat
|
|
|
|
((@prop...) ->).call context = {}, 0, nonce, 0
|
|
|
|
eq nonce, context.prop[1]
|
|
|
|
|
|
|
|
# the argument should still be able to be referenced normally
|
|
|
|
eq nonce, (((@prop) -> prop).call {}, nonce)
|
|
|
|
|
|
|
|
test "@-parameters and splats with constructors", ->
|
|
|
|
a = {}
|
|
|
|
b = {}
|
|
|
|
class Klass
|
|
|
|
constructor: (@first, splat..., @last) ->
|
|
|
|
|
|
|
|
obj = new Klass a, 0, 0, b
|
|
|
|
eq a, obj.first
|
|
|
|
eq b, obj.last
|
|
|
|
|
|
|
|
test "destructuring in function definition", ->
|
|
|
|
(([{a: [b], c}]...) ->
|
|
|
|
eq 1, b
|
|
|
|
eq 2, c
|
|
|
|
) {a: [1], c: 2}
|
|
|
|
|
|
|
|
test "default values", ->
|
|
|
|
nonceA = {}
|
|
|
|
nonceB = {}
|
|
|
|
a = (_,_,arg=nonceA) -> arg
|
|
|
|
eq nonceA, a()
|
|
|
|
eq nonceA, a(0)
|
|
|
|
eq nonceB, a(0,0,nonceB)
|
|
|
|
eq nonceA, a(0,0,undefined)
|
|
|
|
eq nonceA, a(0,0,null)
|
|
|
|
eq false , a(0,0,false)
|
|
|
|
eq nonceB, a(undefined,undefined,nonceB,undefined)
|
|
|
|
b = (_,arg=nonceA,_,_) -> arg
|
|
|
|
eq nonceA, b()
|
|
|
|
eq nonceA, b(0)
|
|
|
|
eq nonceB, b(0,nonceB)
|
|
|
|
eq nonceA, b(0,undefined)
|
|
|
|
eq nonceA, b(0,null)
|
|
|
|
eq false , b(0,false)
|
|
|
|
eq nonceB, b(undefined,nonceB,undefined)
|
|
|
|
c = (arg=nonceA,_,_) -> arg
|
|
|
|
eq nonceA, c()
|
|
|
|
eq 0, c(0)
|
|
|
|
eq nonceB, c(nonceB)
|
|
|
|
eq nonceA, c(undefined)
|
|
|
|
eq nonceA, c(null)
|
|
|
|
eq false , c(false)
|
|
|
|
eq nonceB, c(nonceB,undefined,undefined)
|
|
|
|
|
|
|
|
test "default values with @-parameters", ->
|
|
|
|
a = {}
|
|
|
|
b = {}
|
|
|
|
obj = f: (q = a, @p = b) -> q
|
|
|
|
eq a, obj.f()
|
|
|
|
eq b, obj.p
|
|
|
|
|
|
|
|
test "default values with splatted arguments", ->
|
|
|
|
withSplats = (a = 2, b..., c = 3, d = 5) -> a * (b.length + 1) * c * d
|
|
|
|
eq 30, withSplats()
|
|
|
|
eq 15, withSplats(1)
|
|
|
|
eq 5, withSplats(1,1)
|
|
|
|
eq 1, withSplats(1,1,1)
|
|
|
|
eq 2, withSplats(1,1,1,1)
|