2018-11-11 15:50:46 -08:00
|
|
|
#!/usr/bin/env nomsu -V4.10.12.7
|
2018-11-11 15:37:57 -08:00
|
|
|
#
|
|
|
|
A library for simple object oriented programming.
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-11-06 15:15:21 -08:00
|
|
|
%globals.METAMETHOD_MAP = {..}
|
2018-11-11 15:50:46 -08:00
|
|
|
"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"
|
2018-11-06 15:15:21 -08:00
|
|
|
|
|
|
|
test:
|
2018-11-12 14:25:11 -08:00
|
|
|
an (Empty) is a thing
|
2018-11-06 15:15:21 -08:00
|
|
|
a (Dog) is a thing:
|
|
|
|
that can (set up) by:
|
|
|
|
%its.barks or= 0
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-11-06 15:15:21 -08:00
|
|
|
whose [bark, woof] all mean:
|
2018-11-11 15:50:46 -08:00
|
|
|
%barks = [: for % in 1 to %its.barks: add "Bark!"]
|
2018-11-06 15:15:21 -08:00
|
|
|
return (%barks::joined with " ")
|
2018-11-11 15:50:46 -08:00
|
|
|
|
|
|
|
that can (get pissed off) by: %its.barks += 1
|
2018-11-06 15:15:21 -08:00
|
|
|
(Dog).genus = "Canus"
|
2018-11-11 15:50:46 -08:00
|
|
|
%d = (a Dog with {barks: 2})
|
2018-11-06 15:15:21 -08:00
|
|
|
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"
|
2018-11-11 15:50:46 -08:00
|
|
|
with {%d: a Dog with {barks: 1}}:
|
2018-11-06 15:15:21 -08:00
|
|
|
assume ((%d::bark) == "Bark!")
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-11-06 15:15:21 -08:00
|
|
|
a (Corgi) is a thing:
|
|
|
|
that can [set up, get pissed off] like a (Dog)
|
|
|
|
whose (sploot) means "splooted"
|
|
|
|
whose [bark, woof] all mean:
|
2018-11-11 15:50:46 -08:00
|
|
|
%barks = [: for % in 1 to %its.barks: add "Yip!"]
|
2018-11-06 15:15:21 -08:00
|
|
|
return (%barks::joined with " ")
|
|
|
|
|
|
|
|
%corg = (a Corgi)
|
|
|
|
assume (%corg.barks == 0)
|
2018-11-11 15:50:46 -08:00
|
|
|
with {%d: a Corgi with {barks: 1}}:
|
2018-11-06 15:15:21 -08:00
|
|
|
assume ((%d::sploot) == "splooted") or barf "subclass method failed"
|
|
|
|
assume ((%d::bark) == "Yip!") or barf "inheritance failed"
|
|
|
|
assume ((%d::woof) == "Yip!")
|
|
|
|
|
2018-11-11 15:50:46 -08:00
|
|
|
with {%d: a Dog with {barks: 2}}:
|
2018-11-06 15:15:21 -08:00
|
|
|
assume ((%d::bark) == "Bark! Bark!")
|
|
|
|
|
|
|
|
[..]
|
2018-11-11 15:50:46 -08:00
|
|
|
that can %actions by %body, whose %actions means %body
|
|
|
|
whose %actions all mean %body
|
2018-11-06 15:15:21 -08:00
|
|
|
..all compile to:
|
2018-11-11 15:50:46 -08:00
|
|
|
unless (%actions.type == "List"):
|
|
|
|
%actions = [%actions]
|
|
|
|
|
2018-11-06 15:15:21 -08:00
|
|
|
lua> "\
|
|
|
|
..local fn_name = \%actions[1].stub:as_lua_id()
|
|
|
|
local \%args = List{\(\%its), unpack(\%actions[1]:get_args())}
|
2018-11-11 15:50:46 -08:00
|
|
|
local lua = LuaCode("class.", fn_name, " = ", \(what (%args -> %body) compiles to))
|
2018-11-06 15:15:21 -08:00
|
|
|
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(")
|
2018-11-08 15:23:22 -08:00
|
|
|
lua:concat_append(table.map(\%alias_args, function(a) return compile(a) end), ", ")
|
2018-11-06 15:15:21 -08:00
|
|
|
lua:append(")\\n return class.", fn_name, "(")
|
2018-11-08 15:23:22 -08:00
|
|
|
lua:concat_append(table.map(\%args, function(a) return compile(a) end), ", ")
|
2018-11-06 15:15:21 -08:00
|
|
|
lua:append(")\\nend")
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return lua"
|
|
|
|
|
|
|
|
[..]
|
2018-11-11 15:50:46 -08:00
|
|
|
that can %actions like a %class, that can %actions like an %class
|
|
|
|
that has %actions like a %class, that has %actions like an %class
|
2018-11-06 15:15:21 -08:00
|
|
|
..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
|
|
|
|
|
2018-11-12 14:25:11 -08:00
|
|
|
[..]
|
|
|
|
a %classname is a thing with %members %class_body
|
|
|
|
an %classname is a thing with %members %class_body
|
|
|
|
..all compile to:
|
2018-11-06 15:15:21 -08:00
|
|
|
unless (%classname.type == "Action"):
|
2018-11-11 15:50:46 -08:00
|
|
|
compile error at %classname "\
|
|
|
|
..Expected this to be an action, not a \%classname.type"
|
|
|
|
|
2018-11-06 15:15:21 -08:00
|
|
|
for % in %classname:
|
|
|
|
unless (% is text):
|
|
|
|
compile error at % "Class names should not have arguments."
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-11-06 15:15:21 -08:00
|
|
|
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)
|
2018-11-08 15:23:22 -08:00
|
|
|
_ENV[("a "..class.name):as_lua_id()] = class
|
|
|
|
_ENV[("an "..class.name):as_lua_id()] = class
|
|
|
|
_ENV[("a "..class.name.." with"):as_lua_id()] = class
|
|
|
|
_ENV[("an "..class.name.." with"):as_lua_id()] = class
|
|
|
|
_ENV[class.name:as_lua_id()] = function() return class end
|
2018-11-06 15:15:21 -08:00
|
|
|
class.__index = class
|
|
|
|
class.class = class
|
|
|
|
local dict_tostring = getmetatable(Dict{}).__tostring
|
|
|
|
class.__tostring = function(inst)
|
|
|
|
return inst.name..dict_tostring(inst)
|
|
|
|
end
|
2018-11-12 14:25:11 -08:00
|
|
|
\((%class_body as lua) if %class_body else "")
|
2018-11-06 15:15:21 -08:00
|
|
|
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"
|
|
|
|
|
2018-11-12 14:25:11 -08:00
|
|
|
[a %classname is a thing %class_body, an %classname is a thing] all parse as (..)
|
2018-11-11 15:50:46 -08:00
|
|
|
a %classname is a thing with (nil) %class_body
|