Converted DictEntry to be an actual tree, instead of a pseudo-tree, made 'parse % as %'
generate lua code with already-substituted tree literals instead of reparsing and substituting at parse time, and made some general optimizations.
This commit is contained in:
parent
3ffeaf1f5d
commit
af9dc07025
@ -6,108 +6,116 @@
|
|||||||
immediately
|
immediately
|
||||||
lua> ".."
|
lua> ".."
|
||||||
nomsu:define_compile_action("compile %actions to %lua", function(tree, \%actions, \%lua)
|
nomsu:define_compile_action("compile %actions to %lua", function(tree, \%actions, \%lua)
|
||||||
local lua = Lua(tree.source, "nomsu:define_compile_action(");
|
local lua = Lua(tree.source, "nomsu:define_compile_action(")
|
||||||
local stubs = {};
|
local stubs = {}
|
||||||
for i, action in ipairs(\%actions.value) do
|
for i, action in ipairs(\%actions.value) do
|
||||||
stubs[i] = action:get_stub(true);
|
stubs[i] = action:get_stub(true)
|
||||||
end
|
end
|
||||||
stubs = repr(stubs);
|
stubs = repr(stubs)
|
||||||
if #stubs > 80 then
|
if #stubs > 80 then
|
||||||
lua:append("\n ",stubs,",\n ");
|
lua:append("\n ",stubs,",\n ")
|
||||||
else
|
else
|
||||||
lua:append(stubs,", ");
|
lua:append(stubs,", ")
|
||||||
end
|
end
|
||||||
lua:append("function(tree");
|
lua:append("function(tree")
|
||||||
local args = {};
|
local args = {}
|
||||||
for i,tok in ipairs(\%actions.value[1].value) do
|
for i,tok in ipairs(\%actions.value[1].value) do
|
||||||
if tok.type == "Var" then args[#args+1] = tok:as_lua(nomsu); end
|
if tok.type == "Var" then args[#args+1] = tok:as_lua(nomsu) end
|
||||||
end
|
end
|
||||||
for i, arg in ipairs(args) do
|
for i, arg in ipairs(args) do
|
||||||
lua:append(", ");
|
lua:append(", ", arg)
|
||||||
lua:append(arg);
|
|
||||||
end
|
end
|
||||||
local body_lua = \%lua:as_lua(nomsu);
|
local body_lua = \%lua:as_lua(nomsu):as_statements("return ")
|
||||||
body_lua = body_lua:as_statements("return ");
|
body_lua:remove_free_vars(args)
|
||||||
body_lua:remove_free_vars(args);
|
body_lua:declare_locals()
|
||||||
body_lua:declare_locals();
|
lua:append(")\n ", body_lua, "\nend);")
|
||||||
lua:append(")\n ", body_lua, "\nend);");
|
return lua
|
||||||
return lua;
|
|
||||||
end);
|
end);
|
||||||
|
|
||||||
# Compile-time action to make actions
|
# Compile-time action to make actions
|
||||||
immediately
|
immediately
|
||||||
compile [action %actions %body] to
|
compile [action %actions %body] to
|
||||||
lua> ".."
|
lua> ".."
|
||||||
local lua = Lua(tree.source, "nomsu:define_action(");
|
local lua = Lua(tree.source, "nomsu:define_action(")
|
||||||
local stubs = {};
|
local stubs = {}
|
||||||
for i, action in ipairs(\%actions.value) do
|
for i, action in ipairs(\%actions.value) do
|
||||||
stubs[i] = action:get_stub(true);
|
stubs[i] = action:get_stub(true)
|
||||||
end
|
end
|
||||||
stubs = repr(stubs);
|
stubs = repr(stubs)
|
||||||
if #stubs > 80 then
|
if #stubs > 80 then
|
||||||
lua:append("\n ",stubs,",\n ");
|
lua:append("\n ",stubs,",\n ")
|
||||||
else
|
else
|
||||||
lua:append(stubs,", ");
|
lua:append(stubs,", ")
|
||||||
end
|
end
|
||||||
lua:append("function(");
|
lua:append("function(")
|
||||||
local args = {};
|
local args = {}
|
||||||
for i,tok in ipairs(\%actions.value[1].value) do
|
for i,tok in ipairs(\%actions.value[1].value) do
|
||||||
if tok.type == "Var" then args[#args+1] = tok:as_lua(nomsu); end
|
if tok.type == "Var" then args[#args+1] = tok:as_lua(nomsu) end
|
||||||
end
|
end
|
||||||
for i, arg in ipairs(args) do
|
for i, arg in ipairs(args) do
|
||||||
lua:append(arg);
|
lua:append(arg)
|
||||||
if i < #args then lua:append(", ") end
|
if i < #args then lua:append(", ") end
|
||||||
end
|
end
|
||||||
local body_lua = \%body:as_lua(nomsu);
|
local body_lua = \%body:as_lua(nomsu):as_statements("return ")
|
||||||
body_lua = body_lua:as_statements("return ");
|
body_lua:remove_free_vars(args)
|
||||||
body_lua:remove_free_vars(args);
|
body_lua:declare_locals()
|
||||||
body_lua:declare_locals();
|
|
||||||
lua:append(")\n ", body_lua, "\nend);")
|
lua:append(")\n ", body_lua, "\nend);")
|
||||||
return lua;
|
return lua
|
||||||
|
|
||||||
# Macro to make nomsu macros
|
# Macro to make nomsu macros
|
||||||
immediately
|
immediately
|
||||||
compile [parse %shorthand as %longhand] to
|
compile [parse %shorthand as %longhand] to
|
||||||
lua> ".."
|
lua> ".."
|
||||||
local lua = Lua(tree.source, "nomsu:define_compile_action(");
|
local lua = Lua(tree.source, "nomsu:define_compile_action(")
|
||||||
local stubs = {};
|
local stubs = {}
|
||||||
for i, action in ipairs(\%shorthand.value) do
|
for i, action in ipairs(\%shorthand.value) do
|
||||||
stubs[i] = action:get_stub(true);
|
stubs[i] = action:get_stub(true)
|
||||||
end
|
end
|
||||||
stubs = repr(stubs);
|
stubs = repr(stubs)
|
||||||
if #stubs > 80 then
|
if #stubs > 80 then
|
||||||
lua:append("\n ",stubs,",\n ");
|
lua:append("\n ",stubs,",\n ")
|
||||||
else
|
else
|
||||||
lua:append(stubs,", ");
|
lua:append(stubs,", ")
|
||||||
end
|
end
|
||||||
lua:append("function(tree");
|
lua:append("function(tree")
|
||||||
local args = {};
|
local replacements = {}
|
||||||
for i,tok in ipairs(\%shorthand.value[1].value) do
|
for i,tok in ipairs(\%shorthand.value[1].value) do
|
||||||
if tok.type == "Var" then args[#args+1] = tostring(tok:as_lua(nomsu)); end
|
if tok.type == "Var" then
|
||||||
|
local lua_var = tostring(tok:as_lua(nomsu))
|
||||||
|
replacements[tok.value] = lua_var
|
||||||
|
lua:append(", ", lua_var)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
for i, arg in ipairs(args) do
|
local function make_tree(t)
|
||||||
lua:append(", ");
|
if Tuple:is_instance(t) then
|
||||||
lua:append(arg);
|
local bits = {}
|
||||||
|
for i, entry in ipairs(t) do
|
||||||
|
bits[i] = make_tree(entry)
|
||||||
|
end
|
||||||
|
return "Tuple("..table.concat(bits, ", ")..")"
|
||||||
|
elseif type(t) ~= 'table' and type(t) ~= 'userdata' then
|
||||||
|
return repr(t)
|
||||||
|
elseif t.type == "Var" and replacements[t.value] then
|
||||||
|
return replacements[t.value]
|
||||||
|
else
|
||||||
|
return t.type.."("..make_tree(t.value)..")"
|
||||||
|
end
|
||||||
end
|
end
|
||||||
local template = repr(tostring(\%longhand:as_nomsu(true)));
|
|
||||||
local replacements = {};
|
|
||||||
for i, a in ipairs(args) do replacements[i] = a.."="..a; end
|
|
||||||
replacements = "{"..table.concat(replacements, ", ").."}";
|
|
||||||
lua:append([[)
|
lua:append([[)
|
||||||
local template = nomsu:parse(Nomsu(]]..repr(tree.source)..[[, ]]..template..[[));
|
local tree = ]], make_tree(\%longhand), [[
|
||||||
local replacement = nomsu:tree_with_replacements(template, ]]..replacements..[[);
|
|
||||||
return replacement:as_lua(nomsu);
|
return tree:as_lua(nomsu)
|
||||||
end);]]);
|
end);]])
|
||||||
return lua;
|
return lua
|
||||||
|
|
||||||
action [remove action %stub]
|
action [remove action %stub]
|
||||||
lua> ".."
|
lua> ".."
|
||||||
local fn = ACTIONS[\%stub];
|
local fn = ACTIONS[\%stub]
|
||||||
local stubs = ARG_ORDERS[fn];
|
local stubs = ARG_ORDERS[fn]
|
||||||
for stub in pairs(stubs) do
|
for stub in pairs(stubs) do
|
||||||
ACTIONS[stub] = nil;
|
ACTIONS[stub] = nil
|
||||||
end
|
end
|
||||||
ARG_ORDERS[fn] = nil;
|
ARG_ORDERS[fn] = nil
|
||||||
|
|
||||||
immediately
|
immediately
|
||||||
action [%tree as lua]
|
action [%tree as lua]
|
||||||
@ -115,11 +123,11 @@ immediately
|
|||||||
|
|
||||||
action [%tree as lua expr]
|
action [%tree as lua expr]
|
||||||
lua> ".."
|
lua> ".."
|
||||||
local lua = \%tree:as_lua(nomsu);
|
local lua = \%tree:as_lua(nomsu)
|
||||||
if not lua.is_value then
|
if not lua.is_value then
|
||||||
error("Invalid thing to convert to lua expr: "..\%tree.source:get_text());
|
error("Invalid thing to convert to lua expr: "..\%tree.source:get_text())
|
||||||
end
|
end
|
||||||
return lua;
|
return lua
|
||||||
|
|
||||||
action [%tree as lua statements]
|
action [%tree as lua statements]
|
||||||
=lua "\%tree:as_lua(nomsu):as_statements()"
|
=lua "\%tree:as_lua(nomsu):as_statements()"
|
||||||
|
@ -66,7 +66,7 @@ immediately
|
|||||||
lua> ".."
|
lua> ".."
|
||||||
local lhs, rhs = Lua(tree.source), Lua(\%assignments.source);
|
local lhs, rhs = Lua(tree.source), Lua(\%assignments.source);
|
||||||
for i, item in ipairs(\%assignments.value) do
|
for i, item in ipairs(\%assignments.value) do
|
||||||
local target, value = item.key, item.value;
|
local target, value = item.value[1], item.value[2];
|
||||||
local target_lua = target:as_lua(nomsu);
|
local target_lua = target:as_lua(nomsu);
|
||||||
if not target_lua.is_value then error("Invalid target for assignment: "..target:get_src()); end
|
if not target_lua.is_value then error("Invalid target for assignment: "..target:get_src()); end
|
||||||
local value_lua = value:as_lua(nomsu);
|
local value_lua = value:as_lua(nomsu);
|
||||||
@ -102,7 +102,7 @@ immediately
|
|||||||
local lhs, rhs = Lua(tree.source), Lua(\%assignments.source);
|
local lhs, rhs = Lua(tree.source), Lua(\%assignments.source);
|
||||||
local vars = {};
|
local vars = {};
|
||||||
for i, item in ipairs(\%assignments.value) do
|
for i, item in ipairs(\%assignments.value) do
|
||||||
local target, value = item.key, item.value;
|
local target, value = item.value[1], item.value[2];
|
||||||
if not target.type == "Var" then
|
if not target.type == "Var" then
|
||||||
error("Invalid target for 'with' assignment: "..tostring(target.source:get_text()));
|
error("Invalid target for 'with' assignment: "..tostring(target.source:get_text()));
|
||||||
end
|
end
|
||||||
|
16
nomsu.lua
16
nomsu.lua
@ -149,9 +149,6 @@ do
|
|||||||
_with_0.Tuple = function(values)
|
_with_0.Tuple = function(values)
|
||||||
return Tuple(unpack(values))
|
return Tuple(unpack(values))
|
||||||
end
|
end
|
||||||
_with_0.DictEntry = function(k, v)
|
|
||||||
return Types.DictEntry(k, v)
|
|
||||||
end
|
|
||||||
_with_0.nl = P("\r") ^ -1 * P("\n")
|
_with_0.nl = P("\r") ^ -1 * P("\n")
|
||||||
_with_0.ws = S(" \t")
|
_with_0.ws = S(" \t")
|
||||||
_with_0.tonumber = tonumber
|
_with_0.tonumber = tonumber
|
||||||
@ -467,24 +464,15 @@ do
|
|||||||
if not (Types.is_node(tree)) then
|
if not (Types.is_node(tree)) then
|
||||||
return
|
return
|
||||||
end
|
end
|
||||||
local _exp_0 = tree.type
|
if Tuple:is_instance(tree.value) then
|
||||||
if "List" == _exp_0 or "Block" == _exp_0 or "Action" == _exp_0 or "Text" == _exp_0 or "IndexChain" == _exp_0 then
|
|
||||||
local _list_0 = tree.value
|
local _list_0 = tree.value
|
||||||
for _index_0 = 1, #_list_0 do
|
for _index_0 = 1, #_list_0 do
|
||||||
local v = _list_0[_index_0]
|
local v = _list_0[_index_0]
|
||||||
self:walk_tree(v, depth + 1)
|
self:walk_tree(v, depth + 1)
|
||||||
end
|
end
|
||||||
elseif "Dict" == _exp_0 then
|
|
||||||
local _list_0 = tree.value
|
|
||||||
for _index_0 = 1, #_list_0 do
|
|
||||||
local e = _list_0[_index_0]
|
|
||||||
self:walk_tree(e.key, depth + 1)
|
|
||||||
self:walk_tree(e.value, depth + 1)
|
|
||||||
end
|
|
||||||
else
|
else
|
||||||
self:walk_tree(tree.value, depth + 1)
|
return self:walk_tree(v, depth + 1)
|
||||||
end
|
end
|
||||||
return nil
|
|
||||||
end,
|
end,
|
||||||
tree_with_replacements = function(self, tree, replacements)
|
tree_with_replacements = function(self, tree, replacements)
|
||||||
if not (next(replacements)) then
|
if not (next(replacements)) then
|
||||||
|
16
nomsu.moon
16
nomsu.moon
@ -131,7 +131,6 @@ NOMSU_DEFS = with {}
|
|||||||
-- Newline supports either windows-style CR+LF or unix-style LF
|
-- Newline supports either windows-style CR+LF or unix-style LF
|
||||||
.Tuple = (values)->
|
.Tuple = (values)->
|
||||||
return Tuple(unpack(values))
|
return Tuple(unpack(values))
|
||||||
.DictEntry = (k,v) -> Types.DictEntry(k,v)
|
|
||||||
.nl = P("\r")^-1 * P("\n")
|
.nl = P("\r")^-1 * P("\n")
|
||||||
.ws = S(" \t")
|
.ws = S(" \t")
|
||||||
.tonumber = tonumber
|
.tonumber = tonumber
|
||||||
@ -429,16 +428,11 @@ class NomsuCompiler
|
|||||||
walk_tree: (tree, depth=0)=>
|
walk_tree: (tree, depth=0)=>
|
||||||
coroutine.yield(tree, depth)
|
coroutine.yield(tree, depth)
|
||||||
return unless Types.is_node(tree)
|
return unless Types.is_node(tree)
|
||||||
switch tree.type
|
if Tuple\is_instance(tree.value)
|
||||||
when "List", "Block", "Action", "Text", "IndexChain"
|
for v in *tree.value
|
||||||
for v in *tree.value
|
@walk_tree(v, depth+1)
|
||||||
@walk_tree(v, depth+1)
|
else
|
||||||
when "Dict"
|
@walk_tree(v, depth+1)
|
||||||
for e in *tree.value
|
|
||||||
@walk_tree(e.key, depth+1)
|
|
||||||
@walk_tree(e.value, depth+1)
|
|
||||||
else @walk_tree(tree.value, depth+1)
|
|
||||||
return nil
|
|
||||||
|
|
||||||
tree_with_replacements: (tree, replacements)=>
|
tree_with_replacements: (tree, replacements)=>
|
||||||
return tree unless next(replacements)
|
return tree unless next(replacements)
|
||||||
|
11
nomsu.peg
11
nomsu.peg
@ -107,7 +107,7 @@ inline_list_item: inline_block / inline_action / inline_expression
|
|||||||
|
|
||||||
inline_dict (Dict):
|
inline_dict (Dict):
|
||||||
!('{..}')
|
!('{..}')
|
||||||
"{" %ws* ({| (inline_dict_item (comma inline_dict_item)*)? |} -> Tuple) %ws*
|
"{" %ws* ({| (inline_dict_entry (comma inline_dict_entry)*)? |} -> Tuple) %ws*
|
||||||
("}"
|
("}"
|
||||||
/ (({} (%ws* comma? (!. / &%nl)->"Failed to find a closing } on the same line")) => error)
|
/ (({} (%ws* comma? (!. / &%nl)->"Failed to find a closing } on the same line")) => error)
|
||||||
/ (({} ([^%nl]*->"Error while parsing dictionary")) => error))
|
/ (({} ([^%nl]*->"Error while parsing dictionary")) => error))
|
||||||
@ -117,10 +117,11 @@ indented_dict (Dict):
|
|||||||
|} -> Tuple)
|
|} -> Tuple)
|
||||||
(dedent / (({} (non_dedent_error -> "Error while parsing dict")) => error))
|
(dedent / (({} (non_dedent_error -> "Error while parsing dict")) => error))
|
||||||
dict_line:
|
dict_line:
|
||||||
((dict_key %ws* ":" %ws* (action / expression)) -> DictEntry !comma)
|
(dict_entry !comma) / (inline_dict_entry (comma dict_line?)?)
|
||||||
/ (inline_dict_item (comma dict_line?)?)
|
dict_entry(DictEntry):
|
||||||
inline_dict_item:
|
{| dict_key %ws* ":" %ws* (action / expression) |} -> Tuple
|
||||||
((dict_key %ws* (":" %ws* (inline_block / inline_action / inline_expression)?)?)-> DictEntry)
|
inline_dict_entry(DictEntry):
|
||||||
|
{| dict_key %ws* (":" %ws* (inline_block / inline_action / inline_expression)?) |} -> Tuple
|
||||||
dict_key:
|
dict_key:
|
||||||
text_word / inline_expression
|
text_word / inline_expression
|
||||||
|
|
||||||
|
239
nomsu_tree.lua
239
nomsu_tree.lua
@ -14,12 +14,6 @@ do
|
|||||||
end
|
end
|
||||||
local MAX_LINE = 80
|
local MAX_LINE = 80
|
||||||
local Types = { }
|
local Types = { }
|
||||||
Types.DictEntry = immutable({
|
|
||||||
"key",
|
|
||||||
"value"
|
|
||||||
}, {
|
|
||||||
name = "DictEntry"
|
|
||||||
})
|
|
||||||
Types.is_node = function(n)
|
Types.is_node = function(n)
|
||||||
return type(n) == 'userdata' and getmetatable(n) and Types[n.type] == getmetatable(n)
|
return type(n) == 'userdata' and getmetatable(n) and Types[n.type] == getmetatable(n)
|
||||||
end
|
end
|
||||||
@ -46,6 +40,28 @@ Tree = function(name, methods)
|
|||||||
local ret = tostring(self.source:get_text()):gsub("\n" .. ((" "):rep(leading_space)), "\n")
|
local ret = tostring(self.source:get_text()):gsub("\n" .. ((" "):rep(leading_space)), "\n")
|
||||||
return ret
|
return ret
|
||||||
end
|
end
|
||||||
|
methods.map = function(self, fn)
|
||||||
|
do
|
||||||
|
local mapped = fn(self)
|
||||||
|
if mapped then
|
||||||
|
return mapped
|
||||||
|
end
|
||||||
|
end
|
||||||
|
if Tuple:is_instance(self.value) then
|
||||||
|
return self:with_value(Tuple(unpack((function()
|
||||||
|
local _accum_0 = { }
|
||||||
|
local _len_0 = 1
|
||||||
|
local _list_0 = self.value
|
||||||
|
for _index_0 = 1, #_list_0 do
|
||||||
|
local v = _list_0[_index_0]
|
||||||
|
_accum_0[_len_0] = v.map and v:map(fn) or v
|
||||||
|
_len_0 = _len_0 + 1
|
||||||
|
end
|
||||||
|
return _accum_0
|
||||||
|
end)())))
|
||||||
|
end
|
||||||
|
return self
|
||||||
|
end
|
||||||
end
|
end
|
||||||
Types[name] = immutable({
|
Types[name] = immutable({
|
||||||
"value",
|
"value",
|
||||||
@ -66,9 +82,6 @@ Tree("Nomsu", {
|
|||||||
return nomsu and Nomsu(self.source, "\\:\n ", nomsu)
|
return nomsu and Nomsu(self.source, "\\:\n ", nomsu)
|
||||||
end
|
end
|
||||||
return nomsu and Nomsu(self.source, "\\(", nomsu, ")")
|
return nomsu and Nomsu(self.source, "\\(", nomsu, ")")
|
||||||
end,
|
|
||||||
map = function(self, fn)
|
|
||||||
return fn(self) or self:with_value(self.value:map(fn))
|
|
||||||
end
|
end
|
||||||
})
|
})
|
||||||
Tree("Block", {
|
Tree("Block", {
|
||||||
@ -113,19 +126,6 @@ Tree("Block", {
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
return nomsu
|
return nomsu
|
||||||
end,
|
|
||||||
map = function(self, fn)
|
|
||||||
return fn(self) or self:with_value(Tuple(unpack((function()
|
|
||||||
local _accum_0 = { }
|
|
||||||
local _len_0 = 1
|
|
||||||
local _list_0 = self.value
|
|
||||||
for _index_0 = 1, #_list_0 do
|
|
||||||
local v = _list_0[_index_0]
|
|
||||||
_accum_0[_len_0] = v:map(fn)
|
|
||||||
_len_0 = _len_0 + 1
|
|
||||||
end
|
|
||||||
return _accum_0
|
|
||||||
end)())))
|
|
||||||
end
|
end
|
||||||
})
|
})
|
||||||
local math_expression = re.compile([[ ([+-] " ")* "%" (" " [*/^+-] (" " [+-])* " %")+ !. ]])
|
local math_expression = re.compile([[ ([+-] " ")* "%" (" " [*/^+-] (" " [+-])* " %")+ !. ]])
|
||||||
@ -349,19 +349,6 @@ Tree("Action", {
|
|||||||
end
|
end
|
||||||
return nomsu
|
return nomsu
|
||||||
end
|
end
|
||||||
end,
|
|
||||||
map = function(self, fn)
|
|
||||||
return fn(self) or self:with_value(Tuple(unpack((function()
|
|
||||||
local _accum_0 = { }
|
|
||||||
local _len_0 = 1
|
|
||||||
local _list_0 = self.value
|
|
||||||
for _index_0 = 1, #_list_0 do
|
|
||||||
local v = _list_0[_index_0]
|
|
||||||
_accum_0[_len_0] = v:map(fn)
|
|
||||||
_len_0 = _len_0 + 1
|
|
||||||
end
|
|
||||||
return _accum_0
|
|
||||||
end)())))
|
|
||||||
end
|
end
|
||||||
})
|
})
|
||||||
Tree("Text", {
|
Tree("Text", {
|
||||||
@ -469,19 +456,6 @@ Tree("Text", {
|
|||||||
end
|
end
|
||||||
return nomsu
|
return nomsu
|
||||||
end
|
end
|
||||||
end,
|
|
||||||
map = function(self, fn)
|
|
||||||
return fn(self) or self:with_value(Tuple(unpack((function()
|
|
||||||
local _accum_0 = { }
|
|
||||||
local _len_0 = 1
|
|
||||||
local _list_0 = self.value
|
|
||||||
for _index_0 = 1, #_list_0 do
|
|
||||||
local v = _list_0[_index_0]
|
|
||||||
_accum_0[_len_0] = type(v) == 'string' and v or v:map(fn)
|
|
||||||
_len_0 = _len_0 + 1
|
|
||||||
end
|
|
||||||
return _accum_0
|
|
||||||
end)())))
|
|
||||||
end
|
end
|
||||||
})
|
})
|
||||||
Tree("List", {
|
Tree("List", {
|
||||||
@ -568,19 +542,6 @@ Tree("List", {
|
|||||||
end
|
end
|
||||||
return nomsu
|
return nomsu
|
||||||
end
|
end
|
||||||
end,
|
|
||||||
map = function(self, fn)
|
|
||||||
return fn(self) or self:with_value(Tuple(unpack((function()
|
|
||||||
local _accum_0 = { }
|
|
||||||
local _len_0 = 1
|
|
||||||
local _list_0 = self.value
|
|
||||||
for _index_0 = 1, #_list_0 do
|
|
||||||
local v = _list_0[_index_0]
|
|
||||||
_accum_0[_len_0] = v:map(fn)
|
|
||||||
_len_0 = _len_0 + 1
|
|
||||||
end
|
|
||||||
return _accum_0
|
|
||||||
end)())))
|
|
||||||
end
|
end
|
||||||
})
|
})
|
||||||
Tree("Dict", {
|
Tree("Dict", {
|
||||||
@ -588,29 +549,14 @@ Tree("Dict", {
|
|||||||
local lua = Lua.Value(self.source, "{")
|
local lua = Lua.Value(self.source, "{")
|
||||||
local line_length = 0
|
local line_length = 0
|
||||||
for i, entry in ipairs(self.value) do
|
for i, entry in ipairs(self.value) do
|
||||||
local key_lua = entry.key:as_lua(nomsu)
|
local entry_lua = entry:as_lua(nomsu)
|
||||||
if not (key_lua.is_value) then
|
lua:append(entry_lua)
|
||||||
local line, src = key.source:get_line(), key.source:get_text()
|
local entry_lua_str = tostring(entry_lua)
|
||||||
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(src)) .. " as a dict key, since it's not an expression.", 0)
|
local last_line = entry_lua_str:match("\n([^\n]*)$")
|
||||||
end
|
if last_line then
|
||||||
local value_lua = entry.value and entry.value:as_lua(nomsu) or Lua.Value(entry.key.source, "true")
|
|
||||||
if not (value_lua.is_value) then
|
|
||||||
local line, src = value.source:get_line(), value.source:get_text()
|
|
||||||
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(src)) .. " as a dict value, since it's not an expression.", 0)
|
|
||||||
end
|
|
||||||
local key_str = tostring(key_lua):match([=[["']([a-zA-Z_][a-zA-Z0-9_]*)['"]]=])
|
|
||||||
if key_str then
|
|
||||||
lua:append(key_str, "=", value_lua)
|
|
||||||
elseif tostring(key_lua):sub(1, 1) == "[" then
|
|
||||||
lua:append("[ ", key_lua, "]=", value_lua)
|
|
||||||
else
|
|
||||||
lua:append("[", key_lua, "]=", value_lua)
|
|
||||||
end
|
|
||||||
local newlines, last_line = ("[" .. tostring(key_lua) .. "=" .. tostring(value_lua)):match("^(.-)([^\n]*)$")
|
|
||||||
if #newlines > 0 then
|
|
||||||
line_length = #last_line
|
line_length = #last_line
|
||||||
else
|
else
|
||||||
line_length = line_length + #last_line
|
line_length = line_length + #entry_lua_str
|
||||||
end
|
end
|
||||||
if i < #self.value then
|
if i < #self.value then
|
||||||
if line_length >= MAX_LINE then
|
if line_length >= MAX_LINE then
|
||||||
@ -632,21 +578,14 @@ Tree("Dict", {
|
|||||||
if inline then
|
if inline then
|
||||||
local nomsu = Nomsu(self.source, "{")
|
local nomsu = Nomsu(self.source, "{")
|
||||||
for i, entry in ipairs(self.value) do
|
for i, entry in ipairs(self.value) do
|
||||||
local key_nomsu = entry.key:as_nomsu(true)
|
local entry_nomsu = entry:as_nomsu(true)
|
||||||
if not (key_nomsu) then
|
if not (entry_nomsu) then
|
||||||
return nil
|
|
||||||
end
|
|
||||||
if entry.key.type == "Action" or entry.key.type == "Block" then
|
|
||||||
key_nomsu:parenthesize()
|
|
||||||
end
|
|
||||||
local value_nomsu = entry.value and entry.value:as_nomsu(true) or Nomsu(entry.key.source, "")
|
|
||||||
if not (value_nomsu) then
|
|
||||||
return nil
|
return nil
|
||||||
end
|
end
|
||||||
if i > 1 then
|
if i > 1 then
|
||||||
nomsu:append(", ")
|
nomsu:append(", ")
|
||||||
end
|
end
|
||||||
nomsu:append(key_nomsu, ":", value_nomsu)
|
nomsu:append(entry_nomsu)
|
||||||
end
|
end
|
||||||
nomsu:append("}")
|
nomsu:append("}")
|
||||||
return nomsu
|
return nomsu
|
||||||
@ -660,37 +599,21 @@ Tree("Dict", {
|
|||||||
local _list_0 = self.value
|
local _list_0 = self.value
|
||||||
for _index_0 = 1, #_list_0 do
|
for _index_0 = 1, #_list_0 do
|
||||||
local entry = _list_0[_index_0]
|
local entry = _list_0[_index_0]
|
||||||
local key_nomsu = entry.key:as_nomsu(true)
|
local entry_nomsu = entry:as_nomsu()
|
||||||
if not (key_nomsu) then
|
if not (entry_nomsu) then
|
||||||
return nil
|
return nil
|
||||||
end
|
end
|
||||||
if entry.key.type == "Action" or entry.key.type == "Block" then
|
if #line + #tostring(entry_nomsu) <= MAX_LINE then
|
||||||
key_nomsu:parenthesize()
|
|
||||||
end
|
|
||||||
local value_nomsu = entry.value and entry.value:as_nomsu(true) or Nomsu(entry.key.source, "")
|
|
||||||
if value_nomsu and #line + #", " + #key_nomsu + #":" + #value_nomsu <= MAX_LINE then
|
|
||||||
if #line.bits > 1 then
|
if #line.bits > 1 then
|
||||||
line:append(", ")
|
line:append(", ")
|
||||||
end
|
end
|
||||||
line:append(key_nomsu)
|
line:append(entry_nomsu)
|
||||||
if entry.value then
|
|
||||||
line:append(":", value_nomsu)
|
|
||||||
end
|
|
||||||
else
|
else
|
||||||
if not (value_nomsu) then
|
|
||||||
value_nomsu = entry.value:as_nomsu()
|
|
||||||
if not (value_nomsu) then
|
|
||||||
return nil
|
|
||||||
end
|
|
||||||
end
|
|
||||||
if #line.bits > 1 then
|
if #line.bits > 1 then
|
||||||
nomsu:append(line)
|
nomsu:append(line)
|
||||||
line = Nomsu(bit.source, "\n ")
|
line = Nomsu(bit.source, "\n ")
|
||||||
end
|
end
|
||||||
line:append(key_nomsu)
|
line:append(entry_nomsu)
|
||||||
if entry.value then
|
|
||||||
line:append(":", value_nomsu)
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
if #line.bits > 1 then
|
if #line.bits > 1 then
|
||||||
@ -698,20 +621,61 @@ Tree("Dict", {
|
|||||||
end
|
end
|
||||||
return nomsu
|
return nomsu
|
||||||
end
|
end
|
||||||
|
end
|
||||||
|
})
|
||||||
|
Tree("DictEntry", {
|
||||||
|
as_lua = function(self, nomsu)
|
||||||
|
local key, value = self.value[1], self.value[2]
|
||||||
|
local key_lua = key:as_lua(nomsu)
|
||||||
|
if not (key_lua.is_value) then
|
||||||
|
local line, src = key.source:get_line(), key.source:get_text()
|
||||||
|
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(src)) .. " as a dict key, since it's not an expression.", 0)
|
||||||
|
end
|
||||||
|
local value_lua = value and value:as_lua(nomsu) or Lua.Value(key.source, "true")
|
||||||
|
if not (value_lua.is_value) then
|
||||||
|
local line, src = value.source:get_line(), value.source:get_text()
|
||||||
|
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(src)) .. " as a dict value, since it's not an expression.", 0)
|
||||||
|
end
|
||||||
|
local key_str = tostring(key_lua):match([=[["']([a-zA-Z_][a-zA-Z0-9_]*)['"]]=])
|
||||||
|
if key_str then
|
||||||
|
return Lua(key.source, key_str, "=", value_lua)
|
||||||
|
elseif tostring(key_lua):sub(1, 1) == "[" then
|
||||||
|
return Lua(key.source, "[ ", key_lua, "]=", value_lua)
|
||||||
|
else
|
||||||
|
return Lua(key.source, "[", key_lua, "]=", value_lua)
|
||||||
|
end
|
||||||
end,
|
end,
|
||||||
map = function(self, fn)
|
as_nomsu = function(self, inline)
|
||||||
local DictEntry = Types.DictEntry
|
if inline == nil then
|
||||||
return fn(self) or self:with_value(Tuple(unpack((function()
|
inline = true
|
||||||
local _accum_0 = { }
|
end
|
||||||
local _len_0 = 1
|
local key, value = self.value[1], self.value[2]
|
||||||
local _list_0 = self.value
|
local key_nomsu = key:as_nomsu(true)
|
||||||
for _index_0 = 1, #_list_0 do
|
if not (key_nomsu) then
|
||||||
local e = _list_0[_index_0]
|
return nil
|
||||||
_accum_0[_len_0] = DictEntry(e.key:map(fn), e.value:map(fn))
|
end
|
||||||
_len_0 = _len_0 + 1
|
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
|
||||||
|
value_nomsu = Nomsu(key.source, "")
|
||||||
|
end
|
||||||
|
if inline and not value_nomsu then
|
||||||
|
return nil
|
||||||
|
end
|
||||||
|
if not value_nomsu then
|
||||||
|
if inline then
|
||||||
|
return nil
|
||||||
end
|
end
|
||||||
return _accum_0
|
value_nomsu = value:as_nomsu()
|
||||||
end)())))
|
if not (value_nomsu) then
|
||||||
|
return nil
|
||||||
|
end
|
||||||
|
end
|
||||||
|
return Nomsu(key.source, key_nomsu, ":", value_nomsu)
|
||||||
end
|
end
|
||||||
})
|
})
|
||||||
Tree("IndexChain", {
|
Tree("IndexChain", {
|
||||||
@ -771,19 +735,6 @@ Tree("IndexChain", {
|
|||||||
nomsu:append(bit_nomsu)
|
nomsu:append(bit_nomsu)
|
||||||
end
|
end
|
||||||
return nomsu
|
return nomsu
|
||||||
end,
|
|
||||||
map = function(self, fn)
|
|
||||||
return fn(self) or self:with_value(Tuple(unpack((function()
|
|
||||||
local _accum_0 = { }
|
|
||||||
local _len_0 = 1
|
|
||||||
local _list_0 = self.value
|
|
||||||
for _index_0 = 1, #_list_0 do
|
|
||||||
local v = _list_0[_index_0]
|
|
||||||
_accum_0[_len_0] = v:map(fn)
|
|
||||||
_len_0 = _len_0 + 1
|
|
||||||
end
|
|
||||||
return _accum_0
|
|
||||||
end)())))
|
|
||||||
end
|
end
|
||||||
})
|
})
|
||||||
Tree("Number", {
|
Tree("Number", {
|
||||||
@ -795,9 +746,6 @@ Tree("Number", {
|
|||||||
inline = false
|
inline = false
|
||||||
end
|
end
|
||||||
return Nomsu(self.source, tostring(self.value))
|
return Nomsu(self.source, tostring(self.value))
|
||||||
end,
|
|
||||||
map = function(self, fn)
|
|
||||||
return fn(self) or self
|
|
||||||
end
|
end
|
||||||
})
|
})
|
||||||
Tree("Var", {
|
Tree("Var", {
|
||||||
@ -818,9 +766,6 @@ Tree("Var", {
|
|||||||
inline = false
|
inline = false
|
||||||
end
|
end
|
||||||
return Nomsu(self.source, "%", self.value)
|
return Nomsu(self.source, "%", self.value)
|
||||||
end,
|
|
||||||
map = function(self, fn)
|
|
||||||
return fn(self) or self
|
|
||||||
end
|
end
|
||||||
})
|
})
|
||||||
Tree("Word", {
|
Tree("Word", {
|
||||||
@ -832,9 +777,6 @@ Tree("Word", {
|
|||||||
inline = false
|
inline = false
|
||||||
end
|
end
|
||||||
return Nomsu(self.source, self.value)
|
return Nomsu(self.source, self.value)
|
||||||
end,
|
|
||||||
map = function(self, fn)
|
|
||||||
return fn(self) or self
|
|
||||||
end
|
end
|
||||||
})
|
})
|
||||||
Tree("Comment", {
|
Tree("Comment", {
|
||||||
@ -853,9 +795,6 @@ Tree("Comment", {
|
|||||||
else
|
else
|
||||||
return Nomsu(self.source, "#", self.value)
|
return Nomsu(self.source, "#", self.value)
|
||||||
end
|
end
|
||||||
end,
|
|
||||||
map = function(self, fn)
|
|
||||||
return fn(self) or self
|
|
||||||
end
|
end
|
||||||
})
|
})
|
||||||
return Types
|
return Types
|
||||||
|
130
nomsu_tree.moon
130
nomsu_tree.moon
@ -9,7 +9,6 @@ immutable = require 'immutable'
|
|||||||
MAX_LINE = 80 -- For beautification purposes, try not to make lines much longer than this value
|
MAX_LINE = 80 -- For beautification purposes, try not to make lines much longer than this value
|
||||||
|
|
||||||
Types = {}
|
Types = {}
|
||||||
Types.DictEntry = immutable({"key","value"}, {name:"DictEntry"})
|
|
||||||
Types.is_node = (n)->
|
Types.is_node = (n)->
|
||||||
type(n) == 'userdata' and getmetatable(n) and Types[n.type] == getmetatable(n)
|
type(n) == 'userdata' and getmetatable(n) and Types[n.type] == getmetatable(n)
|
||||||
|
|
||||||
@ -29,6 +28,12 @@ Tree = (name, methods)->
|
|||||||
leading_space = 0
|
leading_space = 0
|
||||||
ret = tostring(@source\get_text!)\gsub("\n"..((" ")\rep(leading_space)), "\n")
|
ret = tostring(@source\get_text!)\gsub("\n"..((" ")\rep(leading_space)), "\n")
|
||||||
return ret
|
return ret
|
||||||
|
.map = (fn)=>
|
||||||
|
if mapped = fn(self)
|
||||||
|
return mapped
|
||||||
|
if Tuple\is_instance(@value)
|
||||||
|
return @with_value(Tuple(unpack([v.map and v\map(fn) or v for v in *@value])))
|
||||||
|
return self
|
||||||
|
|
||||||
Types[name] = immutable {"value","source"}, methods
|
Types[name] = immutable {"value","source"}, methods
|
||||||
|
|
||||||
@ -44,9 +49,6 @@ Tree "Nomsu",
|
|||||||
return nomsu and Nomsu(@source, "\\:\n ", nomsu)
|
return nomsu and Nomsu(@source, "\\:\n ", nomsu)
|
||||||
return nomsu and Nomsu(@source, "\\(", nomsu, ")")
|
return nomsu and Nomsu(@source, "\\(", nomsu, ")")
|
||||||
|
|
||||||
map: (fn)=>
|
|
||||||
fn(self) or @with_value(@value\map(fn))
|
|
||||||
|
|
||||||
Tree "Block",
|
Tree "Block",
|
||||||
as_lua: (nomsu)=>
|
as_lua: (nomsu)=>
|
||||||
lua = Lua(@source)
|
lua = Lua(@source)
|
||||||
@ -77,9 +79,6 @@ Tree "Block",
|
|||||||
nomsu\append "\n"
|
nomsu\append "\n"
|
||||||
return nomsu
|
return nomsu
|
||||||
|
|
||||||
map: (fn)=>
|
|
||||||
fn(self) or @with_value(Tuple(unpack([v\map(fn) for v in *@value])))
|
|
||||||
|
|
||||||
math_expression = re.compile [[ ([+-] " ")* "%" (" " [*/^+-] (" " [+-])* " %")+ !. ]]
|
math_expression = re.compile [[ ([+-] " ")* "%" (" " [*/^+-] (" " [+-])* " %")+ !. ]]
|
||||||
Tree "Action",
|
Tree "Action",
|
||||||
as_lua: (nomsu)=>
|
as_lua: (nomsu)=>
|
||||||
@ -203,9 +202,6 @@ Tree "Action",
|
|||||||
next_space = "\n.."
|
next_space = "\n.."
|
||||||
return nomsu
|
return nomsu
|
||||||
|
|
||||||
map: (fn)=>
|
|
||||||
fn(self) or @with_value(Tuple(unpack([v\map(fn) for v in *@value])))
|
|
||||||
|
|
||||||
Tree "Text",
|
Tree "Text",
|
||||||
as_lua: (nomsu)=>
|
as_lua: (nomsu)=>
|
||||||
lua = Lua.Value(@source)
|
lua = Lua.Value(@source)
|
||||||
@ -273,9 +269,6 @@ Tree "Text",
|
|||||||
nomsu\append "\n .."
|
nomsu\append "\n .."
|
||||||
return nomsu
|
return nomsu
|
||||||
|
|
||||||
map: (fn)=>
|
|
||||||
fn(self) or @with_value(Tuple(unpack([type(v) == 'string' and v or v\map(fn) for v in *@value])))
|
|
||||||
|
|
||||||
Tree "List",
|
Tree "List",
|
||||||
as_lua: (nomsu)=>
|
as_lua: (nomsu)=>
|
||||||
lua = Lua.Value @source, "{"
|
lua = Lua.Value @source, "{"
|
||||||
@ -337,39 +330,20 @@ Tree "List",
|
|||||||
nomsu\append line
|
nomsu\append line
|
||||||
return nomsu
|
return nomsu
|
||||||
|
|
||||||
map: (fn)=>
|
|
||||||
fn(self) or @with_value(Tuple(unpack([v\map(fn) for v in *@value])))
|
|
||||||
|
|
||||||
Tree "Dict",
|
Tree "Dict",
|
||||||
as_lua: (nomsu)=>
|
as_lua: (nomsu)=>
|
||||||
lua = Lua.Value @source, "{"
|
lua = Lua.Value @source, "{"
|
||||||
line_length = 0
|
line_length = 0
|
||||||
for i, entry in ipairs @value
|
for i, entry in ipairs @value
|
||||||
key_lua = entry.key\as_lua(nomsu)
|
entry_lua = entry\as_lua(nomsu)
|
||||||
unless key_lua.is_value
|
lua\append entry_lua
|
||||||
line, src = key.source\get_line!, key.source\get_text!
|
entry_lua_str = tostring(entry_lua)
|
||||||
error "#{line}: Cannot use #{colored.yellow src} as a dict key, since it's not an expression.", 0
|
|
||||||
value_lua = entry.value and entry.value\as_lua(nomsu) or Lua.Value(entry.key.source, "true")
|
|
||||||
unless value_lua.is_value
|
|
||||||
line, src = value.source\get_line!, value.source\get_text!
|
|
||||||
error "#{line}: Cannot use #{colored.yellow src} as a dict value, since it's not an expression.", 0
|
|
||||||
key_str = tostring(key_lua)\match([=[["']([a-zA-Z_][a-zA-Z0-9_]*)['"]]=])
|
|
||||||
if key_str
|
|
||||||
lua\append key_str,"=",value_lua
|
|
||||||
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"]
|
|
||||||
lua\append "[ ",key_lua,"]=",value_lua
|
|
||||||
else
|
|
||||||
lua\append "[",key_lua,"]=",value_lua
|
|
||||||
|
|
||||||
-- TODO: maybe make this more accurate? It's only a heuristic, so eh...
|
-- TODO: maybe make this more accurate? It's only a heuristic, so eh...
|
||||||
newlines, last_line = ("[#{key_lua}=#{value_lua}")\match("^(.-)([^\n]*)$")
|
last_line = entry_lua_str\match("\n([^\n]*)$")
|
||||||
if #newlines > 0
|
if last_line
|
||||||
line_length = #last_line
|
line_length = #last_line
|
||||||
else
|
else
|
||||||
line_length += #last_line
|
line_length += #entry_lua_str
|
||||||
if i < #@value
|
if i < #@value
|
||||||
if line_length >= MAX_LINE
|
if line_length >= MAX_LINE
|
||||||
lua\append ",\n "
|
lua\append ",\n "
|
||||||
@ -384,15 +358,11 @@ Tree "Dict",
|
|||||||
if inline
|
if inline
|
||||||
nomsu = Nomsu(@source, "{")
|
nomsu = Nomsu(@source, "{")
|
||||||
for i, entry in ipairs @value
|
for i, entry in ipairs @value
|
||||||
key_nomsu = entry.key\as_nomsu(true)
|
entry_nomsu = entry\as_nomsu(true)
|
||||||
return nil unless key_nomsu
|
return nil unless entry_nomsu
|
||||||
if entry.key.type == "Action" or entry.key.type == "Block"
|
|
||||||
key_nomsu\parenthesize!
|
|
||||||
value_nomsu = entry.value and entry.value\as_nomsu(true) or Nomsu(entry.key.source, "")
|
|
||||||
return nil unless value_nomsu
|
|
||||||
if i > 1
|
if i > 1
|
||||||
nomsu\append ", "
|
nomsu\append ", "
|
||||||
nomsu\append key_nomsu,":",value_nomsu
|
nomsu\append entry_nomsu
|
||||||
nomsu\append "}"
|
nomsu\append "}"
|
||||||
return nomsu
|
return nomsu
|
||||||
else
|
else
|
||||||
@ -401,32 +371,59 @@ Tree "Dict",
|
|||||||
nomsu = Nomsu(@source, "{..}")
|
nomsu = Nomsu(@source, "{..}")
|
||||||
line = Nomsu(@source, "\n ")
|
line = Nomsu(@source, "\n ")
|
||||||
for entry in *@value
|
for entry in *@value
|
||||||
key_nomsu = entry.key\as_nomsu(true)
|
entry_nomsu = entry\as_nomsu!
|
||||||
return nil unless key_nomsu
|
return nil unless entry_nomsu
|
||||||
if entry.key.type == "Action" or entry.key.type == "Block"
|
if #line + #tostring(entry_nomsu) <= MAX_LINE
|
||||||
key_nomsu\parenthesize!
|
|
||||||
value_nomsu = entry.value and entry.value\as_nomsu(true) or Nomsu(entry.key.source, "")
|
|
||||||
if value_nomsu and #line + #", " + #key_nomsu + #":" + #value_nomsu <= MAX_LINE
|
|
||||||
if #line.bits > 1
|
if #line.bits > 1
|
||||||
line\append ", "
|
line\append ", "
|
||||||
line\append key_nomsu
|
line\append entry_nomsu
|
||||||
if entry.value then line\append ":",value_nomsu
|
|
||||||
else
|
else
|
||||||
unless value_nomsu
|
|
||||||
value_nomsu = entry.value\as_nomsu!
|
|
||||||
return nil unless value_nomsu
|
|
||||||
if #line.bits > 1
|
if #line.bits > 1
|
||||||
nomsu\append line
|
nomsu\append line
|
||||||
line = Nomsu(bit.source, "\n ")
|
line = Nomsu(bit.source, "\n ")
|
||||||
line\append key_nomsu
|
line\append entry_nomsu
|
||||||
if entry.value then line\append ":",value_nomsu
|
|
||||||
if #line.bits > 1
|
if #line.bits > 1
|
||||||
nomsu\append line
|
nomsu\append line
|
||||||
return nomsu
|
return nomsu
|
||||||
|
|
||||||
map: (fn)=>
|
Tree "DictEntry",
|
||||||
DictEntry = Types.DictEntry
|
as_lua: (nomsu)=>
|
||||||
fn(self) or @with_value(Tuple(unpack([DictEntry(e.key\map(fn), e.value\map(fn)) for e in *@value])))
|
key, value = @value[1], @value[2]
|
||||||
|
key_lua = key\as_lua(nomsu)
|
||||||
|
unless key_lua.is_value
|
||||||
|
line, src = key.source\get_line!, key.source\get_text!
|
||||||
|
error "#{line}: Cannot use #{colored.yellow src} as a dict key, since it's not an expression.", 0
|
||||||
|
value_lua = value and value\as_lua(nomsu) or Lua.Value(key.source, "true")
|
||||||
|
unless value_lua.is_value
|
||||||
|
line, src = value.source\get_line!, value.source\get_text!
|
||||||
|
error "#{line}: Cannot use #{colored.yellow src} as a dict value, since it's not an expression.", 0
|
||||||
|
key_str = tostring(key_lua)\match([=[["']([a-zA-Z_][a-zA-Z0-9_]*)['"]]=])
|
||||||
|
return if key_str
|
||||||
|
Lua key.source, key_str,"=",value_lua
|
||||||
|
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"]
|
||||||
|
Lua key.source, "[ ",key_lua,"]=",value_lua
|
||||||
|
else
|
||||||
|
Lua key.source, "[",key_lua,"]=",value_lua
|
||||||
|
|
||||||
|
as_nomsu: (inline=true)=>
|
||||||
|
key, value = @value[1], @value[2]
|
||||||
|
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)
|
||||||
|
else Nomsu(key.source, "")
|
||||||
|
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
|
||||||
|
return Nomsu key.source, key_nomsu, ":", value_nomsu
|
||||||
|
|
||||||
|
|
||||||
Tree "IndexChain",
|
Tree "IndexChain",
|
||||||
as_lua: (nomsu)=>
|
as_lua: (nomsu)=>
|
||||||
@ -468,9 +465,6 @@ Tree "IndexChain",
|
|||||||
nomsu\append bit_nomsu
|
nomsu\append bit_nomsu
|
||||||
return nomsu
|
return nomsu
|
||||||
|
|
||||||
map: (fn)=>
|
|
||||||
fn(self) or @with_value(Tuple(unpack([v\map(fn) for v in *@value])))
|
|
||||||
|
|
||||||
Tree "Number",
|
Tree "Number",
|
||||||
as_lua: (nomsu)=>
|
as_lua: (nomsu)=>
|
||||||
Lua.Value(@source, tostring(@value))
|
Lua.Value(@source, tostring(@value))
|
||||||
@ -478,8 +472,6 @@ Tree "Number",
|
|||||||
as_nomsu: (inline=false)=>
|
as_nomsu: (inline=false)=>
|
||||||
return Nomsu(@source, tostring(@value))
|
return Nomsu(@source, tostring(@value))
|
||||||
|
|
||||||
map: (fn)=> fn(self) or self
|
|
||||||
|
|
||||||
Tree "Var",
|
Tree "Var",
|
||||||
as_lua_id: (v)->
|
as_lua_id: (v)->
|
||||||
"_"..(v\gsub("%W", (c)-> if c == "_" then "__" else ("_%x")\format(c\byte!)))
|
"_"..(v\gsub("%W", (c)-> if c == "_" then "__" else ("_%x")\format(c\byte!)))
|
||||||
@ -490,8 +482,6 @@ Tree "Var",
|
|||||||
as_nomsu: (inline=false)=>
|
as_nomsu: (inline=false)=>
|
||||||
return Nomsu(@source, "%", @value)
|
return Nomsu(@source, "%", @value)
|
||||||
|
|
||||||
map: (fn)=> fn(self) or self
|
|
||||||
|
|
||||||
Tree "Word",
|
Tree "Word",
|
||||||
as_lua: (nomsu)=>
|
as_lua: (nomsu)=>
|
||||||
error("Attempt to convert Word to lua")
|
error("Attempt to convert Word to lua")
|
||||||
@ -499,8 +489,6 @@ Tree "Word",
|
|||||||
as_nomsu: (inline=false)=>
|
as_nomsu: (inline=false)=>
|
||||||
return Nomsu(@source, @value)
|
return Nomsu(@source, @value)
|
||||||
|
|
||||||
map: (fn)=> fn(self) or self
|
|
||||||
|
|
||||||
Tree "Comment",
|
Tree "Comment",
|
||||||
as_lua: (nomsu)=>
|
as_lua: (nomsu)=>
|
||||||
Lua(@source, "--"..@value\gsub("\n","\n--").."\n")
|
Lua(@source, "--"..@value\gsub("\n","\n--").."\n")
|
||||||
@ -512,6 +500,4 @@ Tree "Comment",
|
|||||||
else
|
else
|
||||||
return Nomsu(@source, "#", @value)
|
return Nomsu(@source, "#", @value)
|
||||||
|
|
||||||
map: (fn)=> fn(self) or self
|
|
||||||
|
|
||||||
return Types
|
return Types
|
||||||
|
Loading…
Reference in New Issue
Block a user