2018-04-17 14:18:23 -07:00
local utils = require ( ' utils ' )
local repr , stringify , min , max , equivalent , set , is_list , sum
repr , stringify , min , max , equivalent , set , is_list , sum = utils.repr , utils.stringify , utils.min , utils.max , utils.equivalent , utils.set , utils.is_list , utils.sum
local immutable = require ( ' immutable ' )
local insert , remove , concat
do
local _obj_0 = table
insert , remove , concat = _obj_0.insert , _obj_0.remove , _obj_0.concat
end
2018-04-25 15:37:13 -07:00
local Lua , Nomsu , Location
2018-04-17 14:18:23 -07:00
do
2018-04-26 14:00:01 -07:00
local _obj_0 = require ( " code_obj " )
2018-04-25 15:37:13 -07:00
Lua , Nomsu , Location = _obj_0.Lua , _obj_0.Nomsu , _obj_0.Location
2018-04-17 14:18:23 -07:00
end
2018-04-25 16:30:49 -07:00
local MAX_LINE = 80
2018-04-17 14:18:23 -07:00
local Types = { }
Types.is_node = function ( n )
return type ( n ) == ' userdata ' and getmetatable ( n ) and Types [ n.type ] == getmetatable ( n )
end
local Tree
2018-05-16 18:12:56 -07:00
Tree = function ( name , kind , methods )
assert ( ( kind == ' single ' ) or ( kind == ' multi ' ) )
local is_multi = ( kind == ' multi ' )
2018-04-17 14:18:23 -07:00
do
methods.with_value = function ( self , value )
2018-05-16 18:12:56 -07:00
return getmetatable ( self ) ( value )
2018-04-17 14:18:23 -07:00
end
methods.type = name
methods.name = name
2018-05-16 18:12:56 -07:00
methods.is_multi = is_multi
if is_multi then
methods.__tostring = function ( self )
return tostring ( self.name ) .. " ( " .. tostring ( table.concat ( ( function ( )
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1 , # self do
local v = self [ _index_0 ]
_accum_0 [ _len_0 ] = repr ( v )
_len_0 = _len_0 + 1
end
return _accum_0
end ) ( ) , ' , ' ) ) .. " ) "
2018-04-19 19:23:15 -07:00
end
2018-05-16 18:12:56 -07:00
methods.map = function ( self , fn )
do
local ret = fn ( self )
if ret then
return ret
end
2018-05-16 15:44:07 -07:00
end
2018-05-16 18:12:56 -07:00
local new_vals
do
2018-05-16 15:44:07 -07:00
local _accum_0 = { }
local _len_0 = 1
2018-05-16 18:12:56 -07:00
for _index_0 = 1 , # self do
local v = self [ _index_0 ]
2018-05-16 15:44:07 -07:00
_accum_0 [ _len_0 ] = v.map and v : map ( fn ) or v
_len_0 = _len_0 + 1
end
2018-05-16 18:12:56 -07:00
new_vals = _accum_0
end
local ret = getmetatable ( self ) ( unpack ( new_vals ) )
return ret
end
else
methods.__tostring = function ( self )
return tostring ( self.name ) .. " ( " .. tostring ( repr ( self.value ) ) .. " ) "
end
methods.map = function ( self , fn )
return fn ( self ) or self
2018-05-16 15:44:07 -07:00
end
end
2018-04-17 14:18:23 -07:00
end
2018-05-16 18:12:56 -07:00
if is_multi then
Types [ name ] = immutable ( nil , methods )
else
Types [ name ] = immutable ( {
" value "
} , methods )
end
2018-04-17 14:18:23 -07:00
end
2018-05-16 18:12:56 -07:00
Tree ( " EscapedNomsu " , ' single ' , {
2018-04-17 14:18:23 -07:00
as_lua = function ( self , nomsu )
2018-05-16 18:12:56 -07:00
local make_tree
make_tree = function ( t )
if type ( t ) ~= ' userdata ' then
return repr ( t )
end
if t.is_multi then
local bits
do
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1 , # t do
local bit = t [ _index_0 ]
_accum_0 [ _len_0 ] = make_tree ( bit )
_len_0 = _len_0 + 1
end
bits = _accum_0
end
return t.type .. " ( " .. table.concat ( bits , " , " ) .. " ) "
else
return t.type .. " ( " .. make_tree ( t.value ) .. " ) "
end
end
return Lua.Value ( nil , make_tree ( self.value ) )
2018-04-24 20:38:25 -07:00
end ,
as_nomsu = function ( self , inline )
if inline == nil then
inline = false
end
local nomsu = self.value : as_nomsu ( true )
if nomsu == nil and not inline then
nomsu = self.value : as_nomsu ( )
2018-05-16 18:12:56 -07:00
return nomsu and Nomsu ( nil , " \\ : \n " , nomsu )
2018-04-24 20:38:25 -07:00
end
2018-05-16 18:12:56 -07:00
return nomsu and Nomsu ( nil , " \\ ( " , nomsu , " ) " )
end ,
map = function ( self , fn )
return fn ( self ) or self : map ( fn )
2018-04-17 14:18:23 -07:00
end
} )
2018-05-16 18:12:56 -07:00
Tree ( " Block " , ' multi ' , {
2018-04-17 14:18:23 -07:00
as_lua = function ( self , nomsu )
2018-05-16 18:12:56 -07:00
local lua = Lua ( )
for i , line in ipairs ( self ) do
2018-04-17 14:18:23 -07:00
local line_lua = line : as_lua ( nomsu )
2018-04-24 20:38:25 -07:00
if i > 1 then
2018-04-17 14:18:23 -07:00
lua : append ( " \n " )
end
2018-05-15 14:53:37 -07:00
lua : append ( line_lua : as_statements ( ) )
2018-04-17 14:18:23 -07:00
end
return lua
2018-04-24 20:38:25 -07:00
end ,
as_nomsu = function ( self , inline )
if inline == nil then
inline = false
end
if inline then
2018-05-16 18:12:56 -07:00
local nomsu = Nomsu ( )
for i , line in ipairs ( self ) do
2018-04-25 17:43:48 -07:00
if i > 1 then
nomsu : append ( " ; " )
end
local line_nomsu = line : as_nomsu ( true )
if not ( line_nomsu ) then
return nil
end
nomsu : append ( line_nomsu )
end
return nomsu
2018-04-24 20:38:25 -07:00
end
2018-05-16 18:12:56 -07:00
local nomsu = Nomsu ( )
for i , line in ipairs ( self ) do
2018-04-28 15:20:48 -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 < # self then
2018-04-24 20:38:25 -07:00
nomsu : append ( " \n " )
2018-05-10 22:47:03 -07:00
if tostring ( line ) : match ( " \n " ) then
nomsu : append ( " \n " )
end
2018-04-24 20:38:25 -07:00
end
end
return nomsu
2018-04-17 14:18:23 -07:00
end
} )
2018-05-03 16:30:55 -07:00
local 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 = function ( self , nomsu )
local stub = self : get_stub ( )
2018-05-03 22:33:44 -07:00
local compile_action = nomsu.environment . COMPILE_ACTIONS [ stub ]
if compile_action then
2018-04-17 14:18:23 -07:00
local args
do
local _accum_0 = { }
local _len_0 = 1
2018-05-16 18:12:56 -07:00
for _index_0 = 1 , # self do
local arg = self [ _index_0 ]
2018-04-17 14:18:23 -07:00
if arg.type ~= " Word " then
_accum_0 [ _len_0 ] = arg
_len_0 = _len_0 + 1
end
end
args = _accum_0
end
2018-05-03 21:55:46 -07:00
do
local _accum_0 = { }
local _len_0 = 1
2018-05-03 22:33:44 -07:00
local _list_0 = nomsu.environment . ARG_ORDERS [ compile_action ] [ stub ]
2018-05-03 21:55:46 -07:00
for _index_0 = 1 , # _list_0 do
local p = _list_0 [ _index_0 ]
_accum_0 [ _len_0 ] = args [ p - 1 ]
_len_0 = _len_0 + 1
2018-04-17 14:18:23 -07:00
end
2018-05-03 21:55:46 -07:00
args = _accum_0
2018-04-17 14:18:23 -07:00
end
2018-05-03 22:33:44 -07:00
local ret = compile_action ( self , unpack ( args ) )
2018-05-10 22:47:03 -07:00
if not ret then
error ( " Failed to produce any Lua " )
end
2018-04-19 19:23:15 -07:00
return ret
2018-04-17 14:18:23 -07:00
end
2018-05-03 21:55:46 -07:00
local action = rawget ( nomsu.environment . ACTIONS , stub )
2018-05-16 18:12:56 -07:00
local lua = Lua.Value ( )
2018-05-03 21:55:46 -07:00
if not action and math_expression : match ( stub ) then
2018-05-16 18:12:56 -07:00
for i , tok in ipairs ( self ) do
2018-04-17 14:18:23 -07:00
if tok.type == " Word " then
lua : append ( tok.value )
else
local tok_lua = tok : as_lua ( nomsu )
if not ( tok_lua.is_value ) then
2018-05-16 18:12:56 -07:00
error ( " non-expression value inside math expression: " .. tostring ( colored.yellow ( repr ( tok ) ) ) )
2018-04-17 14:18:23 -07:00
end
2018-05-03 16:37:48 -07:00
if tok.type == " Action " then
tok_lua : parenthesize ( )
end
2018-04-17 14:18:23 -07:00
lua : append ( tok_lua )
end
2018-05-16 18:12:56 -07:00
if i < # self then
2018-04-17 14:18:23 -07:00
lua : append ( " " )
end
end
return lua
end
local args = { }
2018-05-16 18:12:56 -07:00
for i , tok in ipairs ( self ) do
2018-04-17 14:18:23 -07:00
local _continue_0 = false
repeat
if tok.type == " Word " then
_continue_0 = true
break
end
local arg_lua = tok : as_lua ( nomsu )
if not ( arg_lua.is_value ) then
2018-05-16 18:12:56 -07:00
error ( " Cannot use: \n " .. tostring ( colored.yellow ( repr ( tok ) ) ) .. " \n as an argument to " .. tostring ( stub ) .. " , since it's not an expression, it produces: " .. tostring ( repr ( arg_lua ) ) , 0 )
2018-04-17 14:18:23 -07:00
end
insert ( args , arg_lua )
_continue_0 = true
until true
if not _continue_0 then
break
end
end
2018-05-03 21:55:46 -07:00
if action then
2018-04-17 14:18:23 -07:00
do
local _accum_0 = { }
local _len_0 = 1
2018-05-03 21:55:46 -07:00
local _list_0 = nomsu.environment . ARG_ORDERS [ action ] [ stub ]
2018-04-17 14:18:23 -07:00
for _index_0 = 1 , # _list_0 do
local p = _list_0 [ _index_0 ]
_accum_0 [ _len_0 ] = args [ p ]
_len_0 = _len_0 + 1
end
args = _accum_0
end
end
2018-04-19 19:23:15 -07:00
lua : append ( " ACTIONS[ " , repr ( stub ) , " ]( " )
2018-04-17 14:18:23 -07:00
for i , arg in ipairs ( args ) do
lua : append ( arg )
if i < # args then
lua : append ( " , " )
end
end
lua : append ( " ) " )
return lua
end ,
2018-04-19 19:23:15 -07:00
get_stub = function ( self , include_names )
if include_names == nil then
include_names = false
end
local bits
if include_names then
do
local _accum_0 = { }
local _len_0 = 1
2018-05-16 18:12:56 -07:00
for _index_0 = 1 , # self do
local t = self [ _index_0 ]
2018-04-19 19:23:15 -07:00
_accum_0 [ _len_0 ] = ( t.type == " Word " and t.value or " % " .. tostring ( t.value ) )
_len_0 = _len_0 + 1
end
bits = _accum_0
end
else
do
local _accum_0 = { }
local _len_0 = 1
2018-05-16 18:12:56 -07:00
for _index_0 = 1 , # self do
local t = self [ _index_0 ]
2018-04-19 19:23:15 -07:00
_accum_0 [ _len_0 ] = ( t.type == " Word " and t.value or " % " )
_len_0 = _len_0 + 1
end
bits = _accum_0
2018-04-17 14:18:23 -07:00
end
2018-04-19 19:23:15 -07:00
end
return concat ( bits , " " )
2018-04-24 20:38:25 -07:00
end ,
2018-04-28 15:20:48 -07:00
as_nomsu = function ( self , inline , can_use_colon )
2018-04-24 20:38:25 -07:00
if inline == nil then
inline = false
end
2018-04-28 15:20:48 -07:00
if can_use_colon == nil then
can_use_colon = false
end
2018-04-24 20:38:25 -07:00
if inline then
2018-05-16 18:12:56 -07:00
local nomsu = Nomsu ( )
for i , bit in ipairs ( self ) do
2018-04-24 20:38:25 -07:00
if bit.type == " Word " then
if i > 1 then
nomsu : append ( " " )
end
nomsu : append ( bit.value )
else
local arg_nomsu = bit : as_nomsu ( true )
if not ( arg_nomsu ) then
return nil
end
if not ( i == 1 ) then
nomsu : append ( " " )
end
2018-04-25 16:04:46 -07:00
if bit.type == " Action " or bit.type == " Block " then
2018-04-24 20:38:25 -07:00
arg_nomsu : parenthesize ( )
end
nomsu : append ( arg_nomsu )
end
end
return nomsu
else
2018-05-16 18:12:56 -07:00
local nomsu = Nomsu ( )
2018-04-28 15:20:48 -07:00
local next_space = " "
local last_colon = nil
2018-05-16 18:12:56 -07:00
for i , bit in ipairs ( self ) do
2018-04-24 20:38:25 -07:00
if bit.type == " Word " then
2018-04-28 15:20:48 -07:00
nomsu : append ( next_space , bit.value )
next_space = " "
2018-04-24 20:38:25 -07:00
else
2018-04-28 20:45:17 -07:00
local arg_nomsu
if last_colon == i - 1 and bit.type == " Action " then
arg_nomsu = nil
elseif bit.type == " Block " then
arg_nomsu = nil
else
arg_nomsu = bit : as_nomsu ( true )
end
2018-04-25 16:30:49 -07:00
if arg_nomsu and # arg_nomsu < MAX_LINE then
2018-04-28 15:20:48 -07:00
if bit.type == " Action " then
if can_use_colon and i > 1 then
nomsu : append ( next_space : match ( " [^ ]* " ) , " : " , arg_nomsu )
next_space = " \n .. "
last_colon = i
else
nomsu : append ( next_space , " ( " , arg_nomsu , " ) " )
next_space = " "
end
else
nomsu : append ( next_space , arg_nomsu )
next_space = " "
2018-04-24 20:38:25 -07:00
end
else
2018-04-28 15:20:48 -07:00
arg_nomsu = bit : as_nomsu ( nil , true )
2018-04-24 20:38:25 -07:00
if not ( nomsu ) then
return nil
end
2018-04-28 15:20:48 -07:00
if bit.type ~= " List " and bit.type ~= " Dict " and bit.type ~= " Text " then
if i == 1 then
2018-05-16 18:12:56 -07:00
arg_nomsu = Nomsu ( nil , " (..) \n " , arg_nomsu )
2018-04-28 15:20:48 -07:00
else
2018-05-16 18:12:56 -07:00
arg_nomsu = Nomsu ( nil , " \n " , arg_nomsu )
2018-04-25 16:30:49 -07:00
end
2018-04-24 20:38:25 -07:00
end
2018-04-28 15:20:48 -07:00
if last_colon == i - 1 and ( bit.type == " Action " or bit.type == " Block " ) then
next_space = " "
end
nomsu : append ( next_space , arg_nomsu )
next_space = " \n .. "
2018-04-24 20:38:25 -07:00
end
2018-04-28 17:08:28 -07:00
if next_space == " " and # ( tostring ( nomsu ) : match ( " [^ \n ]*$ " ) ) > MAX_LINE then
next_space = " \n .. "
end
2018-04-24 20:38:25 -07:00
end
end
return nomsu
end
2018-04-17 14:18:23 -07:00
end
} )
2018-05-16 18:12:56 -07:00
Tree ( " Text " , ' multi ' , {
2018-04-17 14:18:23 -07:00
as_lua = function ( self , nomsu )
2018-05-16 18:12:56 -07:00
local lua = Lua.Value ( )
2018-04-17 14:18:23 -07:00
local string_buffer = " "
2018-05-16 18:12:56 -07:00
for _index_0 = 1 , # self do
2018-04-17 14:18:23 -07:00
local _continue_0 = false
repeat
2018-05-16 18:12:56 -07:00
local bit = self [ _index_0 ]
2018-04-17 14:18:23 -07:00
if type ( bit ) == " string " then
string_buffer = string_buffer .. bit
_continue_0 = true
break
end
if string_buffer ~= " " then
if # lua.bits > 0 then
lua : append ( " .. " )
end
lua : append ( repr ( string_buffer ) )
string_buffer = " "
end
local bit_lua = bit : as_lua ( nomsu )
if not ( bit_lua.is_value ) then
2018-05-16 18:12:56 -07:00
error ( " Cannot use " .. tostring ( colored.yellow ( repr ( bit ) ) ) .. " as a string interpolation value, since it's not an expression. " , 0 )
2018-04-17 14:18:23 -07:00
end
if # lua.bits > 0 then
lua : append ( " .. " )
end
if bit.type ~= " Text " then
2018-05-16 18:12:56 -07:00
bit_lua = Lua.Value ( nil , " stringify( " , bit_lua , " ) " )
2018-04-17 14:18:23 -07:00
end
lua : append ( bit_lua )
_continue_0 = true
until true
if not _continue_0 then
break
end
end
if string_buffer ~= " " or # lua.bits == 0 then
if # lua.bits > 0 then
lua : append ( " .. " )
end
lua : append ( repr ( string_buffer ) )
end
if # lua.bits > 1 then
lua : parenthesize ( )
end
return lua
2018-04-24 20:38:25 -07:00
end ,
as_nomsu = function ( self , inline )
if inline == nil then
inline = false
end
if inline then
2018-05-16 18:12:56 -07:00
local nomsu = Nomsu ( nil , ' " ' )
for _index_0 = 1 , # self do
local bit = self [ _index_0 ]
2018-04-24 20:38:25 -07:00
if type ( bit ) == ' string ' then
nomsu : append ( ( bit : gsub ( " \\ " , " \\ \\ " ) : gsub ( " \n " , " \\ n " ) ) )
else
local interp_nomsu = bit : as_nomsu ( true )
if interp_nomsu then
if bit.type ~= " Word " and bit.type ~= " List " and bit.type ~= " Dict " and bit.type ~= " Text " then
interp_nomsu : parenthesize ( )
end
nomsu : append ( " \\ " , interp_nomsu )
else
return nil
end
end
end
nomsu : append ( ' " ' )
2018-04-25 16:04:46 -07:00
return nomsu
2018-04-24 20:38:25 -07:00
else
2018-04-25 16:04:46 -07:00
local inline_version = self : as_nomsu ( true )
2018-04-25 16:30:49 -07:00
if inline_version and # inline_version <= MAX_LINE then
2018-04-25 16:04:46 -07:00
return inline_version
end
2018-05-16 18:12:56 -07:00
local nomsu = Nomsu ( nil , ' ".." \n ' )
for i , bit in ipairs ( self ) do
2018-04-24 20:38:25 -07:00
if type ( bit ) == ' string ' then
nomsu : append ( ( bit : gsub ( " \\ " , " \\ \\ " ) : gsub ( " \n " , " \n " ) ) )
else
2018-04-25 16:04:46 -07:00
local interp_nomsu = bit : as_nomsu ( true )
2018-04-24 20:38:25 -07:00
if interp_nomsu then
if bit.type ~= " Word " and bit.type ~= " List " and bit.type ~= " Dict " and bit.type ~= " Text " then
interp_nomsu : parenthesize ( )
end
nomsu : append ( " \\ " , interp_nomsu )
else
2018-04-25 16:04:46 -07:00
interp_nomsu = bit : as_nomsu ( )
2018-04-24 20:38:25 -07:00
if not ( interp_nomsu ) then
return nil
end
2018-04-25 15:37:13 -07:00
nomsu : append ( " \\ \n " , interp_nomsu )
2018-05-16 18:12:56 -07:00
if i < # self then
2018-04-25 15:37:13 -07:00
nomsu : append ( " \n .. " )
2018-04-24 20:38:25 -07:00
end
end
end
end
return nomsu
end
2018-04-17 14:18:23 -07:00
end
} )
2018-05-16 18:12:56 -07:00
Tree ( " List " , ' multi ' , {
2018-04-17 14:18:23 -07:00
as_lua = function ( self , nomsu )
2018-05-16 18:12:56 -07:00
local lua = Lua.Value ( nil , " { " )
2018-04-17 14:18:23 -07:00
local line_length = 0
2018-05-16 18:12:56 -07:00
for i , item in ipairs ( self ) do
2018-04-17 14:18:23 -07:00
local item_lua = item : as_lua ( nomsu )
if not ( item_lua.is_value ) then
2018-05-16 18:12:56 -07:00
error ( " Cannot use " .. tostring ( colored.yellow ( repr ( item ) ) ) .. " as a list item, since it's not an expression. " , 0 )
2018-04-17 14:18:23 -07:00
end
lua : append ( item_lua )
2018-04-24 20:38:25 -07:00
local item_string = tostring ( item_lua )
local last_line = item_string : match ( " [^ \n ]*$ " )
if item_string : match ( " \n " ) then
2018-04-17 14:18:23 -07:00
line_length = # last_line
else
line_length = line_length + # last_line
end
2018-05-16 18:12:56 -07:00
if i < # self then
2018-04-25 16:30:49 -07:00
if line_length >= MAX_LINE then
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 = line_length + 2
end
end
end
lua : append ( " } " )
return lua
2018-04-24 20:38:25 -07:00
end ,
as_nomsu = function ( self , inline )
if inline == nil then
inline = false
end
if inline then
2018-05-16 18:12:56 -07:00
local nomsu = Nomsu ( nil , " [ " )
for i , item in ipairs ( self ) do
2018-04-24 20:38:25 -07:00
local item_nomsu = item : as_nomsu ( true )
if not ( item_nomsu ) then
return nil
end
if i > 1 then
nomsu : append ( " , " )
end
nomsu : append ( item_nomsu )
end
nomsu : append ( " ] " )
return nomsu
else
2018-04-25 16:04:46 -07:00
local inline_version = self : as_nomsu ( true )
2018-04-25 16:30:49 -07:00
if inline_version and # inline_version <= MAX_LINE then
2018-04-25 16:04:46 -07:00
return inline_version
end
2018-05-16 18:12:56 -07:00
local nomsu = Nomsu ( nil , " [..] " )
local line = Nomsu ( nil , " \n " )
for _index_0 = 1 , # self do
local item = self [ _index_0 ]
2018-04-24 20:38:25 -07:00
local item_nomsu = item : as_nomsu ( true )
2018-04-25 16:30:49 -07:00
if item_nomsu and # line + # " , " + # item_nomsu <= MAX_LINE then
2018-04-24 20:38:25 -07:00
if # line.bits > 1 then
line : append ( " , " )
end
line : append ( item_nomsu )
else
if not ( item_nomsu ) then
item_nomsu = item : as_nomsu ( )
if not ( item_nomsu ) then
return nil
end
end
if # line.bits > 1 then
nomsu : append ( line )
2018-05-16 18:12:56 -07:00
line = Nomsu ( nil , " \n " )
2018-04-24 20:38:25 -07:00
end
line : append ( item_nomsu )
end
end
if # line.bits > 1 then
nomsu : append ( line )
end
return nomsu
end
2018-04-17 14:18:23 -07:00
end
} )
2018-05-16 18:12:56 -07:00
Tree ( " Dict " , ' multi ' , {
2018-04-17 14:18:23 -07:00
as_lua = function ( self , nomsu )
2018-05-16 18:12:56 -07:00
local lua = Lua.Value ( nil , " { " )
2018-04-17 14:18:23 -07:00
local line_length = 0
2018-05-16 18:12:56 -07:00
for i , entry in ipairs ( self ) do
2018-05-16 15:44:07 -07:00
local entry_lua = entry : as_lua ( nomsu )
lua : append ( entry_lua )
local entry_lua_str = tostring ( entry_lua )
local last_line = entry_lua_str : match ( " \n ([^ \n ]*)$ " )
if last_line then
2018-04-17 14:18:23 -07:00
line_length = # last_line
else
2018-05-16 15:44:07 -07:00
line_length = line_length + # entry_lua_str
2018-04-17 14:18:23 -07:00
end
2018-05-16 18:12:56 -07:00
if i < # self then
2018-04-25 16:30:49 -07:00
if line_length >= MAX_LINE then
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 = line_length + 2
end
end
end
lua : append ( " } " )
return lua
2018-04-24 20:38:25 -07:00
end ,
as_nomsu = function ( self , inline )
if inline == nil then
inline = false
end
if inline then
2018-05-16 18:12:56 -07:00
local nomsu = Nomsu ( nil , " { " )
for i , entry in ipairs ( self ) do
2018-05-16 15:44:07 -07:00
local entry_nomsu = entry : as_nomsu ( true )
if not ( entry_nomsu ) then
2018-04-24 20:38:25 -07:00
return nil
end
if i > 1 then
nomsu : append ( " , " )
end
2018-05-16 15:44:07 -07:00
nomsu : append ( entry_nomsu )
2018-04-24 20:38:25 -07:00
end
nomsu : append ( " } " )
return nomsu
else
2018-04-25 16:04:46 -07:00
local inline_version = self : as_nomsu ( true )
if inline_version then
return inline_version
end
2018-05-16 18:12:56 -07:00
local nomsu = Nomsu ( nil , " {..} " )
local line = Nomsu ( nil , " \n " )
for _index_0 = 1 , # self do
local entry = self [ _index_0 ]
2018-05-16 15:44:07 -07:00
local entry_nomsu = entry : as_nomsu ( )
if not ( entry_nomsu ) then
2018-04-24 20:38:25 -07:00
return nil
end
2018-05-16 15:44:07 -07:00
if # line + # tostring ( entry_nomsu ) <= MAX_LINE then
2018-04-24 20:38:25 -07:00
if # line.bits > 1 then
line : append ( " , " )
end
2018-05-16 15:44:07 -07:00
line : append ( entry_nomsu )
2018-04-24 20:38:25 -07:00
else
if # line.bits > 1 then
nomsu : append ( line )
2018-05-16 18:12:56 -07:00
line = Nomsu ( nil , " \n " )
2018-04-24 20:38:25 -07:00
end
2018-05-16 15:44:07 -07:00
line : append ( entry_nomsu )
2018-04-24 20:38:25 -07:00
end
end
if # line.bits > 1 then
nomsu : append ( line )
end
return nomsu
end
2018-05-16 15:44:07 -07:00
end
} )
2018-05-16 18:12:56 -07:00
Tree ( " DictEntry " , ' multi ' , {
2018-05-16 15:44:07 -07:00
as_lua = function ( self , nomsu )
2018-05-16 18:12:56 -07:00
local key , value = self [ 1 ] , self [ 2 ]
2018-05-16 15:44:07 -07:00
local key_lua = key : as_lua ( nomsu )
if not ( key_lua.is_value ) then
2018-05-16 18:12:56 -07:00
error ( " Cannot use " .. tostring ( colored.yellow ( repr ( key ) ) ) .. " as a dict key, since it's not an expression. " , 0 )
2018-05-16 15:44:07 -07:00
end
2018-05-16 18:12:56 -07:00
local value_lua = value and value : as_lua ( nomsu ) or Lua.Value ( nil , " true " )
2018-05-16 15:44:07 -07:00
if not ( value_lua.is_value ) then
2018-05-16 18:12:56 -07:00
error ( " Cannot use " .. tostring ( colored.yellow ( repr ( value ) ) ) .. " as a dict value, since it's not an expression. " , 0 )
2018-05-16 15:44:07 -07:00
end
local key_str = tostring ( key_lua ) : match ( [=[["']([a-zA-Z_][a-zA-Z0-9_]*)['"]]=] )
if key_str then
2018-05-16 18:12:56 -07:00
return Lua ( nil , key_str , " = " , value_lua )
2018-05-16 15:44:07 -07:00
elseif tostring ( key_lua ) : sub ( 1 , 1 ) == " [ " then
2018-05-16 18:12:56 -07:00
return Lua ( nil , " [ " , key_lua , " ]= " , value_lua )
2018-05-16 15:44:07 -07:00
else
2018-05-16 18:12:56 -07:00
return Lua ( nil , " [ " , key_lua , " ]= " , value_lua )
2018-05-16 15:44:07 -07:00
end
2018-05-03 21:55:46 -07:00
end ,
2018-05-16 15:44:07 -07:00
as_nomsu = function ( self , inline )
if inline == nil then
inline = true
end
2018-05-16 18:12:56 -07:00
local key , value = self [ 1 ] , self [ 2 ]
2018-05-16 15:44:07 -07:00
local key_nomsu = key : as_nomsu ( true )
if not ( key_nomsu ) then
return nil
end
if key.type == " Action " or key.type == " Block " then
key_nomsu : parenthesize ( )
end
local value_nomsu
if value then
value_nomsu = value : as_nomsu ( true )
else
2018-05-16 18:12:56 -07:00
value_nomsu = Nomsu ( nil , " " )
2018-05-16 15:44:07 -07:00
end
if inline and not value_nomsu then
return nil
end
if not value_nomsu then
if inline then
return nil
end
value_nomsu = value : as_nomsu ( )
if not ( value_nomsu ) then
return nil
2018-05-03 21:55:46 -07:00
end
2018-05-16 15:44:07 -07:00
end
2018-05-16 18:12:56 -07:00
return Nomsu ( nil , key_nomsu , " : " , value_nomsu )
2018-04-17 14:18:23 -07:00
end
} )
2018-05-16 18:12:56 -07:00
Tree ( " IndexChain " , ' multi ' , {
2018-04-17 14:18:23 -07:00
as_lua = function ( self , nomsu )
2018-05-16 18:12:56 -07:00
local lua = self [ 1 ] : as_lua ( nomsu )
2018-04-17 14:18:23 -07:00
if not ( lua.is_value ) then
2018-05-16 18:12:56 -07:00
error ( " Cannot index " .. tostring ( colored.yellow ( repr ( self [ 1 ] ) ) ) .. " , since it's not an expression. " , 0 )
2018-04-17 14:18:23 -07:00
end
2018-04-19 19:23:15 -07:00
local first_char = tostring ( lua ) : sub ( 1 , 1 )
if first_char == " { " or first_char == ' " ' or first_char == " [ " then
2018-04-17 14:18:23 -07:00
lua : parenthesize ( )
end
2018-05-16 18:12:56 -07:00
for i = 2 , # self do
local key = self [ i ]
local key_lua = key : as_lua ( nomsu )
if not ( key_lua.is_value ) then
error ( " Cannot use " .. tostring ( colored.yellow ( repr ( key ) ) ) .. " as an index, since it's not an expression. " , 0 )
end
local key_lua_str = tostring ( key_lua )
do
local lua_id = key_lua_str : match ( " ^[' \" ]([a-zA-Z_][a-zA-Z0-9_]*)[' \" ]$ " )
if lua_id then
lua : append ( " . " .. tostring ( lua_id ) )
elseif key_lua_str : sub ( 1 , 1 ) == ' [ ' then
lua : append ( " [ " , key_lua , " ] " )
2018-04-17 14:18:23 -07:00
else
lua : append ( " [ " , key_lua , " ] " )
end
end
end
return lua
2018-04-24 20:38:25 -07:00
end ,
as_nomsu = function ( self , inline )
if inline == nil then
inline = false
end
2018-05-16 18:12:56 -07:00
local nomsu = Nomsu ( )
for i , bit in ipairs ( self ) do
2018-04-24 20:38:25 -07:00
if i > 1 then
nomsu : append ( " . " )
end
local bit_nomsu = bit : as_nomsu ( true )
if not ( bit_nomsu ) then
return nil
end
2018-05-10 22:47:03 -07:00
if bit.type == " Action " or bit.type == " Block " then
bit_nomsu : parenthesize ( )
end
2018-04-24 20:38:25 -07:00
nomsu : append ( bit_nomsu )
end
return nomsu
2018-04-17 14:18:23 -07:00
end
} )
2018-05-16 18:12:56 -07:00
Tree ( " Number " , ' single ' , {
2018-04-17 14:18:23 -07:00
as_lua = function ( self , nomsu )
2018-05-16 18:12:56 -07:00
return Lua.Value ( nil , tostring ( self.value ) )
2018-04-24 20:38:25 -07:00
end ,
as_nomsu = function ( self , inline )
if inline == nil then
inline = false
end
2018-05-16 18:12:56 -07:00
return Nomsu ( nil , tostring ( self.value ) )
2018-04-17 14:18:23 -07:00
end
} )
2018-05-16 18:12:56 -07:00
Tree ( " Var " , ' single ' , {
2018-05-15 15:21:32 -07:00
as_lua_id = function ( v )
return " _ " .. ( v : gsub ( " %W " , function ( c )
if c == " _ " then
2018-04-17 14:18:23 -07:00
return " __ "
else
2018-05-15 15:21:32 -07:00
return ( " _%x " ) : format ( c : byte ( ) )
2018-04-17 14:18:23 -07:00
end
end ) )
2018-05-15 15:21:32 -07:00
end ,
as_lua = function ( self , nomsu )
2018-05-16 18:12:56 -07:00
return Lua.Value ( nil , self.as_lua_id ( self.value ) )
2018-04-24 20:38:25 -07:00
end ,
as_nomsu = function ( self , inline )
if inline == nil then
inline = false
end
2018-05-16 18:12:56 -07:00
return Nomsu ( nil , " % " , self.value )
2018-04-17 14:18:23 -07:00
end
} )
2018-05-16 18:12:56 -07:00
Tree ( " Word " , ' single ' , {
2018-04-17 14:18:23 -07:00
as_lua = function ( self , nomsu )
return error ( " Attempt to convert Word to lua " )
2018-04-24 20:38:25 -07:00
end ,
as_nomsu = function ( self , inline )
if inline == nil then
inline = false
end
2018-05-16 18:12:56 -07:00
return Nomsu ( nil , self.value )
2018-04-17 14:18:23 -07:00
end
} )
2018-05-16 18:12:56 -07:00
Tree ( " Comment " , ' single ' , {
2018-04-17 14:18:23 -07:00
as_lua = function ( self , nomsu )
2018-05-16 18:12:56 -07:00
return Lua ( nil , " -- " .. self.value : gsub ( " \n " , " \n -- " ) .. " \n " )
2018-04-24 20:38:25 -07:00
end ,
as_nomsu = function ( self , inline )
if inline == nil then
inline = false
end
if inline then
return nil
end
if self.value : match ( " \n " ) then
2018-05-16 18:12:56 -07:00
return Nomsu ( nil , " #.. " , self.value : gsub ( " \n " , " \n " ) )
2018-04-24 20:38:25 -07:00
else
2018-05-16 18:12:56 -07:00
return Nomsu ( nil , " # " , self.value )
2018-04-24 20:38:25 -07:00
end
2018-04-17 14:18:23 -07:00
end
} )
return Types