2018-04-17 14:18:23 -07:00
|
|
|
-- 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-05-26 15:58:32 -07:00
|
|
|
{:Lua, :Nomsu, :Source} = require "code_obj"
|
2018-04-17 14:18:23 -07:00
|
|
|
|
2018-04-25 16:30:49 -07:00
|
|
|
MAX_LINE = 80 -- For beautification purposes, try not to make lines much longer than this value
|
2018-04-17 14:18:23 -07:00
|
|
|
|
|
|
|
Types = {}
|
|
|
|
Types.is_node = (n)->
|
|
|
|
type(n) == 'userdata' and getmetatable(n) and Types[n.type] == getmetatable(n)
|
|
|
|
|
|
|
|
-- Helper method:
|
2018-06-04 17:56:09 -07:00
|
|
|
Tree = (name, fields, methods)->
|
2018-05-16 19:08:16 -07:00
|
|
|
methods or= {}
|
2018-06-04 17:56:09 -07:00
|
|
|
is_multi = true
|
|
|
|
for f in *fields do is_multi and= (f != "value")
|
2018-04-17 14:18:23 -07:00
|
|
|
with methods
|
|
|
|
.type = name
|
|
|
|
.name = name
|
2018-06-04 17:56:09 -07:00
|
|
|
.__new or= (source, ...)=>
|
2018-05-26 15:04:31 -07:00
|
|
|
assert source
|
2018-05-26 15:58:32 -07:00
|
|
|
if type(source) == 'string'
|
|
|
|
source = Source\from_string(source)
|
2018-06-04 17:56:09 -07:00
|
|
|
--assert Source\is_instance(source)
|
|
|
|
return source, ...
|
2018-05-16 18:12:56 -07:00
|
|
|
.is_multi = is_multi
|
2018-05-16 19:08:16 -07:00
|
|
|
.map = (fn)=>
|
|
|
|
if type(fn) == 'table'
|
|
|
|
return @ unless next(fn)
|
|
|
|
_replacements = fn
|
|
|
|
fn = (k)-> _replacements[k]
|
|
|
|
return @_map(fn)
|
2018-05-16 18:12:56 -07:00
|
|
|
if is_multi
|
2018-06-04 17:56:09 -07:00
|
|
|
.__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-06-04 17:56:09 -07:00
|
|
|
new_vals = [v._map and v\_map(fn) or v for v in *@]
|
|
|
|
ret = getmetatable(self)(@source, unpack(new_vals))
|
2018-05-16 18:12:56 -07:00
|
|
|
return ret
|
|
|
|
else
|
|
|
|
.__tostring = => "#{@name}(#{repr(@value)})"
|
2018-05-24 20:27:08 -07:00
|
|
|
._map = (fn)=>
|
2018-05-16 19:08:16 -07:00
|
|
|
fn(@) or @
|
2018-04-17 14:18:23 -07:00
|
|
|
|
2018-06-04 17:56:09 -07:00
|
|
|
Types[name] = immutable fields, methods
|
2018-04-17 14:18:23 -07:00
|
|
|
|
2018-06-04 17:56:09 -07:00
|
|
|
Tree "Block", {"source"}
|
|
|
|
Tree "EscapedNomsu", {"source"}
|
|
|
|
Tree "Text", {"source"}
|
|
|
|
Tree "List", {"source"}
|
|
|
|
Tree "Dict", {"source"}
|
|
|
|
Tree "DictEntry", {"source"}
|
|
|
|
Tree "IndexChain", {"source"}
|
|
|
|
Tree "Number", {"source", "value"}
|
|
|
|
Tree "Var", {"source", "value"}
|
2018-04-24 20:16:46 -07:00
|
|
|
|
2018-06-04 17:56:09 -07:00
|
|
|
Tree "Action", {"source", "stub"},
|
|
|
|
__new: (source, ...)=>
|
2018-05-30 17:20:22 -07:00
|
|
|
assert source
|
|
|
|
if type(source) == 'string'
|
|
|
|
source = Source\from_string(source)
|
2018-06-04 17:56:09 -07:00
|
|
|
--assert Source\is_instance(source)
|
|
|
|
stub_bits = {}
|
|
|
|
for i=1,select("#",...)
|
|
|
|
a = select(i, ...)
|
|
|
|
stub_bits[i] = type(a) == 'string' and a or "%"
|
|
|
|
stub = concat stub_bits, " "
|
|
|
|
return source, stub, ...
|
2018-05-30 17:20:22 -07:00
|
|
|
get_spec: =>
|
2018-06-04 17:56:09 -07:00
|
|
|
concat [type(a) == "string" and a or "%#{a.value}" for a in *@], " "
|
2018-04-24 20:16:46 -07:00
|
|
|
|
2018-04-17 14:18:23 -07:00
|
|
|
return Types
|