Improving nomsu codegen.

This commit is contained in:
Bruce Hill 2018-07-17 23:33:49 -07:00
parent ceb29321ed
commit af441330ff
4 changed files with 51 additions and 13 deletions

View File

@ -854,6 +854,9 @@ do
inline = true inline = true
})) }))
end end
if tostring(arg_nomsu) ~= '"\\n"' and tostring(arg_nomsu):match("\\n") then
arg_nomsu = nil
end
if bit.type == "Block" then if bit.type == "Block" then
next_space = match(next_space, "[^ ]*") next_space = match(next_space, "[^ ]*")
end end
@ -931,10 +934,13 @@ do
local make_text local make_text
make_text = function(tree) make_text = function(tree)
local nomsu = NomsuCode(tree.source) local nomsu = NomsuCode(tree.source)
for _index_0 = 1, #tree do for i, bit in ipairs(tree) do
local bit = tree[_index_0]
if type(bit) == 'string' then if type(bit) == 'string' then
nomsu:append((gsub(gsub(gsub(bit, "\\", "\\\\"), "\n", "\\n"), '"', '\\"'))) bit = (gsub(gsub(gsub(bit, "\\", "\\\\"), "\n", "\\n"), '"', '\\"'))
bit = gsub(bit, "%G", (function(c)
return c == ' ' and c or "\\" .. tostring(c:byte())
end))
nomsu:append(bit)
elseif bit.type == "Text" then elseif bit.type == "Text" then
nomsu:append(make_text(bit)) nomsu:append(make_text(bit))
else else
@ -943,6 +949,8 @@ do
})) }))
if bit.type ~= "Var" and bit.type ~= "List" and bit.type ~= "Dict" then if bit.type ~= "Var" and bit.type ~= "List" and bit.type ~= "Dict" then
interp_nomsu:parenthesize() interp_nomsu:parenthesize()
elseif bit.type == "Var" and type(tree[i + 1]) == 'string' and not match(tree[i + 1], "^[ \n\t,.:;#(){}[%]]") then
interp_nomsu:parenthesize()
end end
nomsu:append("\\", interp_nomsu) nomsu:append("\\", interp_nomsu)
end end
@ -1002,6 +1010,8 @@ do
if interp_nomsu then if interp_nomsu then
if bit.type ~= "Var" and bit.type ~= "List" and bit.type ~= "Dict" then if bit.type ~= "Var" and bit.type ~= "List" and bit.type ~= "Dict" then
interp_nomsu:parenthesize() interp_nomsu:parenthesize()
elseif bit.type == "Var" and type(tree[i + 1]) == 'string' and not match(tree[i + 1], "^[ \n\t,.:;#(){}[%]]") then
interp_nomsu:parenthesize()
end end
nomsu:append("\\", interp_nomsu) nomsu:append("\\", interp_nomsu)
else else
@ -1009,7 +1019,11 @@ do
if not (interp_nomsu) then if not (interp_nomsu) then
return nil return nil
end end
nomsu:append("\\\n ", interp_nomsu) if bit.type ~= "List" and bit.type ~= "Dict" and bit.type ~= "Text" and bit.type ~= "Block" then
nomsu:append("\\\n(..) ", interp_nomsu)
else
nomsu:append("\\", interp_nomsu)
end
if i < #tree then if i < #tree then
nomsu:append("\n..") nomsu:append("\n..")
end end
@ -1136,9 +1150,14 @@ do
end end
elseif "DictEntry" == _exp_0 then elseif "DictEntry" == _exp_0 then
local key, value = tree[1], tree[2] local key, value = tree[1], tree[2]
local key_nomsu = assert(recurse(key, { local key_nomsu
if key.type == "Text" and #key == 1 and Parser.is_identifier(key[1]) then
key_nomsu = NomsuCode(key.source, key[1])
else
key_nomsu = assert(recurse(key, {
inline = true inline = true
})) }))
end
if key.type == "Action" or key.type == "Block" then if key.type == "Action" or key.type == "Block" then
key_nomsu:parenthesize() key_nomsu:parenthesize()
end end

View File

