Added things library (improved version of lib/object.nom).

This commit is contained in:
Bruce Hill 2018-11-06 15:15:21 -08:00
parent c8ccbe5f42
commit 1f3660f393

141
lib/things.nom Normal file
View File

@ -0,0 +1,141 @@
%globals.METAMETHOD_MAP = {..}
"as text": "__tostring", "clean up": "__gc",
"+": "__add", "-": "__sub", "*": "__mul", "/": "__div",
"negative": "__unm", "//": "__idiv", "mod": "__mod", "^": "__pow",
"&": "__band", "|": "__bor", "~": "__bxor", "~": "__bnot",
"<<": "__bshl", ">>": "__bshr", "==": "__eq", "<": "__lt",
"<=": "__le", "set 1 =": "__newindex", "size": "__len",
"iterate": "__ipairs", "iterate all": "__pairs",
test:
a (Dog) is a thing:
that can (set up) by:
%its.barks or= 0
whose [bark, woof] all mean:
%barks = ("Bark!" for % in 1 to %its.barks)
return (%barks::joined with " ")
that can (get pissed off) by:
%its.barks += 1
(Dog).genus = "Canus"
%d = (a Dog with {barks:2})
assume (type of %d) == "Dog"
assume (%d is a "Dog")
assume %d.barks == 2
assume ((%d::bark) == "Bark! Bark!")
assume ((%d::woof) == "Bark! Bark!")
%d::get pissed off
assume (%d.barks == 3)
assume ((%d::bark) == "Bark! Bark! Bark!")
assume (%d.genus == "Canus")
assume ("\(%d.class)" == "Dog")
assume (%d.genus == "Canus")
assume (%d.barks == 3)
%d2 = (a Dog)
assume (%d2.barks == 0) or barf "Default initializer failed"
with {%d:a Dog with {barks:1}}:
assume ((%d::bark) == "Bark!")
a (Corgi) is a thing:
that can [set up, get pissed off] like a (Dog)
whose (sploot) means "splooted"
whose [bark, woof] all mean:
%barks = ("Yip!" for % in 1 to %its.barks)
return (%barks::joined with " ")
%corg = (a Corgi)
assume (%corg.barks == 0)
with {%d:a Corgi with {barks:1}}:
assume ((%d::sploot) == "splooted") or barf "subclass method failed"
assume ((%d::bark) == "Yip!") or barf "inheritance failed"
assume ((%d::woof) == "Yip!")
with {%d:a Dog with {barks:2}}:
assume ((%d::bark) == "Bark! Bark!")
[..]
that can %actions by %body,
whose %actions means %body, whose %actions all mean %body,
..all compile to:
unless (%actions.type == "List"): %actions = [%actions]
lua> "\
..local fn_name = \%actions[1].stub:as_lua_id()
local \%args = List{\(\%its), unpack(\%actions[1]:get_args())}
local lua = LuaCode(tree.source, "class.", fn_name, " = ", \(..)
what (%args -> %body) compiles to
..)
for i=2,#\%actions do
local alias = \%actions[i]
local alias_name = alias.stub:as_lua_id()
local \%alias_args = List{\(\%its), unpack(alias:get_args())}
lua:append("\\nclass.", alias_name, " = ")
if \%alias_args == \%args then
lua:append("class.", fn_name)
else
lua:append("function(")
lua:concat_append(table.map(\%alias_args, function(a) return nomsu:compile(a) end), ", ")
lua:append(")\\n return class.", fn_name, "(")
lua:concat_append(table.map(\%args, function(a) return nomsu:compile(a) end), ", ")
lua:append(")\\nend")
end
end
return lua"
[..]
that can %actions like a %class, that can %actions like an %class,
that has %actions like a %class, that has %actions like an %class,
..all compile to:
%lua = (Lua "")
%class_expr = (%class as lua expr)
%lines = []
for %a in %actions:
%lines::add "class.\(%a.stub::as lua id) = \%class_expr.\(%a.stub::as lua id)"
%lua::add %lines joined with "\n"
return %lua
(a %classname is a thing with %members %class_body) compiles to:
unless (%classname.type == "Action"):
compile error at %classname "Expected this to be an action, not a \(%classname.type)"
for % in %classname:
unless (% is text):
compile error at % "Class names should not have arguments."
return (..)
Lua "\
..do
local class = {name=\(quote %classname.stub)}
class.__type = class.name
setmetatable(class, {
__tostring=function(cls) return cls.name end,
__call=function(cls, inst)
inst = setmetatable(inst or {}, cls)
if inst.set_up then inst:set_up() end
return inst
end,
})
class.__members = \(%members as lua expr)
nomsu.environment[("a "..class.name):as_lua_id()] = class
nomsu.environment[("an "..class.name):as_lua_id()] = class
nomsu.environment[("a "..class.name.." with"):as_lua_id()] = class
nomsu.environment[("an "..class.name.." with"):as_lua_id()] = class
nomsu.environment[class.name:as_lua_id()] = function() return class end
class.__index = class
class.class = class
local dict_tostring = getmetatable(Dict{}).__tostring
class.__tostring = function(inst)
return inst.name..dict_tostring(inst)
end
\(%class_body as lua statements)
for stub,metamethod in pairs(globals.METAMETHOD_MAP) do
class[metamethod] = class[stub:as_lua_id()]
end
if class.__members then
assert(select(2, next(class.__members)) == true)
getmetatable(class).__newindex = function(its, key, value)
if class.__members[key] then
rawset(its, key, value)
else error("Not a valid member: "..tostring(key)) end
end
end
end"
(a %classname is a thing %class_body) parses as (a %classname is a thing with (nil) %class_body)