2018-11-08 15:23:22 -08:00
{:NomsuCode} = require "code_obj"
{:find, :sub, :match} = string
{:R,:P,:S} = require 'lpeg'
re = require 're'
2019-03-04 14:25:56 -08:00
pretty_error = require("pretty_errors")
2018-11-08 15:23:22 -08:00
2018-11-11 15:05:18 -08:00
MAX_LINE = 80
2018-11-11 15:25:25 -08:00
GOLDEN_RATIO = ((math.sqrt(5)-1)/2)
2018-11-08 15:23:22 -08:00
-- Parsing helper functions
utf8_char_patt = (
R("\194\223")*R("\128\191") +
R("\224\239")*R("\128\191")*R("\128\191") +
R("\240\244")*R("\128\191")*R("\128\191")*R("\128\191"))
2019-02-05 15:45:27 -08:00
operator_char = S("#'`~@^&*+=<>?/%!|\\-") + (P("\xE2") * (R("\x88\x8B")+R("\xA8\xAB")) * R("\128\191"))
operator_patt = operator_char^1 * -1
identifier_patt = (R("az","AZ","09") + P("_") + (-operator_char*utf8_char_patt))^1 * -1
2018-11-08 15:23:22 -08:00
is_operator = (s)->
2019-02-06 14:31:14 -08:00
return type(s) == 'string' and not not operator_patt\match(s)
2018-11-08 15:23:22 -08:00
is_identifier = (s)->
2019-02-06 14:31:14 -08:00
return type(s) == 'string' and not not identifier_patt\match(s)
2019-02-05 15:45:27 -08:00
can_be_unary = (t)->
2019-03-20 16:40:32 -07:00
t.type == "Action" and #t == 2 and is_operator(t[1]) and type(t[2]) != 'string' and t[2].type != "Block" and not (t[2].type == "Number" and t[1] == "-")
2018-11-08 15:23:22 -08:00
inline_escaper = re.compile("{~ (%utf8_char / ('\"' -> '\\\"') / ('\n' -> '\\n') / ('\t' -> '\\t') / ('\b' -> '\\b') / ('\a' -> '\\a') / ('\v' -> '\\v') / ('\f' -> '\\f') / ('\r' -> '\\r') / ('\\' -> '\\\\') / ([^ -~] -> escape) / .)* ~}", {utf8_char: utf8_char_patt, escape:(=> ("\\%03d")\format(@byte!))})
inline_escape = (s)->
return inline_escaper\match(s)
escaper = re.compile("{~ (%utf8_char / ('\\' -> '\\\\') / [\n\r\t -~] / (. -> escape))* ~}",
{utf8_char: utf8_char_patt, escape:(=> ("\\%03d")\format(@byte!))})
escape = (s)->
return escaper\match(s)
tree_to_inline_nomsu = (tree)->
switch tree.type
when "Action"
2018-11-09 16:40:36 -08:00
nomsu = NomsuCode\from(tree.source)
2019-02-05 15:45:27 -08:00
if can_be_unary(tree)
nomsu\add tree[1]
arg_nomsu = tree_to_inline_nomsu(tree[2])
if tree[2].type == "MethodCall" or tree[2].type == "Action"
arg_nomsu\parenthesize!
nomsu\add arg_nomsu
return nomsu
2019-01-07 21:39:23 -08:00
num_args, num_words = 0, 0
2018-11-08 15:23:22 -08:00
for i,bit in ipairs tree
if type(bit) == "string"
2019-01-07 21:39:23 -08:00
num_words += 1
2018-11-09 17:48:35 -08:00
clump_words = if type(tree[i-1]) == 'string'
is_operator(bit) != is_operator(tree[i-1])
2019-03-20 16:40:32 -07:00
else bit == "'"
2018-11-29 14:57:22 -08:00
nomsu\add " " if i > 1 and not clump_words
nomsu\add bit
2018-11-08 15:23:22 -08:00
else
2018-12-14 17:49:36 -08:00
num_args += 1
2018-11-08 15:23:22 -08:00
arg_nomsu = tree_to_inline_nomsu(bit)
2019-03-20 16:40:32 -07:00
if tree[i+1] == "'" and bit.type == "Action" and can_be_unary(bit)
arg_nomsu\parenthesize!
2018-11-08 15:23:22 -08:00
if bit.type == "Block"
2019-02-05 15:45:27 -08:00
if i != #tree
nomsu\add " " if i > 1
2018-11-08 15:23:22 -08:00
arg_nomsu\parenthesize!
else
2019-02-05 15:45:27 -08:00
nomsu\add " " if i > 1
if bit.type == "MethodCall"
arg_nomsu\parenthesize!
elseif bit.type == "Action" and not can_be_unary(bit)
2018-11-08 15:23:22 -08:00
arg_nomsu\parenthesize!
2018-11-29 14:57:22 -08:00
nomsu\add arg_nomsu
2019-01-07 21:39:23 -08:00
if num_args == 1 and num_words == 0
2019-01-14 16:29:38 -08:00
nomsu\add "()"
2018-11-08 15:23:22 -08:00
return nomsu
2018-12-14 17:49:36 -08:00
when "MethodCall"
2018-12-14 20:09:15 -08:00
target_nomsu = tree_to_inline_nomsu(tree[1])
2018-12-18 19:30:01 -08:00
if tree[1].type == "Block"
2018-12-14 20:09:15 -08:00
target_nomsu\parenthesize!
2018-12-18 19:30:01 -08:00
nomsu = NomsuCode\from(tree.source, target_nomsu, ", ")
2019-01-25 15:01:29 -08:00
if #tree > 2 then nomsu\add "("
2018-12-14 19:23:26 -08:00
for i=2,#tree
nomsu\add "; " if i > 2
nomsu\add tree_to_inline_nomsu(tree[i])
2019-01-25 15:01:29 -08:00
if #tree > 2 then nomsu\add ")"
2018-12-14 19:23:26 -08:00
return nomsu
2018-12-14 17:49:36 -08:00
2018-11-08 15:23:22 -08:00
when "EscapedNomsu"
inner_nomsu = tree_to_inline_nomsu(tree[1])
unless tree[1].type == "List" or tree[1].type == "Dict" or tree[1].type == "Var"
inner_nomsu\parenthesize!
2018-11-09 16:40:36 -08:00
return NomsuCode\from(tree.source, "\\", inner_nomsu)
2018-11-08 15:23:22 -08:00
when "Block"
2018-11-09 16:40:36 -08:00
nomsu = NomsuCode\from(tree.source, ":")
2018-11-08 15:23:22 -08:00
for i,line in ipairs tree
2018-11-29 14:57:22 -08:00
nomsu\add(i == 1 and " " or "; ")
nomsu\add tree_to_inline_nomsu(line)
2018-11-08 15:23:22 -08:00
nomsu\parenthesize! if #tree > 1
return nomsu
when "Text"
add_text = (nomsu, tree)->
for i, bit in ipairs tree
if type(bit) == 'string'
escaped = inline_escape(bit)
2018-11-29 14:57:22 -08:00
nomsu\add inline_escape(bit)
2018-11-08 15:23:22 -08:00
elseif bit.type == "Text"
add_text(nomsu, bit)
else
interp_nomsu = tree_to_inline_nomsu(bit)
if bit.type != "Var" and bit.type != "List" and bit.type != "Dict"
interp_nomsu\parenthesize!
2019-01-10 16:33:37 -08:00
elseif bit.type == "Var" and type(bit[1]) == 'string' and
type(tree[i+1]) == 'string' and not match(tree[i+1], "^[ \n\t,.:;#(){}[%]]")
2018-11-08 15:23:22 -08:00
interp_nomsu\parenthesize!
2018-11-29 14:57:22 -08:00
nomsu\add "\\", interp_nomsu
2018-11-09 16:40:36 -08:00
nomsu = NomsuCode\from(tree.source)
2018-11-08 15:23:22 -08:00
add_text(nomsu, tree)
2018-11-09 16:40:36 -08:00
return NomsuCode\from(tree.source, '"', nomsu, '"')
2018-11-08 15:23:22 -08:00
when "List", "Dict"
2018-11-09 16:40:36 -08:00
nomsu = NomsuCode\from(tree.source, (tree.type == "List" and "[" or "{"))
2018-11-08 15:23:22 -08:00
for i, item in ipairs tree
2018-11-29 14:57:22 -08:00
nomsu\add ", " if i > 1
2018-12-18 19:30:01 -08:00
item_nomsu = tree_to_inline_nomsu(item, true)
--if item.type == "Block" or item.type == "Action" or item.type == "MethodCall"
-- item_nomsu\parenthesize!
2019-02-05 15:45:27 -08:00
if item.type == "MethodCall" or (item.type == "Block" and i < #tree)
2018-12-18 19:30:01 -08:00
item_nomsu\parenthesize!
nomsu\add item_nomsu
2018-11-29 14:57:22 -08:00
nomsu\add(tree.type == "List" and "]" or "}")
2018-11-08 15:23:22 -08:00
return nomsu
when "DictEntry"
key, value = tree[1], tree[2]
2018-12-18 19:30:01 -08:00
nomsu = NomsuCode\from(tree.source)
-- TODO: remove shim
if key.type != "Index"
key = {type:"Index", source:key.source, key}
nomsu\add tree_to_inline_nomsu(key)
2018-11-08 15:23:22 -08:00
if value
2018-12-18 19:30:01 -08:00
nomsu\add " = "
2018-11-08 15:23:22 -08:00
value_nomsu = tree_to_inline_nomsu(value)
2018-12-18 19:30:01 -08:00
value_nomsu\parenthesize! if value.type == "Block" or value.type == "Action" or value.type == "MethodCall"
2018-11-29 14:57:22 -08:00
nomsu\add value_nomsu
2018-11-08 15:23:22 -08:00
return nomsu
2018-12-18 19:30:01 -08:00
when "Index"
key = tree[1]
nomsu = NomsuCode\from(key.source, ".")
key_nomsu = if key.type == "Text" and #key == 1 and is_identifier(key[1])
key[1]
else
tree_to_inline_nomsu(key)
2018-12-30 23:56:28 -08:00
switch key.type
when "Block", "Action", "MethodCall", "IndexChain"
key_nomsu\parenthesize!
2018-12-18 19:30:01 -08:00
return NomsuCode\from(key.source, ".", key_nomsu)
2018-11-08 15:23:22 -08:00
when "IndexChain"
2018-11-09 16:40:36 -08:00
nomsu = NomsuCode\from(tree.source)
2018-12-30 23:56:28 -08:00
target = tree[1]
target_nomsu = tree_to_inline_nomsu(target)
switch target.type
2019-01-15 15:53:31 -08:00
when "Action", "MethodCall", "EscapedNomsu"
2018-12-30 23:56:28 -08:00
target_nomsu\parenthesize!
when "Number"
target_nomsu\parenthesize! if target_nomsu\text!\match("%.")
nomsu\add target_nomsu
for i=2,#tree
-- TODO: remove shim?
if tree[i].type != "Index"
tree[i] = {type:"Index", source:tree[i].source, tree[i]}
nomsu\add tree_to_inline_nomsu(tree[i])
2018-11-08 15:23:22 -08:00
return nomsu
when "Number"
2019-02-06 14:31:14 -08:00
-- Preserve original formatting, just make sure 0xdead_beef -> 0xDEAD_BEEF
n = tostring(tree[1])
s = if n\match("^-*0x")
n\upper!\gsub("0X", "0x")
elseif tree.hex and tonumber((n\gsub("_",""))) < 0
2018-11-11 15:25:25 -08:00
("-0x%X")\format(-tree[1])
elseif tree.hex
("0x%X")\format(tree[1])
2019-02-06 14:31:14 -08:00
else n
2018-11-11 15:25:25 -08:00
return NomsuCode\from(tree.source, s)
2018-11-08 15:23:22 -08:00
when "Var"
2018-12-14 19:23:26 -08:00
varname = tree[1]
2019-01-01 15:52:56 -08:00
if type(varname) == "string"
2018-12-14 17:49:36 -08:00
return NomsuCode\from(tree.source, "$", varname)
else
2019-01-01 15:52:56 -08:00
return NomsuCode\from(tree.source, "$(", tree_to_inline_nomsu(varname), ")")
2018-11-08 15:23:22 -08:00
when "FileChunks"
error("Can't inline a FileChunks")
when "Comment"
-- TODO: implement?
2018-11-09 17:02:39 -08:00
return NomsuCode\from(tree.source)
2018-11-08 15:23:22 -08:00
when "Error"
2019-03-04 14:25:56 -08:00
err_msg = pretty_error{
title:"Parse error"
error:tree.error, hint:tree.hint, source:tree\get_source_file!
start:tree.source.start, stop:tree.source.stop, filename:tree.source.filename
}
-- Coroutine yield here?
error(err_msg)
2018-11-08 15:23:22 -08:00
else
error("Unknown type: #{tree.type}")
tree_to_nomsu = (tree)->
2018-11-09 16:40:36 -08:00
nomsu = NomsuCode\from(tree.source)
2018-11-08 15:23:22 -08:00
-- For concision:
2018-12-30 19:04:34 -08:00
recurse = (t, argnum=nil)->
2018-11-08 15:23:22 -08:00
space = MAX_LINE - nomsu\trailing_line_len!
2018-11-11 15:05:18 -08:00
try_inline = true
2019-02-05 15:45:27 -08:00
for subtree in coroutine.wrap(-> (t\with(coroutine.yield) and nil))
2018-12-30 23:56:28 -08:00
switch subtree.type
when "Comment"
2018-11-11 15:05:18 -08:00
try_inline = false
2018-12-30 23:56:28 -08:00
when "Block"
if #subtree > 1
try_inline = false
when "Text"
indented = tree_to_nomsu(subtree)
indented_lines = [line for line in *indented\text!\lines! when line\match("^ +([^ ].*)")]
for i=#indented_lines,1,-1
if indented_lines[i]\match("^ *\\;$")
table.remove(indented_lines, i)
if #indented_lines > 1 or (#indented_lines == 1 and #indented_lines[1] > MAX_LINE + 8)
try_inline = false
2018-11-08 15:23:22 -08:00
2018-11-11 15:05:18 -08:00
local inline_nomsu
if try_inline
2018-11-08 15:23:22 -08:00
inline_nomsu = tree_to_inline_nomsu(t)
2019-02-05 15:45:27 -08:00
if t.type == "MethodCall"
inline_nomsu\parenthesize!
elseif t.type == "Action" and not can_be_unary(t)
2018-12-30 23:56:28 -08:00
inline_nomsu\parenthesize!
2018-11-11 15:05:18 -08:00
if #inline_nomsu\text! <= space or #inline_nomsu\text! <= 8
2018-12-19 03:02:19 -08:00
if t.type != "Text"
return inline_nomsu
2018-11-08 15:23:22 -08:00
indented = tree_to_nomsu(t)
2018-12-14 17:49:36 -08:00
if t.type == "Action" or t.type == "MethodCall"
2018-11-08 15:23:22 -08:00
if indented\is_multiline!
2019-03-04 14:25:56 -08:00
if argnum == nil or argnum == 1
return NomsuCode\from(t.source, "(\n ", indented, "\n)")
else
return NomsuCode\from(t.source, "\n ", indented)
2018-12-30 19:04:34 -08:00
elseif argnum and argnum > 1
return NomsuCode\from(t.source, "\n ", indented)
2018-12-18 19:30:01 -08:00
else
indented\parenthesize!
2018-12-30 23:56:28 -08:00
indented_lines = [line for line in *indented\text!\lines! when line\match("^ +([^ ].*)")]
if t.type == "Text"
for i=#indented_lines,1,-1
if indented_lines[i]\match("^ *\\;$")
table.remove(indented_lines, i)
if inline_nomsu and (#inline_nomsu\text! < MAX_LINE or #inline_nomsu\text! <= space) and
#indented_lines <= 1
2018-11-11 15:05:18 -08:00
return inline_nomsu
2018-11-08 15:23:22 -08:00
return indented
switch tree.type
when "FileChunks"
2018-11-09 17:32:48 -08:00
if tree.shebang
2019-02-05 15:45:27 -08:00
nomsu\add tree.shebang, "\n"
2018-11-08 15:23:22 -08:00
for chunk_no, chunk in ipairs tree
2018-11-29 14:57:22 -08:00
nomsu\add "\n\n#{("~")\rep(80)}\n\n" if chunk_no > 1
2018-11-08 15:23:22 -08:00
if chunk.type == "Block"
2018-11-29 14:57:22 -08:00
nomsu\add NomsuCode\from(chunk.source, table.unpack(tree_to_nomsu(chunk).bits, 2))
2018-11-08 15:23:22 -08:00
else
2018-11-29 14:57:22 -08:00
nomsu\add tree_to_nomsu(chunk)
2018-11-10 00:29:08 -08:00
2018-11-08 15:23:22 -08:00
return nomsu
when "Action"
2019-02-05 15:45:27 -08:00
if can_be_unary(tree) and not can_be_unary(tree[2])
nomsu\add tree[1]
nomsu\add recurse(tree[2])
return nomsu
2018-11-08 15:23:22 -08:00
next_space = ""
2018-11-10 00:29:08 -08:00
word_buffer = {}
2019-01-07 21:39:23 -08:00
num_args, num_words = 0, 0
2018-11-08 15:23:22 -08:00
for i,bit in ipairs tree
2018-12-30 19:04:34 -08:00
-- TODO: properly wrap super long chains of words
2018-11-08 15:23:22 -08:00
if type(bit) == "string"
2019-01-07 21:39:23 -08:00
num_words += 1
2018-11-10 00:29:08 -08:00
if #word_buffer > 0 and is_operator(bit) == is_operator(word_buffer[#word_buffer])
table.insert word_buffer, " "
table.insert word_buffer, bit
continue
2018-11-08 15:23:22 -08:00
2018-11-10 00:29:08 -08:00
if #word_buffer > 0
words = table.concat(word_buffer)
if next_space == " "
2018-11-11 15:05:18 -08:00
if nomsu\trailing_line_len! + #words > MAX_LINE and nomsu\trailing_line_len! > 8
2018-12-18 19:30:01 -08:00
next_space = "\n.."
2018-11-10 00:29:08 -08:00
elseif word_buffer[1] == "'"
next_space = ""
2018-11-29 14:57:22 -08:00
nomsu\add next_space, words
2018-11-10 00:29:08 -08:00
word_buffer = {}
next_space = " "
2018-12-14 17:49:36 -08:00
num_args += 1
2018-12-30 19:04:34 -08:00
bit_nomsu = recurse(bit, i)
2019-03-20 16:40:32 -07:00
if tree[i+1] == "'" and bit.type == "Action" and not bit_nomsu\is_multiline! and can_be_unary(bit)
bit_nomsu\parenthesize!
2018-12-18 19:30:01 -08:00
if bit.type == "Block"
2018-11-11 15:05:18 -08:00
-- Rule of thumb: nontrivial one-liner block arguments should be no more
-- than golden ratio * the length of the proceeding part of the line
2018-12-30 23:56:28 -08:00
if not bit_nomsu\is_multiline! and
(#bit_nomsu\text! > nomsu\trailing_line_len! * GOLDEN_RATIO and #bit_nomsu\text! > 8) or
#bit_nomsu\text! + nomsu\trailing_line_len! > MAX_LINE
2018-11-10 00:29:08 -08:00
bit_nomsu = tree_to_nomsu(bit)
2018-12-18 19:30:01 -08:00
elseif (not bit_nomsu\is_multiline! and
2018-11-11 15:05:18 -08:00
nomsu\trailing_line_len! + #bit_nomsu\text! > MAX_LINE and
nomsu\trailing_line_len! > 8)
2018-12-18 19:30:01 -08:00
if next_space == " " and #bit_nomsu\text! < MAX_LINE
2018-12-30 23:56:28 -08:00
if i == #tree
bit_nomsu = tree_to_inline_nomsu(bit)
next_space = "\n "
elseif bit.type == "List" or bit.type == "Dict"
bit_nomsu = tree_to_nomsu(bit)
else
next_space = "\n.."
2018-12-18 19:30:01 -08:00
elseif bit.type == 'Action' or bit.type == "MethodCall"
2018-12-30 19:04:34 -08:00
bit_nomsu = NomsuCode\from bit.source, "\n ", tree_to_nomsu(bit)
2018-11-10 00:29:08 -08:00
else
2018-12-18 19:30:01 -08:00
bit_nomsu = tree_to_nomsu(bit)
2018-12-30 19:04:34 -08:00
unless next_space == " " and bit_nomsu\text!\match("^[:\n]")
2018-11-29 14:57:22 -08:00
nomsu\add next_space
2018-11-10 00:29:08 -08:00
2018-11-29 14:57:22 -08:00
nomsu\add bit_nomsu
2018-12-30 23:56:28 -08:00
next_space = (bit.type == "Block" or bit_nomsu\text!\matches("\n [^\n]*$")) and "\n.." or " "
2018-11-08 15:23:22 -08:00
2018-11-10 00:29:08 -08:00
if #word_buffer > 0
words = table.concat(word_buffer)
if next_space == " "
2019-02-06 14:31:14 -08:00
if nomsu\trailing_line_len! + #words > MAX_LINE + 8 and nomsu\trailing_line_len! > 8
2018-12-18 19:30:01 -08:00
next_space = "\n.."
2018-11-10 00:29:08 -08:00
elseif word_buffer[1] == "'"
next_space = ""
2018-11-29 14:57:22 -08:00
nomsu\add next_space, words
2018-11-17 14:38:05 -08:00
next_space = " "
2019-01-07 21:39:23 -08:00
if num_args == 1 and num_words == 0
if next_space != " "
2019-01-14 16:29:38 -08:00
nomsu\add next_space
nomsu\add "()"
2018-12-14 17:49:36 -08:00
return nomsu
2018-11-08 15:23:22 -08:00
2018-12-14 17:49:36 -08:00
when "MethodCall"
target_nomsu = recurse(tree[1])
if tree[1].type == "Block" and not target_nomsu\is_multiline!
target_nomsu\parenthesize!
2018-12-18 19:30:01 -08:00
nomsu\add target_nomsu, ", "
2018-12-14 19:23:26 -08:00
inner_nomsu = NomsuCode!
for i=2,#tree
inner_nomsu\add "\n" if i > 2
inner_nomsu\add tree_to_nomsu(tree[i])
if #tree == 2 and nomsu\trailing_line_len! + #inner_nomsu\text!\match("^[^\n]*") < MAX_LINE
nomsu\add inner_nomsu
else
nomsu\add "\n ", inner_nomsu
2018-11-08 15:23:22 -08:00
return nomsu
when "EscapedNomsu"
2018-11-09 17:20:36 -08:00
nomsu = recurse(tree[1])
if tree[1].type == 'Block' and not nomsu\is_multiline!
nomsu\parenthesize!
2018-11-09 17:43:06 -08:00
return NomsuCode\from tree.source, "\\", nomsu
2018-11-08 15:23:22 -08:00
when "Block"
2018-11-10 00:29:08 -08:00
prev_line, needs_space = nil, {}
2018-11-08 15:23:22 -08:00
for i, line in ipairs tree
line_nomsu = tree_to_nomsu(line)
2018-11-10 00:29:08 -08:00
if i > 1
2018-11-29 14:57:22 -08:00
nomsu\add "\n"
2018-11-10 00:29:08 -08:00
-- Rule of thumb: add a blank line between two lines if both are
2018-12-19 03:02:19 -08:00
-- multi-line non-comments, or if a comment comes after a non-comment,
-- or if the last line starts with ".."
2018-11-10 00:29:08 -08:00
if tree[i-1].type != "Comment"
2019-01-14 16:29:38 -08:00
needs_space[i] = (line_nomsu\is_multiline! and prev_line\is_multiline!)
2019-03-27 14:40:48 -07:00
if (tree[i].type == "Comment" or needs_space[i] or needs_space[i-1] or
prev_line\text!\match("\n [^\n]*$"))
2018-11-29 14:57:22 -08:00
nomsu\add "\n"
nomsu\add line_nomsu
2018-11-10 00:29:08 -08:00
prev_line = line_nomsu
2018-11-09 16:40:36 -08:00
return NomsuCode\from(tree.source, ":\n ", nomsu)
2018-11-08 15:23:22 -08:00
when "Text"
-- Multi-line text has more generous wrap margins
2018-12-18 19:30:01 -08:00
max_line = MAX_LINE + 8
2018-11-08 15:23:22 -08:00
add_text = (tree)->
for i, bit in ipairs tree
if type(bit) == 'string'
bit = escape(bit)
for j, line in ipairs bit\lines!
if j > 1
2018-12-18 19:30:01 -08:00
if nomsu\text!\match(" $")
nomsu\add "\\;"
2018-11-29 14:57:22 -08:00
nomsu\add "\n"
2018-11-08 15:23:22 -08:00
elseif #line > 10 and nomsu\trailing_line_len! > max_line
2018-11-29 14:57:22 -08:00
nomsu\add "\\\n.."
2018-11-08 15:23:22 -08:00
while #line > 0
space = max_line - nomsu\trailing_line_len!
split = find(line, "[%p%s]", space)
2019-01-15 15:53:31 -08:00
if not split or split > space + 16
split = space + 16
if #line - split < 16
2018-11-08 15:23:22 -08:00
split = #line
bite, line = sub(line, 1, split), sub(line, split+1, -1)
2018-11-29 14:57:22 -08:00
nomsu\add bite
nomsu\add "\\\n.." if #line > 0
2018-11-08 15:23:22 -08:00
elseif bit.type == "Text"
add_text(bit)
else
2018-11-29 14:57:22 -08:00
nomsu\add "\\"
2018-11-08 15:23:22 -08:00
interp_nomsu = recurse(bit)
2019-01-01 17:15:51 -08:00
if interp_nomsu\is_multiline!
curr_indent = nomsu\text!\match("\n( *)[^\n]*$") or nomsu\text!\match("^( *)")
interp_nomsu = NomsuCode((interp_nomsu\text!\gsub("\n", "\n"..curr_indent)))
else
2018-12-30 23:56:28 -08:00
space = max_line - nomsu\trailing_line_len!
2018-11-08 15:23:22 -08:00
if bit.type == "Var"
2018-12-14 20:16:18 -08:00
next_str = tree[i+1]
while type(next_str) == 'table' and next_str.type == 'Text'
next_str = next_str[1]
if type(next_str) == 'string' and not match(next_str, "^[ \n\t,.:;#(){}[%]]")
2018-11-08 15:23:22 -08:00
interp_nomsu\parenthesize!
2018-12-30 23:56:28 -08:00
elseif #interp_nomsu\text! > space
interp_nomsu2 = if bit.type == "Action" or bit.type == "MethodCall"
NomsuCode\from(bit.source, "(\n ", tree_to_nomsu(bit), "\n)")
else
tree_to_nomsu(bit)
if #interp_nomsu2\text!\lines! > 3 or #interp_nomsu2\text! >= MAX_LINE*GOLDEN_RATIO
2019-01-01 17:15:51 -08:00
curr_indent = nomsu\text!\match("\n( *)[^\n]*$") or nomsu\text!\match("^( *)")
interp_nomsu2 = NomsuCode((interp_nomsu2\text!\gsub("\n", "\n"..curr_indent)))
2018-12-30 23:56:28 -08:00
interp_nomsu = interp_nomsu2
else
nomsu\add "\n..\\"
if bit.type == "EscapedNomsu" or bit.type == "Block" or bit.type == "IndexChain"
interp_nomsu\parenthesize!
2018-11-11 18:52:45 -08:00
elseif bit.type == "EscapedNomsu" or bit.type == "Block" or bit.type == "IndexChain"
2018-11-08 15:23:22 -08:00
interp_nomsu\parenthesize!
2019-03-20 16:40:32 -07:00
elseif bit.type == "Action" and can_be_unary(bit)
interp_nomsu\parenthesize!
2018-11-29 14:57:22 -08:00
nomsu\add interp_nomsu
2018-12-18 19:30:01 -08:00
if interp_nomsu\is_multiline! and bit.type == "Block"
2018-11-29 14:57:22 -08:00
nomsu\add "\n.."
2018-11-08 15:23:22 -08:00
add_text(tree)
2018-12-18 19:30:01 -08:00
if nomsu\text!\match(" $")
nomsu\add "\\;"
2018-12-19 03:02:19 -08:00
return NomsuCode\from(tree.source, '("\n ', nomsu, '\n")')
2018-11-08 15:23:22 -08:00
when "List", "Dict"
if #tree == 0
2018-11-29 14:57:22 -08:00
nomsu\add(tree.type == "List" and "[]" or "{}")
2018-11-08 15:23:22 -08:00
return nomsu
2019-03-04 14:25:56 -08:00
if #tree == 1 and tree[1].type == "Block"
block_lua = recurse(tree[1])
if block_lua\is_multiline! then block_lua\add "\n"
return if tree.type == "List" then
NomsuCode\from(tree.source, "[", block_lua, "]")
else
NomsuCode\from(tree.source, "{", block_lua, "}")
2018-11-11 15:05:18 -08:00
sep = ''
2019-01-18 14:28:45 -08:00
prev_item, needs_space = nil, {}
2018-11-08 15:23:22 -08:00
for i, item in ipairs tree
2018-12-18 19:30:01 -08:00
local item_nomsu
if item.type == 'MethodCall'
2018-11-08 15:23:22 -08:00
item_nomsu = recurse(item)
2018-12-18 19:30:01 -08:00
elseif item.type == 'Comment'
2018-11-11 15:05:18 -08:00
item_nomsu = tree_to_nomsu(item)
2018-12-18 19:30:01 -08:00
sep = '\n' if i > 1
2018-12-31 01:36:48 -08:00
elseif item.type == 'Block' and #item == 1
-- Comprehensions use the more concise ": for $ in $: ..." form
item_nomsu = tree_to_nomsu(item[1])
item_nomsu\prepend ": "
sep = '\n' if i > 1
2018-12-18 19:30:01 -08:00
else
item_nomsu = tree_to_inline_nomsu(item)
2019-01-07 21:39:23 -08:00
if nomsu\trailing_line_len! + #item_nomsu\text! > MAX_LINE
2018-12-30 23:56:28 -08:00
sep = '\n' if i > 1
2019-01-14 16:29:38 -08:00
item_nomsu = item.type == "Action" and tree_to_nomsu(item) or recurse(item)
2018-11-29 14:57:22 -08:00
nomsu\add sep
2019-01-18 14:28:45 -08:00
if sep == '\n'
-- Rule of thumb: add a blank line between two lines if both are
-- multi-line non-comments, or if a comment comes after a non-comment,
-- or if the last line starts with ".."
2019-03-27 14:40:48 -07:00
if i > 1 and tree[i-1].type != "Comment"
2019-01-18 14:28:45 -08:00
needs_space[i] = (item_nomsu\is_multiline! and prev_item\is_multiline!)
2019-03-27 14:40:48 -07:00
if (tree[i].type == "Comment" or needs_space[i] or needs_space[i-1] or
prev_item\text!\match("\n [^\n]*$"))
2019-01-18 14:28:45 -08:00
nomsu\add "\n"
2018-11-29 14:57:22 -08:00
nomsu\add item_nomsu
2019-01-18 14:28:45 -08:00
prev_item = item_nomsu
2018-12-31 01:36:48 -08:00
if item_nomsu\is_multiline! or item.type == 'Comment' or item.type == "Block" or
nomsu\trailing_line_len! + #tostring(item_nomsu) >= MAX_LINE
2018-11-11 15:05:18 -08:00
sep = '\n'
else
sep = ', '
2018-11-08 15:23:22 -08:00
return if tree.type == "List" then
2018-12-18 19:30:01 -08:00
NomsuCode\from(tree.source, "[\n ", nomsu, "\n]")
2018-11-08 15:23:22 -08:00
else
2018-12-18 19:30:01 -08:00
NomsuCode\from(tree.source, "{\n ", nomsu, "\n}")
2018-11-08 15:23:22 -08:00
when "DictEntry"
key, value = tree[1], tree[2]
2018-12-18 19:30:01 -08:00
nomsu = NomsuCode\from(tree.source)
-- TODO: remove shim
if key.type != "Index"
key = {type:"Index", source:key.source, key}
nomsu\add tree_to_nomsu(key)
2018-11-12 14:08:42 -08:00
if value
2018-12-30 19:04:34 -08:00
value_nomsu = recurse(value)
nomsu\add " = ", value_nomsu
2018-11-08 15:23:22 -08:00
return nomsu
when "Comment"
2019-02-05 15:45:27 -08:00
nomsu\add "###", (tree[1]\gsub("\n", "\n "))
2018-11-08 15:23:22 -08:00
return nomsu
2018-12-18 19:30:01 -08:00
when "IndexChain", "Index", "Number", "Var", "Comment", "Error"
2018-11-08 15:23:22 -08:00
return tree_to_inline_nomsu tree
else
error("Unknown type: #{tree.type}")
return {:tree_to_nomsu, :tree_to_inline_nomsu}