nomsu/nomsu_tree.moon

74 lines
2.3 KiB
Plaintext
Raw Normal View History

-- This file contains the datastructures used to represent parsed Nomsu syntax trees,
-- as well as the logic for converting them to Lua code.
utils = require 'utils'
{:repr, :stringify, :min, :max, :equivalent, :set, :is_list, :sum} = utils
immutable = require 'immutable'
{:insert, :remove, :concat} = table
2018-04-26 14:00:01 -07:00
{:Lua, :Nomsu, :Location} = require "code_obj"
MAX_LINE = 80 -- For beautification purposes, try not to make lines much longer than this value
Types = {}
Types.is_node = (n)->
type(n) == 'userdata' and getmetatable(n) and Types[n.type] == getmetatable(n)
-- Helper method:
2018-05-16 18:12:56 -07:00
Tree = (name, kind, methods)->
methods or= {}
2018-05-16 18:12:56 -07:00
assert((kind == 'single') or (kind == 'multi'))
is_multi = (kind == 'multi')
with methods
2018-05-16 18:12:56 -07:00
.with_value = (value)=> getmetatable(self)(value)
.type = name
.name = name
2018-05-16 18:12:56 -07:00
.is_multi = is_multi
.map = (fn)=>
if type(fn) == 'table'
return @ unless next(fn)
if type(next(fn)) == 'string'
error("SHIT")
_replacements = fn
fn = (k)-> _replacements[k]
return @_map(fn)
2018-05-16 18:12:56 -07:00
if is_multi
.__tostring = => "#{@name}(#{table.concat [repr(v) for v in *@], ', '})"
2018-05-24 20:27:08 -07:00
._map = (fn)=>
2018-05-16 18:12:56 -07:00
if ret = fn(@)
return ret
2018-05-24 20:27:08 -07:00
new_vals = [v._map and v\_map(fn) or v for v in *@]
2018-05-16 18:12:56 -07:00
ret = getmetatable(self)(unpack(new_vals))
return ret
else
.__tostring = => "#{@name}(#{repr(@value)})"
2018-05-24 20:27:08 -07:00
._map = (fn)=>
fn(@) or @
2018-05-16 18:12:56 -07:00
if is_multi
Types[name] = immutable nil, methods
else
Types[name] = immutable {"value"}, methods
Tree "Block", 'multi'
2018-05-24 20:27:08 -07:00
Tree "EscapedNomsu", 'multi'
Tree "Text", 'multi'
Tree "List", 'multi'
Tree "Dict", 'multi'
Tree "DictEntry", 'multi'
Tree "IndexChain", 'multi'
Tree "Number", 'single'
Tree "Word", 'single'
Tree "Comment", 'single'
Tree "Var", 'single',
as_lua_id: =>
"_"..(@value\gsub("%W", (c)-> if c == "_" then "__" else ("_%x")\format(c\byte!)))
2018-05-16 18:12:56 -07:00
Tree "Action", 'multi',
get_stub: (include_names=false)=>
bits = if include_names
2018-05-16 18:12:56 -07:00
[(t.type == "Word" and t.value or "%#{t.value}") for t in *@]
else [(t.type == "Word" and t.value or "%") for t in *@]
return concat(bits, " ")
return Types