@ -552,6 +552,8 @@ with NomsuCompiler
else else
arg_nomsu = if bit.type == "Block" and #bit > 1 then nil arg_nomsu = if bit.type == "Block" and #bit > 1 then nil
else assert recurse(bit,inline:true) else assert recurse(bit,inline:true)
if tostring(arg_nomsu) != '"\\n"' and tostring(arg_nomsu)\match("\\n")
arg_nomsu = nil -- Force long text for multi-line text
next_space = match(next_space, "[^ ]*") if bit.type == "Block" next_space = match(next_space, "[^ ]*") if bit.type == "Block"
nomsu\append next_space nomsu\append next_space
if arg_nomsu and nomsu\trailing_line_len! + #tostring(arg_nomsu) < MAX_LINE if arg_nomsu and nomsu\trailing_line_len! + #tostring(arg_nomsu) < MAX_LINE
@ -611,16 +613,20 @@ with NomsuCompiler
if options.inline if options.inline
make_text = (tree)-> make_text = (tree)->
nomsu = NomsuCode(tree.source) nomsu = NomsuCode(tree.source)
for bit in *tree for i, bit in ipairs tree
if type(bit) == 'string' if type(bit) == 'string'
-- TODO: unescape better? -- TODO: unescape better?
nomsu\append (gsub(gsub(gsub(bit,"\\","\\\\"),"\n","\\n"),'"','\\"')) bit = (gsub(gsub(gsub(bit,"\\","\\\\"),"\n","\\n"),'"','\\"'))
bit = gsub(bit, "%G", ((c)-> c == ' ' and c or "\\#{c\byte!}"))
nomsu\append bit
elseif bit.type == "Text" elseif bit.type == "Text"
nomsu\append(make_text(bit)) nomsu\append(make_text(bit))
else else
interp_nomsu = assert recurse(bit, inline:true) interp_nomsu = assert recurse(bit, inline:true)
if bit.type != "Var" and bit.type != "List" and bit.type != "Dict" if bit.type != "Var" and bit.type != "List" and bit.type != "Dict"
interp_nomsu\parenthesize! interp_nomsu\parenthesize!
elseif bit.type == "Var" and type(tree[i+1]) == 'string' and not match(tree[i+1], "^[ \n\t,.:;#(){}[%]]")
interp_nomsu\parenthesize!
nomsu\append "\\", interp_nomsu nomsu\append "\\", interp_nomsu
return nomsu return nomsu
return NomsuCode(tree.source, '"', make_text(tree), '"') return NomsuCode(tree.source, '"', make_text(tree), '"')
@ -660,11 +666,16 @@ with NomsuCompiler
if interp_nomsu if interp_nomsu
if bit.type != "Var" and bit.type != "List" and bit.type != "Dict" if bit.type != "Var" and bit.type != "List" and bit.type != "Dict"
interp_nomsu\parenthesize! interp_nomsu\parenthesize!
elseif bit.type == "Var" and type(tree[i+1]) == 'string' and not match(tree[i+1], "^[ \n\t,.:;#(){}[%]]")
interp_nomsu\parenthesize!
nomsu\append "\\", interp_nomsu nomsu\append "\\", interp_nomsu
else else
interp_nomsu = assert(recurse(bit)) interp_nomsu = assert(recurse(bit))
return nil unless interp_nomsu return nil unless interp_nomsu
nomsu\append "\\\n ", interp_nomsu if bit.type != "List" and bit.type != "Dict" and bit.type != "Text" and bit.type != "Block"
nomsu\append "\\\n(..) ", interp_nomsu
else
nomsu\append "\\", interp_nomsu
if i < #tree if i < #tree
nomsu\append "\n.." nomsu\append "\n.."
return nomsu return nomsu
@ -742,7 +753,9 @@ with NomsuCompiler
when "DictEntry" when "DictEntry"
key, value = tree[1], tree[2] key, value = tree[1], tree[2]
key_nomsu = assert recurse(key, inline:true) key_nomsu = if key.type == "Text" and #key == 1 and Parser.is_identifier(key[1])
NomsuCode(key.source, key[1])
else assert recurse(key, inline:true)
key_nomsu\parenthesize! if key.type == "Action" or key.type == "Block" key_nomsu\parenthesize! if key.type == "Action" or key.type == "Block"
value_nomsu = if value value_nomsu = if value
assert recurse(value, inline:true) assert recurse(value, inline:true)

View File

@ -229,6 +229,9 @@ Parser.parse = function(nomsu_code, source, version)
return tree return tree
end end
Parser.is_operator = function(s) Parser.is_operator = function(s)
return not not (NOMSU_DEFS.operator_char ^ 1):match(s) return not not (NOMSU_DEFS.operator_char ^ 1 * -1):match(s)
end
Parser.is_identifier = function(s)
return not not (NOMSU_DEFS.ident_char ^ 1 * -1):match(s)
end end
return Parser return Parser

View File

@ -139,6 +139,9 @@ Parser.parse = (nomsu_code, source=nil, version=nil)->
return tree return tree
Parser.is_operator = (s)-> Parser.is_operator = (s)->
return not not (NOMSU_DEFS.operator_char^1)\match(s) return not not (NOMSU_DEFS.operator_char^1 * -1)\match(s)
Parser.is_identifier = (s)->
return not not (NOMSU_DEFS.ident_char^1 * -1)\match(s)
return Parser return Parser