nomsu/code_obj.moon

257 lines
8.1 KiB
Plaintext
Raw Normal View History

2018-06-05 03:39:44 -07:00
-- This file contains objects that are used to track code positions and incrementally
-- build up generated code, while keeping track of where it came from, and managing
-- indentation levels.
2018-04-11 20:05:12 -07:00
{:insert, :remove, :concat} = table
2018-06-18 15:44:29 -07:00
local LuaCode, NomsuCode, Source
2018-04-11 21:07:13 -07:00
export LINE_STARTS
2018-04-11 20:05:12 -07:00
class Source
new: (@filename, @start, @stop)=>
@from_string: (str)=>
filename,start,stop = str\match("^@(.-)%[(%d+):(%d+)%]$")
unless filename
filename,start = str\match("^@(.-)%[(%d+)%]$")
return @(filename or str, tonumber(start or 1), tonumber(stop))
@is_instance: (x)=> type(x) == 'table' and x.__class == @
__tostring: =>
if @stop
"@#{@filename}[#{@start}:#{@stop}]"
else
"@#{@filename}[#{@start}]"
__eq: (other)=>
getmetatable(@) == getmetatable(other) and @filename == other.filename and @start == other.start and @stop == other.stop
2018-04-11 20:05:12 -07:00
__lt: (other)=>
2018-04-13 14:54:35 -07:00
assert(@filename == other.filename, "Cannot compare sources from different files")
2018-04-11 20:05:12 -07:00
return if @start == other.start
(@stop or @start) < (other.stop or other.start)
2018-04-11 20:05:12 -07:00
else @start < other.start
2018-04-11 20:05:12 -07:00
__le: (other)=>
2018-04-13 14:54:35 -07:00
assert(@filename == other.filename, "Cannot compare sources from different files")
2018-04-11 20:05:12 -07:00
return if @start == other.start
(@stop or @start) <= (other.stop or other.start)
2018-04-11 20:05:12 -07:00
else @start <= other.start
__add: (offset)=>
if type(self) == 'number'
offset, self = self, offset
else if type(offset) != 'number' then error("Cannot add Source and #{type(offset)}")
return Source(@filename, @start+offset, @stop)
2018-04-11 20:05:12 -07:00
class Code
2018-04-11 21:07:13 -07:00
new: (@source, ...)=>
2018-07-10 17:34:39 -07:00
@bits = {}
@indents, @current_indent = {}, 0
@trailing_line_len = 0
2018-04-13 14:54:35 -07:00
if type(@source) == 'string'
@source = Source\from_string(@source)
assert(@source and Source\is_instance(@source), "Source has the wrong type")
@append(...)
append: (...)=>
n = select("#",...)
bits, indents = @bits, @indents
match = string.match
for i=1,n
b = select(i, ...)
2018-07-10 17:34:39 -07:00
assert(b, "code bit is nil")
2018-07-12 21:42:09 -07:00
if Source\is_instance(b) then require('ldt').breakpoint!
if b == '' then continue
bits[#bits+1] = b
if type(b) == 'string'
2018-07-10 17:34:39 -07:00
trailing_text, spaces = match(b, "\n(([ ]*)[^\n]*)$")
if trailing_text
@current_indent = #spaces
2018-07-10 17:34:39 -07:00
@trailing_line_len = #trailing_text
else
if #b.indents > 1
@trailing_line_len = b.trailing_line_len
else
@trailing_line_len += #tostring(b)
if @current_indent != 0
indents[#bits] = @current_indent
@__str = nil
2018-06-19 00:44:17 -07:00
concat_append: (values, joiner, wrapping_joiner)=>
wrapping_joiner or= joiner
bits, indents = @bits, @indents
match = string.match
2018-06-19 00:44:17 -07:00
line_len = 0
for i=1,#values
b = values[i]
if i > 1
2018-06-19 00:44:17 -07:00
if line_len > 80
bits[#bits+1] = wrapping_joiner
line_len = 0
else
bits[#bits+1] = joiner
bits[#bits+1] = b
2018-06-19 00:44:17 -07:00
if type(b) != 'string' and @current_indent != 0
indents[#bits] = @current_indent
2018-06-19 00:44:17 -07:00
b_str = tostring(b)
line, spaces = match(b_str, "\n(([ ]*)[^\n]*)$")
if spaces
if type(b) == 'string'
@current_indent = #spaces
line_len = #line
else
line_len += #b
@__str = nil
prepend: (...)=>
n = select("#",...)
bits, indents = @bits, @indents
for i=#bits+n,n+1,-1
bits[i] = bits[i-n]
for i=1,n
bits[i] = select(i, ...)
@current_indent = 0
for i,b in ipairs(bits)
if type(b) == 'string'
if spaces = b\match("\n([ ]*)[^\n]*$")
@current_indent = #spaces
elseif @current_indent != 0
indents[i] = @current_indent
else indents[i] = nil
@__str = nil
2018-06-18 15:44:29 -07:00
class LuaCode extends Code
new: (...)=>
super ...
2018-04-11 20:05:12 -07:00
@free_vars = {}
@is_value = false
2018-04-20 14:33:49 -07:00
@__str = nil
2018-04-11 20:05:12 -07:00
@Value = (...)->
2018-06-18 15:44:29 -07:00
lua = LuaCode(...)
lua.is_value = true
return lua
add_free_vars: (vars)=>
return unless #vars > 0
2018-04-11 20:05:12 -07:00
seen = {[v]:true for v in *@free_vars}
for var in *vars
2018-06-18 18:10:59 -07:00
assert type(var) == 'string'
2018-04-11 20:05:12 -07:00
unless seen[var]
@free_vars[#@free_vars+1] = var
seen[var] = true
2018-04-20 14:33:49 -07:00
@__str = nil
2018-04-19 17:23:44 -07:00
remove_free_vars: (vars)=>
return unless #vars > 0
2018-04-19 17:23:44 -07:00
removals = {}
for var in *vars
2018-06-18 18:10:59 -07:00
assert type(var) == 'string'
removals[var] = true
2018-04-20 16:23:53 -07:00
stack = {self}
while #stack > 0
lua, stack[#stack] = stack[#stack], nil
for i=#lua.free_vars,1,-1
2018-06-18 18:10:59 -07:00
free_var = lua.free_vars[i]
if removals[free_var]
2018-04-20 16:23:53 -07:00
remove lua.free_vars, i
for b in *lua.bits
2018-04-19 17:23:44 -07:00
if type(b) != 'string'
2018-04-20 16:23:53 -07:00
stack[#stack+1] = b
2018-04-20 14:33:49 -07:00
@__str = nil
2018-04-11 20:05:12 -07:00
2018-04-19 17:23:44 -07:00
declare_locals: (to_declare=nil)=>
if to_declare == nil
to_declare, seen = {}, {}
gather_from = =>
for var in *@free_vars
unless seen[var]
seen[var] = true
to_declare[#to_declare+1] = var
for bit in *@bits
2018-06-18 15:44:29 -07:00
if bit.__class == LuaCode
2018-04-19 17:23:44 -07:00
gather_from bit
gather_from self
if #to_declare > 0
@remove_free_vars to_declare
2018-06-18 18:10:59 -07:00
@prepend "local #{concat to_declare, ", "};\n"
return to_declare
2018-04-11 20:05:12 -07:00
2018-06-18 18:10:59 -07:00
as_statements: (prefix="", suffix=";")=>
unless @is_value
return self
statements = LuaCode(@source)
if prefix != ""
statements\append prefix
statements\append self
if suffix != ""
statements\append suffix
return statements
as_expr: =>
if @is_value
return self
error("Cannot convert to expression: #{tostring self}")
2018-06-18 18:10:59 -07:00
2018-04-11 20:05:12 -07:00
__tostring: =>
2018-04-20 14:33:49 -07:00
if @__str == nil
buff, indents = {}, @indents
2018-04-20 14:33:49 -07:00
for i,b in ipairs @bits
b = tostring(b)
if indents[i]
b = b\gsub("\n", "\n"..((" ")\rep(indents[i])))
buff[#buff+1] = b
2018-04-20 14:33:49 -07:00
@__str = concat(buff, "")
return @__str
2018-04-11 20:05:12 -07:00
__len: =>
#tostring(self)
2018-04-11 20:05:12 -07:00
make_offset_table: =>
2018-04-11 20:05:12 -07:00
-- Return a mapping from output (lua) character number to input (nomsu) character number
2018-04-20 16:23:53 -07:00
lua_to_nomsu, nomsu_to_lua = {}, {}
2018-04-20 14:33:49 -07:00
walk = (lua, pos)->
for b in *lua.bits
if type(b) == 'string'
2018-04-20 16:23:53 -07:00
if lua.source
lua_to_nomsu[pos] = lua.source.start
nomsu_to_lua[lua.source.start] = pos
else
2018-04-20 16:23:53 -07:00
walk b, pos
pos += #tostring(b)
2018-04-20 14:33:49 -07:00
walk self, 1
2018-04-20 16:23:53 -07:00
return {
nomsu_filename:@source.filename
lua_filename:tostring(@source)..".lua", lua_file:@stringify!
:lua_to_nomsu, :nomsu_to_lua
}
2018-04-11 20:05:12 -07:00
parenthesize: =>
if @is_value
@prepend "("
@append ")"
else
error "Cannot parenthesize lua statements"
2018-04-11 20:05:12 -07:00
2018-06-18 15:44:29 -07:00
class NomsuCode extends Code
__tostring: =>
if @__str == nil
buff, indents = {}, @indents
for i,b in ipairs @bits
b = tostring(b)
if indents[i]
b = b\gsub("\n", "\n"..((" ")\rep(indents[i])))
buff[#buff+1] = b
@__str = concat(buff, "")
return @__str
__len: =>
#tostring(self)
2018-04-25 16:04:46 -07:00
parenthesize: =>
@prepend "("
@append ")"
2018-06-18 15:44:29 -07:00
return {:Code, :NomsuCode, :LuaCode, :Source}