2018-09-14 19:17:09 -07:00
|
|
|
#!/usr/bin/env nomsu -V4.8.8.6
|
2018-06-15 00:17:01 -07:00
|
|
|
#
|
|
|
|
This file contains the implementation of an Object-Oriented programming system.
|
2018-01-18 01:49:13 -08:00
|
|
|
|
2018-10-29 13:00:08 -07:00
|
|
|
%globals.METAMETHOD_MAP = {..}
|
|
|
|
"as text": "__tostring", "clean up": "__gc",
|
|
|
|
"+ 1": "__add", "- 1": "__sub", "* 1": "__mul", "/ 1": "__div",
|
|
|
|
"-": "__unm", "// 1": "__idiv", "mod 1": "__mod", "^ 1": "__pow",
|
|
|
|
"& 1": "__band", "| 1": "__bor", "~ 1": "__bxor", "~": "__bnot",
|
|
|
|
"<< 1": "__bshl", ">> 1": "__bshr", "== 1": "__eq", "< 1": "__lt",
|
|
|
|
"<= 1": "__le", "set 1 = 2": "__newindex", "size": "__len",
|
|
|
|
"iterate": "__ipairs", "iterate all": "__pairs",
|
|
|
|
|
2018-07-22 16:30:49 -07:00
|
|
|
test:
|
2018-09-21 00:30:28 -07:00
|
|
|
object (Dog):
|
2018-08-28 15:08:00 -07:00
|
|
|
(Dog).genus = "Canus"
|
2018-08-28 15:34:45 -07:00
|
|
|
my action [set up]: %me.barks or= 0
|
2018-08-28 15:08:00 -07:00
|
|
|
my action [bark, woof]:
|
|
|
|
%barks = ("Bark!" for % in 1 to %me.barks)
|
2018-09-10 16:26:08 -07:00
|
|
|
return (%barks::joined with " ")
|
2018-08-28 15:34:45 -07:00
|
|
|
|
|
|
|
my action [get pissed off]: %me.barks += 1
|
2018-07-23 14:40:20 -07:00
|
|
|
|
2018-09-21 00:30:28 -07:00
|
|
|
%d = (Dog {barks:2})
|
|
|
|
assume (type of %d) == "Dog"
|
|
|
|
assume (%d is a "Dog")
|
|
|
|
assume %d.barks == 2
|
2018-08-28 15:08:00 -07:00
|
|
|
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")
|
2018-07-22 16:30:49 -07:00
|
|
|
assume ("\(%d.class)" == "Dog")
|
|
|
|
assume (%d.genus == "Canus")
|
|
|
|
assume (%d.barks == 3)
|
2018-09-21 00:30:28 -07:00
|
|
|
%d2 = (Dog {})
|
2018-08-28 15:08:00 -07:00
|
|
|
assume (%d2.barks == 0) or barf "Default initializer failed"
|
2018-09-21 00:30:28 -07:00
|
|
|
with {%d:Dog {barks:1}}:
|
2018-08-28 15:34:45 -07:00
|
|
|
assume ((%d::bark) == "Bark!")
|
2018-09-21 00:30:28 -07:00
|
|
|
object (Corgi) extends (Dog):
|
2018-08-28 15:08:00 -07:00
|
|
|
my action [sploot] "splooted"
|
|
|
|
my action [bark, woof]:
|
|
|
|
%barks = ("Yip!" for % in 1 to %me.barks)
|
2018-09-10 16:26:08 -07:00
|
|
|
return (%barks::joined with " ")
|
2018-08-28 15:34:45 -07:00
|
|
|
|
2018-09-21 00:30:28 -07:00
|
|
|
%corg = (Corgi {})
|
2018-07-22 16:30:49 -07:00
|
|
|
assume (%corg.barks == 0)
|
2018-09-21 00:30:28 -07:00
|
|
|
with {%d:Corgi {barks:1}}:
|
2018-08-28 15:08:00 -07:00
|
|
|
assume ((%d::sploot) == "splooted") or barf "subclass method failed"
|
|
|
|
assume ((%d::bark) == "Yip!") or barf "inheritance failed"
|
|
|
|
assume ((%d::woof) == "Yip!")
|
2018-08-27 13:38:58 -07:00
|
|
|
|
2018-09-21 00:30:28 -07:00
|
|
|
with {%d:Dog {barks:2}}:
|
2018-08-28 15:08:00 -07:00
|
|
|
assume ((%d::bark) == "Bark! Bark!")
|
|
|
|
compile [my action %actions %body] to:
|
2018-09-14 19:17:09 -07:00
|
|
|
lua> "\
|
|
|
|
..local fn_name = \%actions[1].stub:as_lua_id()
|
2018-08-28 15:08:00 -07:00
|
|
|
local \%args = table.map(\%actions[1]:get_args(), function(a) return tostring(nomsu:compile(a)) end)
|
2018-09-18 19:48:58 -07:00
|
|
|
table.insert(\%args, 1, \(\%me as lua id))
|
2018-08-28 15:34:45 -07:00
|
|
|
local lua = LuaCode(tree.source, "class.", fn_name, " = ", \(..)
|
|
|
|
compile as (%args -> %body)
|
|
|
|
..)
|
2018-08-28 15:08:00 -07:00
|
|
|
for i=2,#\%actions do
|
|
|
|
local alias = \%actions[i]
|
2018-09-10 16:36:51 -07:00
|
|
|
local alias_name = alias.stub:as_lua_id()
|
2018-08-28 15:08:00 -07:00
|
|
|
local \%alias_args = table.map(alias:get_args(), function(a) return tostring(nomsu:compile(a)) end)
|
2018-09-18 19:48:58 -07:00
|
|
|
table.insert(\%alias_args, 1, \(\%me as lua id))
|
2018-08-28 15:08:00 -07:00
|
|
|
lua:append("\\nclass.", alias_name, " = ")
|
|
|
|
if utils.equivalent(\%args, \%alias_args) then
|
|
|
|
lua:append("class.", fn_name)
|
|
|
|
else
|
|
|
|
lua:append("function(")
|
|
|
|
lua:concat_append(\%alias_args, ", ")
|
|
|
|
lua:append(")\\n return class.", fn_name, "(")
|
|
|
|
lua:concat_append(\%args, ", ")
|
|
|
|
lua:append(")\\nend")
|
2018-07-20 20:27:15 -07:00
|
|
|
end
|
2018-08-28 15:08:00 -07:00
|
|
|
end
|
2018-09-14 19:17:09 -07:00
|
|
|
return lua"
|
2018-06-18 15:44:29 -07:00
|
|
|
|
2018-08-27 13:38:58 -07:00
|
|
|
compile [object %classname extends %parent %class_body] to:
|
2018-09-21 00:30:28 -07:00
|
|
|
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."
|
2018-08-27 13:38:58 -07:00
|
|
|
return (..)
|
2018-09-14 19:17:09 -07:00
|
|
|
Lua "\
|
|
|
|
..do
|
2018-09-21 00:30:28 -07:00
|
|
|
local class = {name=\(quote %classname.stub)}
|
2018-09-26 13:05:28 -07:00
|
|
|
class.__type = class.name
|
2018-08-28 15:08:00 -07:00
|
|
|
setmetatable(class, {
|
2018-08-27 13:38:58 -07:00
|
|
|
__index=\(%parent as lua expr),
|
|
|
|
__tostring=function(cls) return cls.name end,
|
|
|
|
__call=function(cls, inst)
|
|
|
|
inst = setmetatable(inst or {}, cls)
|
2018-10-29 13:00:08 -07:00
|
|
|
if inst.set_up then inst:set_up() end
|
2018-08-27 13:38:58 -07:00
|
|
|
return inst
|
|
|
|
end,
|
|
|
|
})
|
2018-09-21 00:30:28 -07:00
|
|
|
nomsu.environment[(class.name.." 1"):as_lua_id()] = class
|
2018-09-16 16:57:14 -07:00
|
|
|
nomsu.environment[class.name:as_lua_id()] = function() return class end
|
2018-08-28 15:08:00 -07:00
|
|
|
class.__index = class
|
|
|
|
class.class = class
|
|
|
|
class.__tostring = function(inst)
|
2018-09-06 12:46:39 -07:00
|
|
|
return inst.name..getmetatable(_Dict{}).__tostring(inst)
|
2018-08-28 15:08:00 -07:00
|
|
|
end
|
2018-08-27 13:38:58 -07:00
|
|
|
\(%class_body as lua statements)
|
2018-10-29 13:00:08 -07:00
|
|
|
for stub,metamethod in pairs(globals.METAMETHOD_MAP) do
|
2018-09-10 16:36:51 -07:00
|
|
|
class[metamethod] = class[stub:as_lua_id()]
|
2018-08-27 13:38:58 -07:00
|
|
|
end
|
2018-09-14 19:17:09 -07:00
|
|
|
end"
|
2018-06-18 15:44:29 -07:00
|
|
|
|
2018-07-17 23:37:20 -07:00
|
|
|
parse [object %classname %class_body] as (..)
|
2018-07-22 15:01:05 -07:00
|
|
|
object %classname extends (nil) %class_body
|
2018-09-18 19:48:58 -07:00
|
|
|
|