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-04-26 14:00:01 -07:00
|
|
|
{:Lua, :Nomsu, :Location} = 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-05-16 18:12:56 -07:00
|
|
|
Tree = (name, kind, methods)->
|
|
|
|
assert((kind == 'single') or (kind == 'multi'))
|
|
|
|
is_multi = (kind == 'multi')
|
2018-04-17 14:18:23 -07:00
|
|
|
with methods
|
2018-05-16 18:12:56 -07:00
|
|
|
.with_value = (value)=> getmetatable(self)(value)
|
2018-04-17 14:18:23 -07:00
|
|
|
.type = name
|
|
|
|
.name = name
|
2018-05-16 18:12:56 -07:00
|
|
|
.is_multi = is_multi
|
|
|
|
if is_multi
|
|
|
|
.__tostring = => "#{@name}(#{table.concat [repr(v) for v in *@], ', '})"
|
|
|
|
.map = (fn)=>
|
|
|
|
if ret = fn(@)
|
|
|
|
return ret
|
|
|
|
new_vals = [v.map and v\map(fn) or v for v in *@]
|
|
|
|
ret = getmetatable(self)(unpack(new_vals))
|
|
|
|
return ret
|
|
|
|
else
|
|
|
|
.__tostring = => "#{@name}(#{repr(@value)})"
|
|
|
|
.map = (fn)=> fn(@) or @
|
2018-04-17 14:18:23 -07:00
|
|
|
|
2018-05-16 18:12:56 -07:00
|
|
|
if is_multi
|
|
|
|
Types[name] = immutable nil, methods
|
|
|
|
else
|
|
|
|
Types[name] = immutable {"value"}, methods
|
2018-04-17 14:18:23 -07:00
|
|
|
|
|
|
|
|
2018-05-16 18:12:56 -07:00
|
|
|
Tree "EscapedNomsu", 'single',
|
2018-04-17 14:18:23 -07:00
|
|
|
as_lua: (nomsu)=>
|
2018-05-16 18:12:56 -07:00
|
|
|
make_tree = (t)->
|
|
|
|
if type(t) != 'userdata'
|
|
|
|
return repr(t)
|
|
|
|
if t.is_multi
|
|
|
|
bits = [make_tree(bit) for bit in *t]
|
|
|
|
return t.type.."("..table.concat(bits, ", ")..")"
|
|
|
|
else
|
|
|
|
return t.type.."("..make_tree(t.value)..")"
|
|
|
|
Lua.Value nil, make_tree(@value)
|
2018-04-17 14:18:23 -07:00
|
|
|
|
2018-04-24 20:16:46 -07:00
|
|
|
as_nomsu: (inline=false)=>
|
|
|
|
nomsu = @value\as_nomsu(true)
|
|
|
|
if nomsu == nil and not inline
|
|
|
|
nomsu = @value\as_nomsu!
|
2018-05-16 18:12:56 -07:00
|
|
|
return nomsu and Nomsu nil, "\\:\n ", nomsu
|
|
|
|
return nomsu and Nomsu nil, "\\(", nomsu, ")"
|
|
|
|
|
|
|
|
map: (fn)=> fn(@) or @\map(fn)
|
2018-04-24 20:16:46 -07:00
|
|
|
|
2018-05-16 18:12:56 -07:00
|
|
|
Tree "Block", 'multi',
|
2018-04-17 14:18:23 -07:00
|
|
|
as_lua: (nomsu)=>
|
2018-05-16 18:12:56 -07:00
|
|
|
lua = Lua!
|
|
|
|
for i,line in ipairs @
|
2018-04-17 14:18:23 -07:00
|
|
|
line_lua = line\as_lua(nomsu)
|
2018-04-24 20:16:46 -07:00
|
|
|
if i > 1
|
2018-04-17 14:18:23 -07:00
|
|
|
lua\append "\n"
|
2018-05-15 14:53:37 -07:00
|
|
|
lua\append line_lua\as_statements!
|
2018-04-17 14:18:23 -07:00
|
|
|
return lua
|
|
|
|
|
2018-04-24 20:16:46 -07:00
|
|
|
as_nomsu: (inline=false)=>
|
2018-04-25 17:43:48 -07:00
|
|
|
if inline
|
2018-05-16 18:12:56 -07:00
|
|
|
nomsu = Nomsu!
|
|
|
|
for i,line in ipairs @
|
2018-04-25 17:43:48 -07:00
|
|
|
if i > 1
|
|
|
|
nomsu\append "; "
|
|
|
|
line_nomsu = line\as_nomsu(true)
|
|
|
|
return nil unless line_nomsu
|
|
|
|
nomsu\append line_nomsu
|
|
|
|
return nomsu
|
2018-05-16 18:12:56 -07:00
|
|
|
nomsu = Nomsu!
|
|
|
|
for i, line in ipairs @
|
2018-04-28 15:25:12 -07:00
|
|
|
line = assert(line\as_nomsu(nil, true), "Could not convert line to nomsu")
|
2018-04-25 16:04:46 -07:00
|
|
|
nomsu\append line
|
2018-05-16 18:12:56 -07:00
|
|
|
if i < #@
|
2018-04-24 20:16:46 -07:00
|
|
|
nomsu\append "\n"
|
2018-05-10 22:47:03 -07:00
|
|
|
if tostring(line)\match("\n")
|
|
|
|
nomsu\append "\n"
|
2018-04-24 20:16:46 -07:00
|
|
|
return nomsu
|
|
|
|
|
2018-05-03 16:30:55 -07:00
|
|
|
math_expression = re.compile [[ ([+-] " ")* "%" (" " [*/^+-] (" " [+-])* " %")+ !. ]]
|
2018-05-16 18:12:56 -07:00
|
|
|
Tree "Action", 'multi',
|
2018-04-17 14:18:23 -07:00
|
|
|
as_lua: (nomsu)=>
|
|
|
|
stub = @get_stub!
|
2018-05-03 22:33:44 -07:00
|
|
|
compile_action = nomsu.environment.COMPILE_ACTIONS[stub]
|
|
|
|
if compile_action
|
2018-05-16 18:12:56 -07:00
|
|
|
args = [arg for arg in *@ when arg.type != "Word"]
|
2018-04-17 14:18:23 -07:00
|
|
|
-- Force all compile-time actions to take a tree location
|
2018-05-03 22:33:44 -07:00
|
|
|
args = [args[p-1] for p in *nomsu.environment.ARG_ORDERS[compile_action][stub]]
|
2018-04-17 14:36:44 -07:00
|
|
|
-- Force Lua to avoid tail call optimization for debugging purposes
|
2018-05-03 22:33:44 -07:00
|
|
|
ret = compile_action(self, unpack(args))
|
2018-05-10 22:47:03 -07:00
|
|
|
if not ret then error("Failed to produce any Lua")
|
2018-04-17 14:36:44 -07:00
|
|
|
return ret
|
2018-05-03 21:56:07 -07:00
|
|
|
action = rawget(nomsu.environment.ACTIONS, stub)
|
2018-05-16 18:12:56 -07:00
|
|
|
lua = Lua.Value!
|
2018-05-03 21:56:07 -07:00
|
|
|
if not action and math_expression\match(stub)
|
2018-04-17 14:18:23 -07:00
|
|
|
-- This is a bit of a hack, but this code handles arbitrarily complex
|
|
|
|
-- math expressions like 2*x + 3^2 without having to define a single
|
|
|
|
-- action for every possibility.
|
2018-05-16 18:12:56 -07:00
|
|
|
for i,tok in ipairs @
|
2018-04-17 14:18:23 -07:00
|
|
|
if tok.type == "Word"
|
|
|
|
lua\append tok.value
|
|
|
|
else
|
|
|
|
tok_lua = tok\as_lua(nomsu)
|
|
|
|
unless tok_lua.is_value
|
2018-05-16 18:12:56 -07:00
|
|
|
error("non-expression value inside math expression: #{colored.yellow repr(tok)}")
|
2018-05-03 16:37:48 -07:00
|
|
|
if tok.type == "Action"
|
|
|
|
tok_lua\parenthesize!
|
2018-04-17 14:18:23 -07:00
|
|
|
lua\append tok_lua
|
2018-05-16 18:12:56 -07:00
|
|
|
if i < #@
|
2018-04-17 14:18:23 -07:00
|
|
|
lua\append " "
|
|
|
|
return lua
|
|
|
|
|
|
|
|
args = {}
|
2018-05-16 18:12:56 -07:00
|
|
|
for i, tok in ipairs @
|
2018-04-17 14:18:23 -07:00
|
|
|
if tok.type == "Word" then continue
|
|
|
|
arg_lua = tok\as_lua(nomsu)
|
|
|
|
unless arg_lua.is_value
|
2018-05-16 18:12:56 -07:00
|
|
|
error "Cannot use:\n#{colored.yellow repr(tok)}\nas an argument to #{stub}, since it's not an expression, it produces: #{repr arg_lua}", 0
|
2018-04-17 14:18:23 -07:00
|
|
|
insert args, arg_lua
|
|
|
|
|
2018-05-03 21:56:07 -07:00
|
|
|
if action
|
|
|
|
args = [args[p] for p in *nomsu.environment.ARG_ORDERS[action][stub]]
|
2018-04-17 14:18:23 -07:00
|
|
|
|
|
|
|
-- Not really worth bothering with ACTIONS.foo(...) style since almost every action
|
|
|
|
-- has arguments, so it won't work
|
|
|
|
lua\append "ACTIONS[",repr(stub),"]("
|
|
|
|
for i, arg in ipairs args
|
|
|
|
lua\append arg
|
|
|
|
if i < #args then lua\append ", "
|
|
|
|
lua\append ")"
|
|
|
|
return lua
|
|
|
|
|
|
|
|
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 *@]
|
2018-04-17 14:18:23 -07:00
|
|
|
return concat(bits, " ")
|
2018-04-24 20:16:46 -07:00
|
|
|
|
2018-04-28 15:25:12 -07:00
|
|
|
as_nomsu: (inline=false, can_use_colon=false)=>
|
2018-04-24 20:16:46 -07:00
|
|
|
if inline
|
2018-05-16 18:12:56 -07:00
|
|
|
nomsu = Nomsu!
|
|
|
|
for i,bit in ipairs @
|
2018-04-24 20:16:46 -07:00
|
|
|
if bit.type == "Word"
|
|
|
|
if i > 1
|
|
|
|
nomsu\append " "
|
|
|
|
nomsu\append bit.value
|
|
|
|
else
|
|
|
|
arg_nomsu = bit\as_nomsu(true)
|
|
|
|
return nil unless arg_nomsu
|
|
|
|
unless i == 1
|
|
|
|
nomsu\append " "
|
2018-04-25 16:04:46 -07:00
|
|
|
if bit.type == "Action" or bit.type == "Block"
|
2018-04-24 20:16:46 -07:00
|
|
|
arg_nomsu\parenthesize!
|
|
|
|
nomsu\append arg_nomsu
|
|
|
|
return nomsu
|
|
|
|
else
|
2018-05-16 18:12:56 -07:00
|
|
|
nomsu = Nomsu!
|
2018-04-28 15:25:12 -07:00
|
|
|
next_space = ""
|
|
|
|
-- TODO: track line length as we go and use 80-that instead of 80 for wrapping
|
|
|
|
last_colon = nil
|
2018-05-16 18:12:56 -07:00
|
|
|
for i,bit in ipairs @
|
2018-04-24 20:16:46 -07:00
|
|
|
if bit.type == "Word"
|
2018-04-28 15:25:12 -07:00
|
|
|
nomsu\append next_space, bit.value
|
|
|
|
next_space = " "
|
2018-04-24 20:16:46 -07:00
|
|
|
else
|
2018-04-28 20:45:17 -07:00
|
|
|
arg_nomsu = if last_colon == i-1 and bit.type == "Action" then nil
|
|
|
|
elseif bit.type == "Block" then nil
|
|
|
|
else bit\as_nomsu(true)
|
|
|
|
|
2018-04-25 16:30:49 -07:00
|
|
|
if arg_nomsu and #arg_nomsu < MAX_LINE
|
2018-04-28 15:25:12 -07:00
|
|
|
if bit.type == "Action"
|
|
|
|
if can_use_colon and i > 1
|
|
|
|
nomsu\append next_space\match("[^ ]*"), ": ", arg_nomsu
|
|
|
|
next_space = "\n.."
|
|
|
|
last_colon = i
|
|
|
|
else
|
|
|
|
nomsu\append next_space, "(", arg_nomsu, ")"
|
|
|
|
next_space = " "
|
|
|
|
else
|
|
|
|
nomsu\append next_space, arg_nomsu
|
|
|
|
next_space = " "
|
2018-04-24 20:16:46 -07:00
|
|
|
else
|
2018-04-28 15:25:12 -07:00
|
|
|
arg_nomsu = bit\as_nomsu(nil, true)
|
2018-04-24 20:16:46 -07:00
|
|
|
return nil unless nomsu
|
2018-04-28 15:25:12 -07:00
|
|
|
-- These types carry their own indentation
|
|
|
|
if bit.type != "List" and bit.type != "Dict" and bit.type != "Text"
|
|
|
|
if i == 1
|
2018-05-16 18:12:56 -07:00
|
|
|
arg_nomsu = Nomsu(nil, "(..)\n ", arg_nomsu)
|
2018-04-28 15:25:12 -07:00
|
|
|
else
|
2018-05-16 18:12:56 -07:00
|
|
|
arg_nomsu = Nomsu(nil, "\n ", arg_nomsu)
|
2018-04-28 15:25:12 -07:00
|
|
|
|
|
|
|
if last_colon == i-1 and (bit.type == "Action" or bit.type == "Block")
|
|
|
|
next_space = ""
|
|
|
|
nomsu\append next_space, arg_nomsu
|
|
|
|
next_space = "\n.."
|
2018-04-28 17:08:28 -07:00
|
|
|
|
|
|
|
if next_space == " " and #(tostring(nomsu)\match("[^\n]*$")) > MAX_LINE
|
|
|
|
next_space = "\n.."
|
2018-04-24 20:16:46 -07:00
|
|
|
return nomsu
|
2018-05-03 21:56:07 -07:00
|
|
|
|
2018-05-16 18:12:56 -07:00
|
|
|
Tree "Text", 'multi',
|
2018-04-17 14:18:23 -07:00
|
|
|
as_lua: (nomsu)=>
|
2018-05-16 18:12:56 -07:00
|
|
|
lua = Lua.Value!
|
2018-04-17 14:18:23 -07:00
|
|
|
string_buffer = ""
|
2018-05-16 18:12:56 -07:00
|
|
|
for bit in *@
|
2018-04-17 14:18:23 -07:00
|
|
|
if type(bit) == "string"
|
|
|
|
string_buffer ..= bit
|
|
|
|
continue
|
|
|
|
if string_buffer ~= ""
|
|
|
|
if #lua.bits > 0 then lua\append ".."
|
|
|
|
lua\append repr(string_buffer)
|
|
|
|
string_buffer = ""
|
|
|
|
bit_lua = bit\as_lua(nomsu)
|
|
|
|
unless bit_lua.is_value
|
2018-05-16 18:12:56 -07:00
|
|
|
error "Cannot use #{colored.yellow repr(bit)} as a string interpolation value, since it's not an expression.", 0
|
2018-04-17 14:18:23 -07:00
|
|
|
if #lua.bits > 0 then lua\append ".."
|
|
|
|
if bit.type != "Text"
|
2018-05-16 18:12:56 -07:00
|
|
|
bit_lua = Lua.Value(nil, "stringify(",bit_lua,")")
|
2018-04-17 14:18:23 -07:00
|
|
|
lua\append bit_lua
|
|
|
|
|
|
|
|
if string_buffer ~= "" or #lua.bits == 0
|
|
|
|
if #lua.bits > 0 then lua\append ".."
|
|
|
|
lua\append repr(string_buffer)
|
|
|
|
|
|
|
|
if #lua.bits > 1
|
|
|
|
lua\parenthesize!
|
|
|
|
return lua
|
|
|
|
|
2018-04-24 20:16:46 -07:00
|
|
|
as_nomsu: (inline=false)=>
|
|
|
|
if inline
|
2018-05-16 18:12:56 -07:00
|
|
|
nomsu = Nomsu(nil, '"')
|
|
|
|
for bit in *@
|
2018-04-24 20:16:46 -07:00
|
|
|
if type(bit) == 'string'
|
2018-04-28 15:25:12 -07:00
|
|
|
-- TODO: unescape better?
|
2018-04-24 20:16:46 -07:00
|
|
|
nomsu\append (bit\gsub("\\","\\\\")\gsub("\n","\\n"))
|
|
|
|
else
|
|
|
|
interp_nomsu = bit\as_nomsu(true)
|
|
|
|
if interp_nomsu
|
|
|
|
if bit.type != "Word" and bit.type != "List" and bit.type != "Dict" and bit.type != "Text"
|
|
|
|
interp_nomsu\parenthesize!
|
|
|
|
nomsu\append "\\", interp_nomsu
|
|
|
|
else return nil
|
|
|
|
nomsu\append '"'
|
2018-04-25 16:04:46 -07:00
|
|
|
return nomsu
|
2018-04-24 20:16:46 -07:00
|
|
|
else
|
2018-04-25 16:04:46 -07:00
|
|
|
inline_version = @as_nomsu(true)
|
2018-04-25 16:30:49 -07:00
|
|
|
if inline_version and #inline_version <= MAX_LINE
|
2018-04-25 16:04:46 -07:00
|
|
|
return inline_version
|
2018-05-16 18:12:56 -07:00
|
|
|
nomsu = Nomsu(nil, '".."\n ')
|
|
|
|
for i, bit in ipairs @
|
2018-04-24 20:16:46 -07:00
|
|
|
if type(bit) == 'string'
|
|
|
|
nomsu\append (bit\gsub("\\","\\\\")\gsub("\n","\n "))
|
|
|
|
else
|
2018-04-25 16:04:46 -07:00
|
|
|
interp_nomsu = bit\as_nomsu(true)
|
2018-04-24 20:16:46 -07:00
|
|
|
if interp_nomsu
|
|
|
|
if bit.type != "Word" and bit.type != "List" and bit.type != "Dict" and bit.type != "Text"
|
|
|
|
interp_nomsu\parenthesize!
|
|
|
|
nomsu\append "\\", interp_nomsu
|
|
|
|
else
|
|
|
|
interp_nomsu = bit\as_nomsu!
|
|
|
|
return nil unless interp_nomsu
|
2018-04-25 16:04:46 -07:00
|
|
|
nomsu\append "\\\n ", interp_nomsu
|
2018-05-16 18:12:56 -07:00
|
|
|
if i < #@
|
2018-04-25 16:04:46 -07:00
|
|
|
nomsu\append "\n .."
|
2018-04-24 20:16:46 -07:00
|
|
|
return nomsu
|
|
|
|
|
2018-05-16 18:12:56 -07:00
|
|
|
Tree "List", 'multi',
|
2018-04-17 14:18:23 -07:00
|
|
|
as_lua: (nomsu)=>
|
2018-05-16 18:12:56 -07:00
|
|
|
lua = Lua.Value nil, "{"
|
2018-04-17 14:18:23 -07:00
|
|
|
line_length = 0
|
2018-05-16 18:12:56 -07:00
|
|
|
for i, item in ipairs @
|
2018-04-17 14:18:23 -07:00
|
|
|
item_lua = item\as_lua(nomsu)
|
|
|
|
unless item_lua.is_value
|
2018-05-16 18:12:56 -07:00
|
|
|
error "Cannot use #{colored.yellow repr(item)} as a list item, since it's not an expression.", 0
|
2018-04-17 14:18:23 -07:00
|
|
|
lua\append item_lua
|
2018-04-24 20:38:25 -07:00
|
|
|
item_string = tostring(item_lua)
|
|
|
|
last_line = item_string\match("[^\n]*$")
|
|
|
|
if item_string\match("\n")
|
2018-04-17 14:18:23 -07:00
|
|
|
line_length = #last_line
|
|
|
|
else
|
|
|
|
line_length += #last_line
|
2018-05-16 18:12:56 -07:00
|
|
|
if i < #@
|
2018-04-25 16:30:49 -07:00
|
|
|
if line_length >= MAX_LINE
|
2018-05-03 16:30:55 -07:00
|
|
|
lua\append ",\n "
|
2018-04-17 14:18:23 -07:00
|
|
|
line_length = 0
|
|
|
|
else
|
|
|
|
lua\append ", "
|
|
|
|
line_length += 2
|
|
|
|
lua\append "}"
|
|
|
|
return lua
|
|
|
|
|
2018-04-24 20:16:46 -07:00
|
|
|
as_nomsu: (inline=false)=>
|
|
|
|
if inline
|
2018-05-16 18:12:56 -07:00
|
|
|
nomsu = Nomsu(nil, "[")
|
|
|
|
for i, item in ipairs @
|
2018-04-24 20:16:46 -07:00
|
|
|
item_nomsu = item\as_nomsu(true)
|
|
|
|
return nil unless item_nomsu
|
|
|
|
if i > 1
|
|
|
|
nomsu\append ", "
|
|
|
|
nomsu\append item_nomsu
|
|
|
|
nomsu\append "]"
|
|
|
|
return nomsu
|
|
|
|
else
|
2018-04-25 16:04:46 -07:00
|
|
|
inline_version = @as_nomsu(true)
|
2018-04-25 16:30:49 -07:00
|
|
|
if inline_version and #inline_version <= MAX_LINE
|
2018-04-25 16:04:46 -07:00
|
|
|
return inline_version
|
2018-05-16 18:12:56 -07:00
|
|
|
nomsu = Nomsu(nil, "[..]")
|
|
|
|
line = Nomsu(nil, "\n ")
|
|
|
|
for item in *@
|
2018-04-24 20:16:46 -07:00
|
|
|
item_nomsu = item\as_nomsu(true)
|
2018-04-25 16:30:49 -07:00
|
|
|
if item_nomsu and #line + #", " + #item_nomsu <= MAX_LINE
|
2018-04-24 20:16:46 -07:00
|
|
|
if #line.bits > 1
|
|
|
|
line\append ", "
|
|
|
|
line\append item_nomsu
|
|
|
|
else
|
|
|
|
unless item_nomsu
|
|
|
|
item_nomsu = item\as_nomsu!
|
|
|
|
return nil unless item_nomsu
|
|
|
|
if #line.bits > 1
|
|
|
|
nomsu\append line
|
2018-05-16 18:12:56 -07:00
|
|
|
line = Nomsu(nil, "\n ")
|
2018-04-24 20:16:46 -07:00
|
|
|
line\append item_nomsu
|
|
|
|
if #line.bits > 1
|
|
|
|
nomsu\append line
|
|
|
|
return nomsu
|
|
|
|
|
2018-05-16 18:12:56 -07:00
|
|
|
Tree "Dict", 'multi',
|
2018-04-17 14:18:23 -07:00
|
|
|
as_lua: (nomsu)=>
|
2018-05-16 18:12:56 -07:00
|
|
|
lua = Lua.Value nil, "{"
|
2018-04-17 14:18:23 -07:00
|
|
|
line_length = 0
|
2018-05-16 18:12:56 -07:00
|
|
|
for i, entry in ipairs @
|
2018-05-16 15:44:07 -07:00
|
|
|
entry_lua = entry\as_lua(nomsu)
|
|
|
|
lua\append entry_lua
|
|
|
|
entry_lua_str = tostring(entry_lua)
|
2018-04-17 14:18:23 -07:00
|
|
|
-- TODO: maybe make this more accurate? It's only a heuristic, so eh...
|
2018-05-16 15:44:07 -07:00
|
|
|
last_line = entry_lua_str\match("\n([^\n]*)$")
|
|
|
|
if last_line
|
2018-04-17 14:18:23 -07:00
|
|
|
line_length = #last_line
|
|
|
|
else
|
2018-05-16 15:44:07 -07:00
|
|
|
line_length += #entry_lua_str
|
2018-05-16 18:12:56 -07:00
|
|
|
if i < #@
|
2018-04-25 16:30:49 -07:00
|
|
|
if line_length >= MAX_LINE
|
2018-05-03 16:30:55 -07:00
|
|
|
lua\append ",\n "
|
2018-04-17 14:18:23 -07:00
|
|
|
line_length = 0
|
|
|
|
else
|
|
|
|
lua\append ", "
|
|
|
|
line_length += 2
|
|
|
|
lua\append "}"
|
|
|
|
return lua
|
|
|
|
|
2018-04-24 20:16:46 -07:00
|
|
|
as_nomsu: (inline=false)=>
|
|
|
|
if inline
|
2018-05-16 18:12:56 -07:00
|
|
|
nomsu = Nomsu(nil, "{")
|
|
|
|
for i, entry in ipairs @
|
2018-05-16 15:44:07 -07:00
|
|
|
entry_nomsu = entry\as_nomsu(true)
|
|
|
|
return nil unless entry_nomsu
|
2018-04-24 20:16:46 -07:00
|
|
|
if i > 1
|
|
|
|
nomsu\append ", "
|
2018-05-16 15:44:07 -07:00
|
|
|
nomsu\append entry_nomsu
|
2018-04-24 20:16:46 -07:00
|
|
|
nomsu\append "}"
|
|
|
|
return nomsu
|
|
|
|
else
|
2018-04-25 16:04:46 -07:00
|
|
|
inline_version = @as_nomsu(true)
|
|
|
|
if inline_version then return inline_version
|
2018-05-16 18:12:56 -07:00
|
|
|
nomsu = Nomsu(nil, "{..}")
|
|
|
|
line = Nomsu(nil, "\n ")
|
|
|
|
for entry in *@
|
2018-05-16 15:44:07 -07:00
|
|
|
entry_nomsu = entry\as_nomsu!
|
|
|
|
return nil unless entry_nomsu
|
|
|
|
if #line + #tostring(entry_nomsu) <= MAX_LINE
|
2018-04-24 20:16:46 -07:00
|
|
|
if #line.bits > 1
|
|
|
|
line\append ", "
|
2018-05-16 15:44:07 -07:00
|
|
|
line\append entry_nomsu
|
2018-04-24 20:16:46 -07:00
|
|
|
else
|
|
|
|
if #line.bits > 1
|
|
|
|
nomsu\append line
|
2018-05-16 18:12:56 -07:00
|
|
|
line = Nomsu(nil, "\n ")
|
2018-05-16 15:44:07 -07:00
|
|
|
line\append entry_nomsu
|
2018-04-24 20:16:46 -07:00
|
|
|
if #line.bits > 1
|
|
|
|
nomsu\append line
|
|
|
|
return nomsu
|
|
|
|
|
2018-05-16 18:12:56 -07:00
|
|
|
Tree "DictEntry", 'multi',
|
2018-05-16 15:44:07 -07:00
|
|
|
as_lua: (nomsu)=>
|
2018-05-16 18:12:56 -07:00
|
|
|
key, value = @[1], @[2]
|
2018-05-16 15:44:07 -07:00
|
|
|
key_lua = key\as_lua(nomsu)
|
|
|
|
unless key_lua.is_value
|
2018-05-16 18:12:56 -07:00
|
|
|
error "Cannot use #{colored.yellow repr(key)} as a dict key, since it's not an expression.", 0
|
|
|
|
value_lua = value and value\as_lua(nomsu) or Lua.Value(nil, "true")
|
2018-05-16 15:44:07 -07:00
|
|
|
unless value_lua.is_value
|
2018-05-16 18:12:56 -07:00
|
|
|
error "Cannot use #{colored.yellow repr(value)} as a dict value, since it's not an expression.", 0
|
2018-05-16 15:44:07 -07:00
|
|
|
key_str = tostring(key_lua)\match([=[["']([a-zA-Z_][a-zA-Z0-9_]*)['"]]=])
|
|
|
|
return if key_str
|
2018-05-16 18:12:56 -07:00
|
|
|
Lua nil, key_str,"=",value_lua
|
2018-05-16 15:44:07 -07:00
|
|
|
elseif tostring(key_lua)\sub(1,1) == "["
|
|
|
|
-- NOTE: this *must* use a space after the [ to avoid freaking out
|
|
|
|
-- Lua's parser if the inner expression is a long string. Lua
|
|
|
|
-- parses x[[[y]]] as x("[y]"), not as x["y"]
|
2018-05-16 18:12:56 -07:00
|
|
|
Lua nil, "[ ",key_lua,"]=",value_lua
|
2018-05-16 15:44:07 -07:00
|
|
|
else
|
2018-05-16 18:12:56 -07:00
|
|
|
Lua nil, "[",key_lua,"]=",value_lua
|
2018-05-16 15:44:07 -07:00
|
|
|
|
|
|
|
as_nomsu: (inline=true)=>
|
2018-05-16 18:12:56 -07:00
|
|
|
key, value = @[1], @[2]
|
2018-05-16 15:44:07 -07:00
|
|
|
key_nomsu = key\as_nomsu(true)
|
|
|
|
return nil unless key_nomsu
|
|
|
|
if key.type == "Action" or key.type == "Block"
|
|
|
|
key_nomsu\parenthesize!
|
|
|
|
value_nomsu = if value
|
|
|
|
value\as_nomsu(true)
|
2018-05-16 18:12:56 -07:00
|
|
|
else Nomsu(nil, "")
|
2018-05-16 15:44:07 -07:00
|
|
|
if inline and not value_nomsu then return nil
|
|
|
|
if not value_nomsu
|
|
|
|
return nil if inline
|
|
|
|
value_nomsu = value\as_nomsu!
|
|
|
|
return nil unless value_nomsu
|
2018-05-16 18:12:56 -07:00
|
|
|
return Nomsu nil, key_nomsu, ":", value_nomsu
|
2018-05-16 15:44:07 -07:00
|
|
|
|
2018-05-03 21:56:07 -07:00
|
|
|
|
2018-05-16 18:12:56 -07:00
|
|
|
Tree "IndexChain", 'multi',
|
2018-04-17 14:18:23 -07:00
|
|
|
as_lua: (nomsu)=>
|
2018-05-16 18:12:56 -07:00
|
|
|
lua = @[1]\as_lua(nomsu)
|
2018-04-17 14:18:23 -07:00
|
|
|
unless lua.is_value
|
2018-05-16 18:12:56 -07:00
|
|
|
error "Cannot index #{colored.yellow repr(@[1])}, since it's not an expression.", 0
|
2018-04-19 17:27:42 -07:00
|
|
|
first_char = tostring(lua)\sub(1,1)
|
|
|
|
if first_char == "{" or first_char == '"' or first_char == "["
|
2018-04-17 14:18:23 -07:00
|
|
|
lua\parenthesize!
|
|
|
|
|
2018-05-16 18:12:56 -07:00
|
|
|
for i=2,#@
|
|
|
|
key = @[i]
|
2018-04-17 14:18:23 -07:00
|
|
|
key_lua = key\as_lua(nomsu)
|
|
|
|
unless key_lua.is_value
|
2018-05-16 18:12:56 -07:00
|
|
|
error "Cannot use #{colored.yellow repr(key)} as an index, since it's not an expression.", 0
|
|
|
|
key_lua_str = tostring(key_lua)
|
|
|
|
if lua_id = key_lua_str\match("^['\"]([a-zA-Z_][a-zA-Z0-9_]*)['\"]$")
|
|
|
|
lua\append ".#{lua_id}"
|
|
|
|
elseif key_lua_str\sub(1,1) == '['
|
|
|
|
-- NOTE: this *must* use a space after the [ to avoid freaking out
|
|
|
|
-- Lua's parser if the inner expression is a long string. Lua
|
|
|
|
-- parses x[[[y]]] as x("[y]"), not as x["y"]
|
|
|
|
lua\append "[ ",key_lua," ]"
|
2018-04-17 14:18:23 -07:00
|
|
|
else
|
|
|
|
lua\append "[",key_lua,"]"
|
|
|
|
return lua
|
|
|
|
|
2018-04-24 20:16:46 -07:00
|
|
|
as_nomsu: (inline=false)=>
|
2018-05-16 18:12:56 -07:00
|
|
|
nomsu = Nomsu!
|
|
|
|
for i, bit in ipairs @
|
2018-04-24 20:16:46 -07:00
|
|
|
if i > 1
|
|
|
|
nomsu\append "."
|
|
|
|
bit_nomsu = bit\as_nomsu(true)
|
|
|
|
return nil unless bit_nomsu
|
2018-05-10 22:47:03 -07:00
|
|
|
if bit.type == "Action" or bit.type == "Block"
|
|
|
|
bit_nomsu\parenthesize!
|
2018-04-24 20:16:46 -07:00
|
|
|
nomsu\append bit_nomsu
|
|
|
|
return nomsu
|
|
|
|
|
2018-05-16 18:12:56 -07:00
|
|
|
Tree "Number", 'single',
|
2018-04-17 14:18:23 -07:00
|
|
|
as_lua: (nomsu)=>
|
2018-05-16 18:12:56 -07:00
|
|
|
Lua.Value(nil, tostring(@value))
|
2018-04-24 20:16:46 -07:00
|
|
|
|
|
|
|
as_nomsu: (inline=false)=>
|
2018-05-16 18:12:56 -07:00
|
|
|
return Nomsu(nil, tostring(@value))
|
2018-04-17 14:18:23 -07:00
|
|
|
|
2018-05-16 18:12:56 -07:00
|
|
|
Tree "Var", 'single',
|
2018-05-15 15:21:32 -07:00
|
|
|
as_lua_id: (v)->
|
|
|
|
"_"..(v\gsub("%W", (c)-> if c == "_" then "__" else ("_%x")\format(c\byte!)))
|
|
|
|
|
2018-04-17 14:18:23 -07:00
|
|
|
as_lua: (nomsu)=>
|
2018-05-16 18:12:56 -07:00
|
|
|
Lua.Value(nil, self.as_lua_id(@value))
|
2018-04-17 14:18:23 -07:00
|
|
|
|
2018-04-24 20:16:46 -07:00
|
|
|
as_nomsu: (inline=false)=>
|
2018-05-16 18:12:56 -07:00
|
|
|
return Nomsu(nil, "%", @value)
|
2018-04-24 20:16:46 -07:00
|
|
|
|
2018-05-16 18:12:56 -07:00
|
|
|
Tree "Word", 'single',
|
2018-04-17 14:18:23 -07:00
|
|
|
as_lua: (nomsu)=>
|
|
|
|
error("Attempt to convert Word to lua")
|
|
|
|
|
2018-04-24 20:16:46 -07:00
|
|
|
as_nomsu: (inline=false)=>
|
2018-05-16 18:12:56 -07:00
|
|
|
return Nomsu(nil, @value)
|
2018-04-24 20:16:46 -07:00
|
|
|
|
2018-05-16 18:12:56 -07:00
|
|
|
Tree "Comment", 'single',
|
2018-04-17 14:18:23 -07:00
|
|
|
as_lua: (nomsu)=>
|
2018-05-16 18:12:56 -07:00
|
|
|
Lua(nil, "--"..@value\gsub("\n","\n--").."\n")
|
2018-04-17 14:18:23 -07:00
|
|
|
|
2018-04-24 20:16:46 -07:00
|
|
|
as_nomsu: (inline=false)=>
|
|
|
|
return nil if inline
|
|
|
|
if @value\match("\n")
|
2018-05-16 18:12:56 -07:00
|
|
|
return Nomsu(nil, "#..", @value\gsub("\n", "\n "))
|
2018-04-24 20:16:46 -07:00
|
|
|
else
|
2018-05-16 18:12:56 -07:00
|
|
|
return Nomsu(nil, "#", @value)
|
2018-04-24 20:16:46 -07:00
|
|
|
|
2018-04-17 14:18:23 -07:00
|
|
|
return Types
|