2018-11-26 16:28:06 -08:00
|
|
|
#!/usr/bin/env nomsu -V4.12.12.8
|
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
|
|
|
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:
|
2018-11-17 14:38:05 -08:00
|
|
|
[%it, %its] = [Dog, Dog]
|
|
|
|
(%its::set up) means:
|
2018-11-06 15:15:21 -08:00
|
|
|
%its.barks or= 0
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-11-17 14:38:05 -08:00
|
|
|
[%its::bark, %its::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
|
|
|
|
2018-11-19 17:37:37 -08:00
|
|
|
(%it::gets pissed off) means: %it.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-17 14:38:05 -08:00
|
|
|
assume "\%d" == "Dog {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!")
|
2018-11-17 14:38:05 -08:00
|
|
|
%d::gets pissed off
|
2018-11-06 15:15:21 -08:00
|
|
|
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:
|
2018-11-17 14:38:05 -08:00
|
|
|
[%it, %its] = [Corgi, Corgi]
|
|
|
|
%it [set up, gets pissed off] like a (Dog)
|
2018-11-19 17:37:37 -08:00
|
|
|
(%it::as text) means "Dogloaf \{: for %k = %v in %it: add %k = %v}"
|
2018-11-17 14:38:05 -08:00
|
|
|
(%its::sploot) means "sploooot"
|
|
|
|
[%its::bark, %its::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-17 14:38:05 -08:00
|
|
|
assume "\%corg" == "Dogloaf {barks: 0}"
|
2018-11-11 15:50:46 -08:00
|
|
|
with {%d: a Corgi with {barks: 1}}:
|
2018-11-17 14:38:05 -08:00
|
|
|
assume ((%d::sploot) == "sploooot") or barf "subclass method failed"
|
2018-11-06 15:15:21 -08:00
|
|
|
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-19 17:37:37 -08:00
|
|
|
|
2018-11-17 14:38:05 -08:00
|
|
|
a (Vec) is a thing with {x, y}:
|
|
|
|
%its = (Vec)
|
|
|
|
(%its::+ %other) means (Vec {x: %its.x + %other.x, y: %its.y + %other.y})
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-11-19 17:37:37 -08:00
|
|
|
assume ((Vec {x: 1, y: 2}) + (Vec {x: 10, y: 10})) == (Vec {x: 11, y: 12})
|
|
|
|
assume (((Vec {x: 1, y: 2}) + (Vec {x: 10, y: 10})) != (Vec {x: 0, y: 0}))
|
|
|
|
|
2018-11-06 15:15:21 -08:00
|
|
|
[..]
|
2018-11-17 14:38:05 -08:00
|
|
|
%it can %actions like a %class, %it can %actions like an %class
|
|
|
|
%it has %actions like a %class, %it has %actions like an %class
|
|
|
|
%it %actions like a %class, %it %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:
|
2018-11-19 17:37:37 -08:00
|
|
|
%lines::add "\
|
|
|
|
..\(%it as lua expr).\(%a.stub::as lua id) = \%class_expr.\(%a.stub::as lua id)"
|
2018-11-06 15:15:21 -08:00
|
|
|
%lua::add %lines joined with "\n"
|
|
|
|
return %lua
|
|
|
|
|
2018-11-17 14:38:05 -08:00
|
|
|
%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"
|
|
|
|
|
|
|
|
((% as text like a dict)'s meaning) = ({}'s metatable).__tostring
|
2018-11-19 17:37:37 -08:00
|
|
|
externally (a class named %classname with %members ((initialize %it)'s meaning)) \
|
2018-11-17 14:38:05 -08:00
|
|
|
..means:
|
|
|
|
%class = {__type: %classname}
|
|
|
|
%class.__index = %class
|
|
|
|
%class.class = %class
|
|
|
|
%class.__tostring = (% -> "\(%.__type) \(% as text like a dict)")
|
|
|
|
%class.__eq = ({}'s metatable).__eq
|
|
|
|
%class.__len = ({}'s metatable).__len
|
|
|
|
if %members:
|
|
|
|
%class.__members = %members
|
|
|
|
%class.__newindex = (..)
|
|
|
|
for (%its %key = %value):
|
|
|
|
if %members.%key:
|
|
|
|
rawset %its %key %value
|
|
|
|
..else:
|
|
|
|
barf "Cannot set \%key, it's not one of the allowed member fields."
|
2018-11-19 17:37:37 -08:00
|
|
|
|
2018-11-17 14:38:05 -08:00
|
|
|
set %class's metatable to {..}
|
2018-11-19 17:37:37 -08:00
|
|
|
__tostring: %class -> %class.__type
|
2018-11-17 14:38:05 -08:00
|
|
|
__call: for (%class with %initial_values):
|
|
|
|
if (%initial_values == (nil)): return %class
|
|
|
|
set %initial_values's metatable to %class
|
2018-11-19 17:37:37 -08:00
|
|
|
if %initial_values.set_up:
|
|
|
|
%initial_values::set up
|
2018-11-17 14:38:05 -08:00
|
|
|
return %initial_values
|
2018-11-19 17:37:37 -08:00
|
|
|
|
2018-11-17 14:38:05 -08:00
|
|
|
if ((initialize)'s meaning):
|
|
|
|
initialize %class
|
|
|
|
for %stub = %metamethod in %METAMETHOD_MAP:
|
|
|
|
if %class.(%stub::as lua id):
|
|
|
|
%class.%metamethod = %class.(%stub::as lua id)
|
2018-11-19 17:37:37 -08:00
|
|
|
|
2018-11-17 14:38:05 -08:00
|
|
|
return %class
|
|
|
|
|
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-17 14:38:05 -08:00
|
|
|
%class_id = (%classname.stub::as lua id)
|
|
|
|
if %class_body:
|
|
|
|
%body_lua = (%class_body as lua)
|
|
|
|
%body_lua::remove free vars [%class_id]
|
|
|
|
%body_lua::declare locals
|
2018-11-19 17:37:37 -08:00
|
|
|
|
2018-11-06 15:15:21 -08:00
|
|
|
return (..)
|
|
|
|
Lua "\
|
2018-11-17 14:38:05 -08:00
|
|
|
..\%class_id = a_class_named_1_with(\(quote %classname.stub), \(%members as lua)\(..)
|
|
|
|
(Lua ", function(\%class_id)\n \%body_lua\nend") if %class_body else ""
|
|
|
|
..)
|
|
|
|
a_\%class_id = function(initial_values) return \(%classname.stub::as lua id)(initial_values or {}) end
|
|
|
|
an_\%class_id, a_\(%class_id)_with, an_\(%class_id)_with = \
|
|
|
|
..a_\%class_id, a_\%class_id, a_\%class_id"
|
2018-11-06 15:15:21 -08:00
|
|
|
|
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
|