helpers.coffee | |
---|---|
This file contains the common helper functions that we'd like to share among the Lexer, Rewriter, and the Nodes. Merge objects, flatten arrays, count characters, that sort of thing. | |
Set up exported variables for both Node.js and the browser. | this.exports: this unless process?
helpers: exports.helpers: {} |
Does a list include a value? | helpers.include: include: (list, value) ->
list.indexOf(value) >= 0 |
Peek at the beginning of a given string to see if it matches a sequence. | helpers.starts: starts: (string, literal, start) ->
string.substring(start, (start or 0) + literal.length) is literal |
Trim out all falsy values from an array. | helpers.compact: compact: (array) -> item for item in array when item |
Count the number of occurences of a character in a string. | helpers.count: count: (string, letter) ->
num: 0
pos: string.indexOf(letter)
while pos isnt -1
num: + 1
pos: string.indexOf(letter, pos + 1)
num |
Merge objects, returning a fresh copy with attributes from both sides.
Used every time | helpers.merge: merge: (options, overrides) ->
fresh: {}
(fresh[key]: val) for key, val of options
(fresh[key]: val) for key, val of overrides if overrides
fresh |
Extend a source object with the properties of another object (shallow copy).
We use this to simulate Node's deprecated | helpers.extend: extend: (object, properties) ->
(object[key]: val) for key, val of properties |
Return a completely flattened version of an array. Handy for getting a
list of | helpers.flatten: flatten: (array) ->
memo: []
for item in array
if item instanceof Array then memo: memo.concat(item) else memo.push(item)
memo |
Delete a key from an object, returning the value. Useful when a node is looking for a particular method in an options hash. | helpers.del: del: (obj, key) ->
val: obj[key]
delete obj[key]
val |
Matches a balanced group such as a single or double-quoted string. Pass in a series of delimiters, all of which must be nested correctly within the contents of the string. This method allows us to have strings within interpolations within strings, ad infinitum. | helpers.balanced_string: balanced_string: (str, delimited, options) ->
options: or {}
slash: delimited[0][0] is '/'
levels: []
i: 0
while i < str.length
if levels.length and starts str, '\\', i
i: + 1
else
for pair in delimited
[open, close]: pair
if levels.length and starts(str, close, i) and levels[levels.length - 1] is pair
levels.pop()
i: + close.length - 1
i: + 1 unless levels.length
break
else if starts str, open, i
levels.push(pair)
i: + open.length - 1
break
break if not levels.length or slash and starts str, '\n', i
i: + 1
if levels.length
return false if slash
throw new Error "SyntaxError: Unterminated ${levels.pop()[0]} starting on line ${@line + 1}"
if not i then false else str.substring(0, i)
|