Mostly working, albeit not pretty.

This commit is contained in:
Bruce Hill 2018-04-02 15:29:44 -07:00
parent 5d27d080d2
commit 9e0bebace1
2 changed files with 433 additions and 188 deletions

358
ldt.lua
View File

@ -3,6 +3,7 @@ local re = require('re')
local repr = require('repr')
local ldb
local AUTO = { }
local PARENT = { }
local _error = error
local _assert = assert
local callstack_range
@ -42,7 +43,7 @@ wrap_text = function(text, width)
end
return lines
end
local color
local Color
do
local color_index = 0
local existing = { }
@ -68,11 +69,11 @@ do
({:bg: "on " {color} :} / ({:fg: color :} (" on " {:bg: color :})?))
{:attrs: {| (" " {attr})* |} :})
|}
attr <- "blink" / "bold" / "dim" / "invis" / "normal" / "protect" / "reverse" / "standout" / "underline"
attr <- "blink" / "bold" / "dim" / "invis" / "normal" / "protect" / "reverse" / "standout" / "underline" / "altcharset"
color <- "black" / "blue" / "cyan" / "green" / "magenta" / "red" / "white" / "yellow" / "default"
]])
C.COLOR_DEFAULT = -1
color = function(s)
Color = function(s)
if s == nil then
s = "default"
end
@ -129,7 +130,7 @@ do
x = x + (self.column_widths[c] - #chunk)
end
if c < #self.columns then
chstr:set_ch(x, C.ACS_VLINE, color("black bold"))
chstr:set_ch(x, C.ACS_VLINE, Color("black bold"))
x = x + 1
end
end
@ -252,8 +253,8 @@ do
self.selected = nil
self.columns = { }
self.column_widths = { }
self.active_frame = color("yellow bold")
self.inactive_frame = color("blue dim")
self.active_frame = Color("yellow bold")
self.inactive_frame = Color("blue dim")
self.colors = { }
for i = 1, select('#', ...) - 1, 2 do
local col = select(i, ...)
@ -265,7 +266,7 @@ do
end
table.insert(self.column_widths, w)
local color_fn = select(i + 1, ...) or (function(self, i)
return color()
return Color()
end)
_assert(type(color_fn) == 'function', "Invalid color function type: " .. tostring(type(color_fn)))
table.insert(self.colors, color_fn)
@ -321,7 +322,7 @@ do
local cols = {
line_nums,
(function(self, i)
return i == self.selected and color() or color("yellow")
return i == self.selected and Color() or Color("yellow")
end),
...
}
@ -354,6 +355,196 @@ do
end
NumberedPad = _class_0
end
local line_matcher = re.compile('lines<-{|(line "\n")* line|} line<-{[^\n]*}')
local expansions = { }
local KEY = { }
local VALUE = { }
local TOP_LOCATION = { }
local locations = { }
local Location
Location = function(old_loc, kind, key)
if not (locations[old_loc]) then
locations[old_loc] = { }
end
if not (locations[old_loc][kind]) then
locations[old_loc][kind] = { }
end
if not (locations[old_loc][kind][key]) then
locations[old_loc][kind][key] = { }
end
return locations[old_loc][kind][key]
end
local expand
expand = function(kind, key, location)
expansions[Location(location, kind, key)] = true
end
local collapse
collapse = function(kind, key, location)
expansions[Location(location, kind, key)] = nil
end
local is_key_expanded
is_key_expanded = function(location, key)
return expansions[Location(location, KEY, key)]
end
local is_value_expanded
is_value_expanded = function(location, key)
return expansions[Location(location, VALUE, key)]
end
local make_lines
make_lines = function(location, x, width)
local type_colors = {
string = Color('blue on black'),
number = Color('magenta'),
boolean = Color('yellow'),
["nil"] = Color('cyan'),
table = Color('white bold'),
["function"] = Color('green'),
userdata = Color('cyan bold'),
thread = Color('blue')
}
setmetatable(type_colors, {
__index = function()
return Color('red bold')
end
})
local _exp_0 = type(x)
if 'string' == _exp_0 then
local lines = { }
local _list_0 = line_matcher:match(x)
for _index_0 = 1, #_list_0 do
local line = _list_0[_index_0]
local wrapped = wrap_text(line, width - 1)
for i, subline in ipairs(wrapped) do
local _line = {
location = location
}
if i > 1 then
table.insert(_line, C.ACS_BULLET)
table.insert(_line, Color('black bold altcharset'))
end
table.insert(_line, subline)
table.insert(_line, Color('blue on black bold'))
table.insert(lines, _line)
end
end
return lines
elseif 'table' == _exp_0 then
local prepend
prepend = function(line, ...)
for i = 1, select('#', ...) do
table.insert(line, i, (select(i, ...)))
end
end
local lines = { }
for k, v in pairs(x) do
if is_key_expanded(location, k) and is_value_expanded(location, k) then
table.insert(lines, {
location = Location(location, KEY, k),
'key/value:',
Color('white')
})
local key_lines = make_lines(Location(location, KEY, k), k, width - 2)
for i, key_line in ipairs(key_lines) do
for j = 2, #key_line, 2 do
key_line[j] = key_line[j] | C.A_REVERSE
end
if i == 1 then
prepend(key_line, C.ACS_LLCORNER, Color(), C.ACS_TTEE, Color())
else
prepend(key_line, ' ', Color(), C.ACS_VLINE, Color())
end
table.insert(lines, key_line)
end
local value_lines = make_lines(Location(location, VALUE, k), v, width - 2)
for i, value_line in ipairs(value_lines) do
if i == 1 then
prepend(value_line, ' ', Color(), C.ACS_LLCORNER, Color())
else
prepend(value_line, ' ', Color())
end
table.insert(lines, value_line)
end
elseif is_value_expanded(location, k) then
local k_str = type(k) == 'string' and k:gsub('\n', '\\n') or repr(k, 2)
if #k_str > width then
k_str = k_str:sub(1, width - 3) .. '...'
end
table.insert(lines, {
location = Location(location, KEY, k),
k_str,
type_colors[type(k)] | C.A_REVERSE
})
local v_lines = make_lines(Location(location, VALUE, k), v, width - 1)
prepend(v_lines[1], C.ACS_LLCORNER, Color())
for i = 2, #v_lines do
prepend(v_lines[i], ' ', Color())
end
for _index_0 = 1, #v_lines do
local v_line = v_lines[_index_0]
table.insert(lines, v_line)
end
elseif is_key_expanded(location, k) then
local k_lines = make_lines(Location(location, KEY, k), k, width - 1)
for _index_0 = 1, #k_lines do
local k_line = k_lines[_index_0]
for i = 2, #k_line, 2 do
k_line[i] = k_lin[i] | C.A_REVERSE
end
end
for i = 1, #k_lines - 1 do
prepend(k_lines[i], ' ', Color())
end
prepend(k_lines[#k_lines - 1], C.ACS_ULCORNER, Color())
for _index_0 = 1, #k_lines do
local k_line = k_lines[_index_0]
table.insert(lines, k_line)
end
local v_str = type(v) == 'string' and v:gsub('\n', '\\n') or repr(v, 2)
if #v_str > width then
v_str = v_str:sub(1, width - 3) .. '...'
end
table.insert(lines, {
location = Location(location, VALUE, k),
v_str,
type_colors[type(v)]
})
else
local k_space = math.floor((width - 3) / 3)
local k_str = type(k) == 'string' and k:gsub('\n', '\\n') or repr(k, 2)
if #k_str > k_space then
k_str = k_str:sub(1, k_space - 3) .. '...'
end
local v_space = (width - 3) - #k_str
local v_str = type(v) == 'string' and v:gsub('\n', '\\n') or repr(v, 2)
if #v_str > v_space then
v_str = v_str:sub(1, v_space - 3) .. '...'
end
table.insert(lines, {
location = Location(location, VALUE, k),
k_str,
type_colors[type(k)] | C.A_REVERSE,
' = ',
Color(),
v_str,
type_colors[type(v)]
})
end
end
return lines
else
local str = repr(x, 2)
if #str > width then
str = str:sub(1, width - 3) .. '...'
end
return {
{
location = location,
str,
type_colors[type(x)]
}
}
end
end
local DataViewer
do
local _class_0
@ -377,12 +568,12 @@ do
end
local old_selected
old_selected, self.selected = self.selected, i
if old_selected then
if old_selected and self.chstrs[old_selected] then
self.chstrs[old_selected]:set_str(0, ' ')
self._pad:mvaddchstr(old_selected - 1, 0, self.chstrs[old_selected])
end
if self.selected then
self.chstrs[self.selected]:set_ch(0, C.ACS_RARROW, color('green bold'))
self.chstrs[self.selected]:set_ch(0, C.ACS_RARROW, Color('green bold'))
self._pad:mvaddchstr(self.selected - 1, 0, self.chstrs[self.selected])
local scrolloff = 3
if self.selected > self.scroll_y + (self.height - 2) - scrolloff then
@ -419,16 +610,16 @@ do
elseif ('K'):byte() == _exp_0 then
return self:scroll(-10, 0)
elseif C.KEY_RIGHT == _exp_0 or ("l"):byte() == _exp_0 then
self.chstr_actions[self.selected]()
expansions[self.chstr_locations[self.selected]] = true
return self:full_refresh()
elseif ("L"):byte() == _exp_0 then
self.chstr_actions[self.selected]()
expansions[self.chstr_locations[self.selected]] = true
return self:full_refresh()
elseif C.KEY_LEFT == _exp_0 or ("h"):byte() == _exp_0 then
self.chstr_actions[self.selected]()
expansions[self.chstr_locations[self.selected]] = nil
return self:full_refresh()
elseif ("H"):byte() == _exp_0 then
self.chstr_actions[self.selected]()
expansions[self.chstr_locations[self.selected]] = nil
return self:full_refresh()
end
end
@ -440,81 +631,48 @@ do
self.data, self.label, self.y, self.x = data, label, y, x
self.scroll_y, self.scroll_x = 1, 1
self.selected = nil
self.active_frame = color("yellow bold")
self.inactive_frame = color("blue dim")
self.active_frame = Color("yellow bold")
self.inactive_frame = Color("blue dim")
self.expansions = { }
self.full_refresh = function()
self.chstrs, self.chstr_actions = { }, { }
local line_matcher = re.compile('lines<-{|(line "\n")* line|} line<-{[^\n]*}')
local old_location = self.selected and self.chstr_locations and self.chstr_locations[self.selected]
self.chstrs, self.chstr_locations = { }, { }
line_matcher = re.compile('lines<-{|(line "\n")* line|} line<-{[^\n]*}')
local W = width - 3
local add_line
add_line = function(text, attr, indent, action)
local requested = line_matcher:match(text)
for _index_0 = 1, #requested do
local line = requested[_index_0]
local wrapped = wrap_text(line, W - 2 * indent)
for i, subline in ipairs(wrapped) do
local chstr = C.new_chstr(W + 1)
chstr:set_str(0, ' ', color())
if i == 1 then
chstr:set_ch(1, C.ACS_CKBOARD, color('black bold'), 2 * indent)
elseif indent > 0 then
chstr:set_ch(1, C.ACS_CKBOARD, color('black bold'), 2 * indent - 1)
chstr:set_ch(1 + 2 * indent - 1, C.ACS_BULLET, color('black bold'))
end
chstr:set_str(1 + 2 * indent, subline, attr)
chstr:set_str(1 + 2 * indent + #subline, ' ', attr, W - (2 * indent + #subline))
table.insert(self.chstrs, chstr)
table.insert(self.chstr_actions, action)
local lines = make_lines(TOP_LOCATION, self.data, W)
for i, line in ipairs(lines) do
local chstr = C.new_chstr(W)
local offset = 1
for j = 1, #line - 1, 2 do
local chunk, attrs = line[j], line[j + 1]
if type(chunk) == 'number' then
chstr:set_ch(offset, chunk, attrs)
offset = offset + 1
else
chstr:set_str(offset, chunk, attrs)
offset = offset + #chunk
end
end
end
local add
add = function(data, expansions, indent)
if type(data) == 'table' then
for k, v in pairs(data) do
if expansions[k] then
add_line("(-)", color('yellow bold'), indent, (function()
expansions[k] = nil
end))
add(k, expansions[k], indent + 1)
add(v, expansions[k], indent + 1)
else
add_line("[" .. tostring(repr(k, 1)) .. "] = " .. tostring(repr(v, 1)), color('yellow bold'), indent, (function()
expansions[k] = { }
end))
end
end
return
if offset < W then
chstr:set_str(offset, ' ', attrs, W - offset)
end
local _color
local _exp_0 = type(data)
if 'string' == _exp_0 then
_color = color('default')
elseif 'number' == _exp_0 then
data = tostring(data)
_color = color('magenta')
elseif 'boolean' == _exp_0 then
data = tostring(data)
_color = data and color('green') or color('red')
elseif 'nil' == _exp_0 then
data = tostring(data)
_color = color('cyan')
elseif 'function' == _exp_0 or 'userdata' == _exp_0 or 'thread' == _exp_0 then
data = repr(data)
_color = color('blue bold')
else
_color = error("Unsupported type: " .. tostring(type(data)))
end
return add_line(data, _color, indent, (function() end))
table.insert(self.chstrs, chstr)
table.insert(self.chstr_locations, line.location)
end
add(self.data, self.expansions, 0)
self._height, self._width = #self.chstrs, self.width - 2
self._pad:resize(self._height, self._width)
for i, chstr in ipairs(self.chstrs) do
self._pad:mvaddchstr(i - 1, 0, chstr)
end
self.dirty = true
if old_location then
for i, loc in ipairs(self.chstr_locations) do
if loc == old_location then
self:select(i)
break
end
end
end
end
self.height, self.width = height, width
self._frame = C.newwin(self.height, self.width, self.y, self.x)
@ -604,7 +762,7 @@ ldb = {
C.start_color()
C.use_default_colors()
do
stdscr:wbkgd(color("yellow on red bold"))
stdscr:wbkgd(Color("yellow on red bold"))
stdscr:clear()
stdscr:refresh()
local lines = wrap_text("ERROR!\n \n " .. err_msg .. "\n \npress any key...", math.floor(SCREEN_W / 2))
@ -625,7 +783,7 @@ ldb = {
stdscr:getch()
end
stdscr:keypad()
stdscr:wbkgd(color())
stdscr:wbkgd(Color())
stdscr:clear()
stdscr:refresh()
local pads = { }
@ -635,9 +793,9 @@ ldb = {
err_msg_lines[i] = (" "):rep(2) .. line
end
pads.err = Pad("Error Message", 0, 0, AUTO, SCREEN_W, err_msg_lines, function(self, i)
return color("red bold")
return Color("red bold")
end)
pads.err._frame:attrset(color("red"))
pads.err._frame:attrset(Color("red"))
pads.err:refresh()
end
local stack_locations = { }
@ -683,9 +841,9 @@ ldb = {
local stack_h = math.max(#stack_names + 2, math.floor(2 / 3 * SCREEN_H))
local stack_w = math.min(max_fn_name + 3 + max_filename, math.floor(1 / 3 * SCREEN_W))
pads.stack = Pad("(C)allstack", pads.err.height, SCREEN_W - stack_w, stack_h, stack_w, stack_names, (function(self, i)
return (i == self.selected) and color("black on green") or color("green bold")
return (i == self.selected) and Color("black on green") or Color("green bold")
end), stack_locations, (function(self, i)
return (i == self.selected) and color("black on cyan") or color("cyan bold")
return (i == self.selected) and Color("black on cyan") or Color("cyan bold")
end))
end
local show_src
@ -695,15 +853,15 @@ ldb = {
pads.src:select(line_no)
pads.src.colors[2] = function(self, i)
if i == line_no and i == self.selected then
return color("yellow on red bold")
return Color("yellow on red bold")
elseif i == line_no then
return color("yellow on red")
return Color("yellow on red")
elseif err_lines[tostring(filename) .. ":" .. tostring(i)] == true then
return color("red on black bold")
return Color("red on black bold")
elseif i == self.selected then
return color("reverse")
return Color("reverse")
else
return color()
return Color()
end
end
for line, _ in pairs(err_lines) do
@ -726,15 +884,15 @@ ldb = {
end
pads.src = NumberedPad("(S)ource Code", pads.err.height, 0, pads.stack.height, pads.stack.x, src_lines, function(self, i)
if i == line_no and i == self.selected then
return color("yellow on red bold")
return Color("yellow on red bold")
elseif i == line_no then
return color("yellow on red")
return Color("yellow on red")
elseif err_lines[tostring(filename) .. ":" .. tostring(i)] == true then
return color("red on black bold")
return Color("red on black bold")
elseif i == self.selected then
return color("reverse")
return Color("reverse")
else
return color()
return Color()
end
end)
pads.src:select(line_no)
@ -747,7 +905,7 @@ ldb = {
s = (" "):rep(math.floor((pads.stack.x - 2 - #s) / 2)) .. s
table.insert(lines, s)
pads.src = Pad("(S)ource Code", pads.err.height, 0, pads.stack.height, pads.stack.x, lines, function()
return color("red")
return Color("red")
end)
end
pads.src.filename = filename
@ -786,7 +944,7 @@ ldb = {
local var_x = 0
local height = SCREEN_H - (pads.err.height + pads.stack.height)
pads.vars = Pad("(V)ars", var_y, var_x, height, AUTO, var_names, (function(self, i)
return i == self.selected and color('reverse') or color()
return i == self.selected and Color('reverse') or Color()
end))
pads.vars.on_select = function(self, var_index)
if var_index == nil then
@ -800,7 +958,7 @@ ldb = {
_ = [[ switch type_str
when "string"
pads.values = Pad "(D)ata [string]",var_y,value_x,pads.vars.height,value_w,
wrap_text(value, value_w-2), (i)=>color()
wrap_text(value, value_w-2), (i)=>Color()
when "table"
value_str = repr(value, 3)
mt = getmetatable(value)
@ -822,22 +980,22 @@ ldb = {
value_str = table.concat ["#{key_repr k} = #{repr v,2}" for k,v in pairs(value)], "\n \n"
pads.values = Pad "(D)ata [#{type_str}]",var_y,value_x,pads.vars.height,value_w,
wrap_text(value_str, value_w-2), (i)=>color("cyan bold")
wrap_text(value_str, value_w-2), (i)=>Color("cyan bold")
when "function"
info = debug.getinfo(value, 'nS')
s = ("function '%s' defined at %s:%s")\format info.name or var_names[var_index],
info.short_src, info.linedefined
pads.values = Pad "(D)ata [#{type_str}]",var_y,value_x,pads.vars.height,value_w,
wrap_text(s, value_w-2), (i)=>color("green bold")
wrap_text(s, value_w-2), (i)=>Color("green bold")
when "number"
pads.values = Pad "(D)ata [#{type_str}]",var_y,value_x,pads.vars.height,value_w,
wrap_text(repr(value), value_w-2), (i)=>color("magenta bold")
wrap_text(repr(value), value_w-2), (i)=>Color("magenta bold")
when "boolean"
pads.values = Pad "(D)ata [#{type_str}]",var_y,value_x,pads.vars.height,value_w,
wrap_text(repr(value), value_w-2), (i)=> value and color("green bold") or color("red bold")
wrap_text(repr(value), value_w-2), (i)=> value and Color("green bold") or Color("red bold")
else
pads.values = Pad "(D)ata [#{type_str}]",var_y,value_x,pads.vars.height,value_w,
wrap_text(repr(value), value_w-2), (i)=>color()
wrap_text(repr(value), value_w-2), (i)=>Color()
]]
collectgarbage()
return collectgarbage()

263
ldt.moon
View File

@ -3,6 +3,7 @@ re = require 're'
repr = require 'repr'
local ldb
AUTO = {} -- Singleton
PARENT = {} -- Singleton
_error = error
_assert = assert
@ -37,7 +38,7 @@ wrap_text = (text, width)->
return lines
local color
local Color
do
color_index = 0
existing = {}
@ -55,11 +56,11 @@ do
({:bg: "on " {color} :} / ({:fg: color :} (" on " {:bg: color :})?))
{:attrs: {| (" " {attr})* |} :})
|}
attr <- "blink" / "bold" / "dim" / "invis" / "normal" / "protect" / "reverse" / "standout" / "underline"
attr <- "blink" / "bold" / "dim" / "invis" / "normal" / "protect" / "reverse" / "standout" / "underline" / "altcharset"
color <- "black" / "blue" / "cyan" / "green" / "magenta" / "red" / "white" / "yellow" / "default"
]]
C.COLOR_DEFAULT = -1
color = (s="default")->
Color = (s="default")->
t = _assert(color_lang\match(s), "Invalid color: #{s}")
if t.fg then t.fg = C["COLOR_"..t.fg\upper!]
if t.bg then t.bg = C["COLOR_"..t.bg\upper!]
@ -76,8 +77,8 @@ class Pad
@columns = {}
@column_widths = {}
@active_frame = color("yellow bold")
@inactive_frame = color("blue dim")
@active_frame = Color("yellow bold")
@inactive_frame = Color("blue dim")
@colors = {}
for i=1,select('#',...)-1,2
col = select(i, ...)
@ -85,7 +86,7 @@ class Pad
w = 0
for chunk in *col do w = math.max(w, #chunk)
table.insert(@column_widths, w)
color_fn = select(i+1,...) or ((i)=>color())
color_fn = select(i+1,...) or ((i)=>Color())
_assert(type(color_fn) == 'function', "Invalid color function type: #{type color_fn}")
table.insert(@colors, color_fn)
@ -127,7 +128,7 @@ class Pad
chstr\set_str(x, " ", attr, @column_widths[c]-#chunk)
x += @column_widths[c]-#chunk
if c < #@columns
chstr\set_ch(x, C.ACS_VLINE, color("black bold"))
chstr\set_ch(x, C.ACS_VLINE, Color("black bold"))
x += 1
@_pad\mvaddchstr(i-1,0,chstr)
@dirty = true
@ -232,75 +233,161 @@ class NumberedPad extends Pad
col1 = select(1, ...)
fmt = "%#{#tostring(#col1)}d"
line_nums = [fmt\format(i) for i=1,#col1]
cols = {line_nums, ((i)=> i == @selected and color() or color("yellow")), ...}
cols = {line_nums, ((i)=> i == @selected and Color() or Color("yellow")), ...}
super @label, @y, @x, height, width, unpack(cols)
line_matcher = re.compile('lines<-{|(line "\n")* line|} line<-{[^\n]*}')
expansions = {}
KEY = {}
VALUE = {}
TOP_LOCATION = {}
locations = {}
Location = (old_loc, kind, key)->
unless locations[old_loc]
locations[old_loc] = {}
unless locations[old_loc][kind]
locations[old_loc][kind] = {}
unless locations[old_loc][kind][key]
locations[old_loc][kind][key] = {}
return locations[old_loc][kind][key]
expand = (kind, key, location)->
expansions[Location(location, kind, key)] = true
collapse = (kind, key, location)->
expansions[Location(location, kind, key)] = nil
is_key_expanded = (location, key)->
expansions[Location(location, KEY, key)]
is_value_expanded = (location, key)->
expansions[Location(location, VALUE, key)]
make_lines = (location, x, width)->
-- Return a list of {location=location, text1, color1, text2, color2, ...}
type_colors = {string:Color('blue on black'), number:Color('magenta'), boolean:Color('yellow'),
nil:Color('cyan'), table:Color('white bold'), function:Color('green'),
userdata:Color('cyan bold'), thread:Color('blue')}
setmetatable(type_colors, {__index: -> Color('red bold')})
switch type(x)
when 'string'
lines = {}
for line in *line_matcher\match(x)
wrapped = wrap_text(line, width-1)
for i,subline in ipairs(wrapped)
_line = {location:location}
if i > 1
table.insert(_line, C.ACS_BULLET)
table.insert(_line, Color('black bold altcharset'))
table.insert(_line, subline)
table.insert(_line, Color('blue on black bold'))
table.insert(lines, _line)
return lines
when 'table'
prepend = (line, ...)->
for i=1,select('#', ...)
table.insert(line, i, (select(i, ...)))
lines = {}
for k,v in pairs(x)
if is_key_expanded(location, k) and is_value_expanded(location, k)
table.insert lines, {location:Location(location,KEY,k), 'key/value:', Color('white')}
key_lines = make_lines(Location(location, KEY, k), k, width-2)
for i,key_line in ipairs(key_lines)
for j=2,#key_line,2
key_line[j] = key_line[j] | C.A_REVERSE
if i == 1
prepend(key_line, C.ACS_LLCORNER, Color!, C.ACS_TTEE, Color!)
else
prepend(key_line, ' ', Color!, C.ACS_VLINE, Color!)
table.insert(lines, key_line)
value_lines = make_lines(Location(location, VALUE, k), v, width-2)
for i,value_line in ipairs(value_lines)
if i == 1
prepend(value_line, ' ', Color!, C.ACS_LLCORNER, Color!)
else
prepend(value_line, ' ', Color!)
table.insert(lines, value_line)
elseif is_value_expanded(location, k)
k_str = type(k) == 'string' and k\gsub('\n','\\n') or repr(k,2)
if #k_str > width then k_str = k_str\sub(1,width-3)..'...'
table.insert(lines, {location:Location(location, KEY, k), k_str, type_colors[type(k)] | C.A_REVERSE})
v_lines = make_lines(Location(location, VALUE, k), v, width-1)
prepend(v_lines[1], C.ACS_LLCORNER, Color!)
for i=2,#v_lines
prepend(v_lines[i], ' ', Color!)
for v_line in *v_lines do table.insert(lines, v_line)
elseif is_key_expanded(location, k)
k_lines = make_lines(Location(location, KEY, k), k, width-1)
for k_line in *k_lines
for i=2,#k_line,2 do k_line[i] = k_lin[i] | C.A_REVERSE
for i=1,#k_lines-1
prepend(k_lines[i], ' ', Color!)
prepend(k_lines[#k_lines-1], C.ACS_ULCORNER, Color!)
for k_line in *k_lines do table.insert(lines, k_line)
v_str = type(v) == 'string' and v\gsub('\n','\\n') or repr(v,2)
if #v_str > width then v_str = v_str\sub(1,width-3)..'...'
table.insert(lines, {location:Location(location, VALUE, k), v_str, type_colors[type(v)]})
else
k_space = math.floor((width-3)/3)
k_str = type(k) == 'string' and k\gsub('\n','\\n') or repr(k,2)
if #k_str > k_space then k_str = k_str\sub(1,k_space-3)..'...'
v_space = (width-3)-#k_str
v_str = type(v) == 'string' and v\gsub('\n','\\n') or repr(v,2)
if #v_str > v_space then v_str = v_str\sub(1,v_space-3)..'...'
table.insert(lines, {
location:Location(location, VALUE, k),
k_str, type_colors[type(k)] | C.A_REVERSE,
' = ', Color!,
v_str, type_colors[type(v)]})
return lines
else
str = repr(x,2)
if #str > width
str = str\sub(1,width-3)..'...'
return {{:location, str, type_colors[type(x)]}}
class DataViewer extends Pad
new: (@data,@label,@y,@x,height,width)=>
@scroll_y, @scroll_x = 1, 1
@selected = nil
@active_frame = color("yellow bold")
@inactive_frame = color("blue dim")
@active_frame = Color("yellow bold")
@inactive_frame = Color("blue dim")
@expansions = {}
@full_refresh = ->
@chstrs, @chstr_actions = {}, {}
old_location = @selected and @chstr_locations and @chstr_locations[@selected]
@chstrs, @chstr_locations = {}, {}
line_matcher = re.compile('lines<-{|(line "\n")* line|} line<-{[^\n]*}')
W = width-3
add_line = (text, attr, indent, action)->
requested = line_matcher\match(text)
for line in *requested
wrapped = wrap_text(line, W-2*indent)
for i,subline in ipairs(wrapped)
chstr = C.new_chstr(W+1)
chstr\set_str(0, ' ', color!)
if i == 1
chstr\set_ch(1, C.ACS_CKBOARD, color('black bold'), 2*indent)
elseif indent > 0
chstr\set_ch(1, C.ACS_CKBOARD, color('black bold'), 2*indent-1)
chstr\set_ch(1+2*indent-1, C.ACS_BULLET, color('black bold'))
chstr\set_str(1+2*indent, subline, attr)
chstr\set_str(1+2*indent+#subline, ' ', attr, W-(2*indent+#subline))
table.insert @chstrs, chstr
table.insert @chstr_actions, action
add = (data, expansions, indent)->
if type(data) == 'table'
for k,v in pairs(data)
if expansions[k]
add_line "(-)", color('yellow bold'), indent, (->expansions[k] = nil)
add(k, expansions[k], indent+1)
add(v, expansions[k], indent+1)
else
add_line "[#{repr k, 1}] = #{repr v, 1}", color('yellow bold'), indent, (->expansions[k] = {})
return
_color = switch type(data)
when 'string'
color('default')
when 'number'
data = tostring(data)
color('magenta')
when 'boolean'
data = tostring(data)
data and color('green') or color('red')
when 'nil'
data = tostring(data)
color('cyan')
when 'function', 'userdata', 'thread'
data = repr(data)
-- TODO: better visualization/expansion here
color('blue bold')
lines = make_lines(TOP_LOCATION, @data, W)
for i,line in ipairs(lines)
chstr = C.new_chstr(W)
offset = 1
for j=1,#line-1,2
chunk, attrs = line[j], line[j+1]
if type(chunk) == 'number'
chstr\set_ch(offset, chunk, attrs)
offset += 1
else
error("Unsupported type: #{type data}")
add_line data, _color, indent, (->)
add(@data, @expansions, 0)
chstr\set_str(offset, chunk, attrs)
offset += #chunk
if offset < W
chstr\set_str(offset, ' ', attrs, W-offset)
table.insert @chstrs, chstr
table.insert @chstr_locations, line.location
@_height, @_width = #@chstrs, @width-2
@_pad\resize(@_height, @_width)
for i,chstr in ipairs(@chstrs)
@_pad\mvaddchstr(i-1,0,chstr)
@dirty = true
if old_location
for i,loc in ipairs(@chstr_locations)
if loc == old_location
@select(i)
break
@height, @width = height, width
@_frame = C.newwin(@height, @width, @y, @x)
@ -326,12 +413,12 @@ class DataViewer extends Pad
old_selected,@selected = @selected,i
if old_selected
if old_selected and @chstrs[old_selected]
@chstrs[old_selected]\set_str(0, ' ')
@_pad\mvaddchstr(old_selected-1,0,@chstrs[old_selected])
if @selected
@chstrs[@selected]\set_ch(0, C.ACS_RARROW, color('green bold'))
@chstrs[@selected]\set_ch(0, C.ACS_RARROW, Color('green bold'))
@_pad\mvaddchstr(@selected-1,0,@chstrs[@selected])
scrolloff = 3
@ -366,17 +453,17 @@ class DataViewer extends Pad
@scroll(-10,0)
when C.KEY_RIGHT, ("l")\byte!
@chstr_actions[@selected]!
expansions[@chstr_locations[@selected]] = true
@full_refresh!
when ("L")\byte!
@chstr_actions[@selected]!
expansions[@chstr_locations[@selected]] = true
@full_refresh!
when C.KEY_LEFT, ("h")\byte!
@chstr_actions[@selected]!
expansions[@chstr_locations[@selected]] = nil
@full_refresh!
when ("H")\byte!
@chstr_actions[@selected]!
expansions[@chstr_locations[@selected]] = nil
@full_refresh!
ok, to_lua = pcall -> require('moonscript.base').to_lua
@ -420,7 +507,7 @@ ldb = {
C.use_default_colors!
do -- Fullscreen flash
stdscr\wbkgd(color"yellow on red bold")
stdscr\wbkgd(Color"yellow on red bold")
stdscr\clear!
stdscr\refresh!
lines = wrap_text("ERROR!\n \n "..err_msg.."\n \npress any key...", math.floor(SCREEN_W/2))
@ -436,7 +523,7 @@ ldb = {
stdscr\getch!
stdscr\keypad!
stdscr\wbkgd(color!)
stdscr\wbkgd(Color!)
stdscr\clear!
stdscr\refresh!
@ -446,8 +533,8 @@ ldb = {
err_msg_lines = wrap_text(err_msg, SCREEN_W - 4)
for i,line in ipairs(err_msg_lines)
err_msg_lines[i] = (" ")\rep(2)..line
pads.err = Pad("Error Message", 0,0,AUTO,SCREEN_W, err_msg_lines, (i)=> color("red bold"))
pads.err._frame\attrset(color("red"))
pads.err = Pad("Error Message", 0,0,AUTO,SCREEN_W, err_msg_lines, (i)=> Color("red bold"))
pads.err._frame\attrset(Color("red"))
pads.err\refresh!
stack_locations = {}
@ -485,19 +572,19 @@ ldb = {
stack_h = math.max(#stack_names+2, math.floor(2/3*SCREEN_H))
stack_w = math.min(max_fn_name + 3 + max_filename, math.floor(1/3*SCREEN_W))
pads.stack = Pad "(C)allstack",pads.err.height,SCREEN_W-stack_w,stack_h,stack_w,
stack_names, ((i)=> (i == @selected) and color("black on green") or color("green bold")),
stack_locations, ((i)=> (i == @selected) and color("black on cyan") or color("cyan bold"))
stack_names, ((i)=> (i == @selected) and Color("black on green") or Color("green bold")),
stack_locations, ((i)=> (i == @selected) and Color("black on cyan") or Color("cyan bold"))
show_src = (filename, line_no)->
if pads.src
if pads.src.filename == filename
pads.src\select(line_no)
pads.src.colors[2] = (i)=>
return if i == line_no and i == @selected then color("yellow on red bold")
elseif i == line_no then color("yellow on red")
elseif err_lines["#{filename}:#{i}"] == true then color("red on black bold")
elseif i == @selected then color("reverse")
else color()
return if i == line_no and i == @selected then Color("yellow on red bold")
elseif i == line_no then Color("yellow on red")
elseif err_lines["#{filename}:#{i}"] == true then Color("red on black bold")
elseif i == @selected then Color("reverse")
else Color()
for line,_ in pairs(err_lines)
_filename, i = line\match("([^:]*):(%d*).*")
if _filename == filename and tonumber(i)
@ -513,11 +600,11 @@ ldb = {
table.insert src_lines, line
pads.src = NumberedPad "(S)ource Code", pads.err.height,0,
pads.stack.height,pads.stack.x, src_lines, (i)=>
return if i == line_no and i == @selected then color("yellow on red bold")
elseif i == line_no then color("yellow on red")
elseif err_lines["#{filename}:#{i}"] == true then color("red on black bold")
elseif i == @selected then color("reverse")
else color()
return if i == line_no and i == @selected then Color("yellow on red bold")
elseif i == line_no then Color("yellow on red")
elseif err_lines["#{filename}:#{i}"] == true then Color("red on black bold")
elseif i == @selected then Color("reverse")
else Color()
pads.src\select(line_no)
else
lines = {}
@ -525,7 +612,7 @@ ldb = {
s = "<no source code found>"
s = (" ")\rep(math.floor((pads.stack.x-2-#s)/2))..s
table.insert(lines, s)
pads.src = Pad "(S)ource Code", pads.err.height,0,pads.stack.height,pads.stack.x,lines, ->color("red")
pads.src = Pad "(S)ource Code", pads.err.height,0,pads.stack.height,pads.stack.x,lines, ->Color("red")
pads.src.filename = filename
local stack_env
@ -556,7 +643,7 @@ ldb = {
var_x = 0
--height = math.min(2+#var_names, SCREEN_H-pads.err.height-pads.stack.height)
height = SCREEN_H-(pads.err.height+pads.stack.height)
pads.vars = Pad "(V)ars", var_y,var_x,height,AUTO,var_names, ((i)=> i == @selected and color('reverse') or color())
pads.vars = Pad "(V)ars", var_y,var_x,height,AUTO,var_names, ((i)=> i == @selected and Color('reverse') or Color())
pads.vars.on_select = (var_index)=>
if var_index == nil then return
@ -570,7 +657,7 @@ ldb = {
switch type_str
when "string"
pads.values = Pad "(D)ata [string]",var_y,value_x,pads.vars.height,value_w,
wrap_text(value, value_w-2), (i)=>color()
wrap_text(value, value_w-2), (i)=>Color()
when "table"
value_str = repr(value, 3)
mt = getmetatable(value)
@ -592,22 +679,22 @@ ldb = {
value_str = table.concat ["#{key_repr k} = #{repr v,2}" for k,v in pairs(value)], "\n \n"
pads.values = Pad "(D)ata [#{type_str}]",var_y,value_x,pads.vars.height,value_w,
wrap_text(value_str, value_w-2), (i)=>color("cyan bold")
wrap_text(value_str, value_w-2), (i)=>Color("cyan bold")
when "function"
info = debug.getinfo(value, 'nS')
s = ("function '%s' defined at %s:%s")\format info.name or var_names[var_index],
info.short_src, info.linedefined
pads.values = Pad "(D)ata [#{type_str}]",var_y,value_x,pads.vars.height,value_w,
wrap_text(s, value_w-2), (i)=>color("green bold")
wrap_text(s, value_w-2), (i)=>Color("green bold")
when "number"
pads.values = Pad "(D)ata [#{type_str}]",var_y,value_x,pads.vars.height,value_w,
wrap_text(repr(value), value_w-2), (i)=>color("magenta bold")
wrap_text(repr(value), value_w-2), (i)=>Color("magenta bold")
when "boolean"
pads.values = Pad "(D)ata [#{type_str}]",var_y,value_x,pads.vars.height,value_w,
wrap_text(repr(value), value_w-2), (i)=> value and color("green bold") or color("red bold")
wrap_text(repr(value), value_w-2), (i)=> value and Color("green bold") or Color("red bold")
else
pads.values = Pad "(D)ata [#{type_str}]",var_y,value_x,pads.vars.height,value_w,
wrap_text(repr(value), value_w-2), (i)=>color()
wrap_text(repr(value), value_w-2), (i)=>Color()
]]
collectgarbage()
collectgarbage()