2018-12-30 19:04:34 -08:00
|
|
|
#!/usr/bin/env nomsu -V6.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-12-14 20:21:03 -08:00
|
|
|
[$it, $its] = [Dog, Dog]
|
2018-12-30 19:04:34 -08:00
|
|
|
($its, set up) means:
|
2018-12-14 20:21:03 -08:00
|
|
|
$its.barks or= 0
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-12-30 19:04:34 -08:00
|
|
|
[($its, bark), ($its, woof)] all mean:
|
2018-12-14 20:21:03 -08:00
|
|
|
$barks = [: for $ in 1 to $its.barks: add "Bark!"]
|
2018-12-30 19:04:34 -08:00
|
|
|
return ($barks, joined with " ")
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-12-30 19:04:34 -08:00
|
|
|
($it, gets pissed off) means: $it.barks += 1
|
2018-11-06 15:15:21 -08:00
|
|
|
(Dog).genus = "Canus"
|
2018-12-30 19:04:34 -08:00
|
|
|
$d = (a Dog with {.barks = 2})
|
2018-12-14 20:21:03 -08:00
|
|
|
assume "\$d" == "Dog {barks: 2}"
|
|
|
|
assume (type of $d) == "Dog"
|
|
|
|
assume ($d is a "Dog")
|
|
|
|
assume $d.barks == 2
|
2018-12-30 19:04:34 -08:00
|
|
|
assume (($d, bark) == "Bark! Bark!")
|
|
|
|
assume (($d, woof) == "Bark! Bark!")
|
|
|
|
$d, gets pissed off
|
2018-12-14 20:21:03 -08:00
|
|
|
assume ($d.barks == 3)
|
2018-12-30 19:04:34 -08:00
|
|
|
assume (($d, bark) == "Bark! Bark! Bark!")
|
2018-12-14 20:21:03 -08:00
|
|
|
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-12-30 19:04:34 -08:00
|
|
|
with [$d = (a Dog with {.barks = 1})]:
|
|
|
|
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-12-14 20:21:03 -08:00
|
|
|
[$it, $its] = [Corgi, Corgi]
|
|
|
|
$it [set up, gets pissed off] like a (Dog)
|
2018-12-30 19:04:34 -08:00
|
|
|
($it, as text) means "Dogloaf \{: for $k = $v in $it: add $k = $v}"
|
|
|
|
($its, sploot) means "sploooot"
|
|
|
|
[($its, bark), ($its, woof)] all mean:
|
2018-12-14 20:21:03 -08:00
|
|
|
$barks = [: for $ in 1 to $its.barks: add "Yip!"]
|
2018-12-30 19:04:34 -08:00
|
|
|
return ($barks, joined with " ")
|
2018-11-06 15:15:21 -08:00
|
|
|
|
2018-12-14 20:21:03 -08:00
|
|
|
$corg = (a Corgi)
|
|
|
|
assume ($corg.barks == 0)
|
|
|
|
assume "\$corg" == "Dogloaf {barks: 0}"
|
2018-12-30 19:04:34 -08:00
|
|
|
with [$d = (a Corgi with {.barks = 1})]:
|
|
|
|
assume (($d, sploot) == "sploooot") or barf "subclass method failed"
|
|
|
|
assume (($d, bark) == "Yip!") or barf "inheritance failed"
|
|
|
|
assume (($d, woof) == "Yip!")
|
2018-11-06 15:15:21 -08:00
|
|
|
|
2018-12-30 19:04:34 -08:00
|
|
|
with [$d = (a Dog with {.barks = 2})]:
|
|
|
|
assume (($d, bark) == "Bark! Bark!")
|
2018-11-19 17:37:37 -08:00
|
|
|
|
2018-12-30 19:04:34 -08:00
|
|
|
a (Vec) is a thing with {.x, .y}:
|
2018-12-14 20:21:03 -08:00
|
|
|
$its = (Vec)
|
2018-12-30 19:04:34 -08:00
|
|
|
($its, + $other) means (Vec {.x = ($its.x + $other.x), .y = ($its.y + $other.y)})
|
2018-11-11 15:50:46 -08:00
|
|
|
|
2018-12-30 19:04:34 -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-19 17:37:37 -08:00
|
|
|
|
2018-12-30 19:04:34 -08:00
|
|
|
[
|
2018-12-14 20:21:03 -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-12-30 19:04:34 -08:00
|
|
|
] all compile to:
|
2018-12-14 20:21:03 -08:00
|
|
|
$lua = (Lua "")
|
|
|
|
$class_expr = ($class as lua expr)
|
|
|
|
$lines = []
|
|
|
|
for $a in $actions:
|
2018-12-30 19:04:34 -08:00
|
|
|
$lines,
|
|
|
|
add "\($it as lua expr).\($a.stub, as lua id) = \$class_expr.\($a.stub, as lua id)"
|
|
|
|
$lua, add $lines joined with "\n"
|
2018-12-14 20:21:03 -08:00
|
|
|
return $lua
|
2018-11-06 15:15:21 -08:00
|
|
|
|
2018-12-30 19:04:34 -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"
|
|
|
|
}
|
2018-11-17 14:38:05 -08:00
|
|
|
|
2018-12-14 20:21:03 -08:00
|
|
|
(($ as text like a dict)'s meaning) = ({}'s metatable).__tostring
|
2018-12-30 19:04:34 -08:00
|
|
|
externally (a class named $classname with $members ((initialize $it)'s meaning))
|
2018-11-17 14:38:05 -08:00
|
|
|
..means:
|
2018-12-30 19:04:34 -08:00
|
|
|
$class = {.__type = $classname}
|
2018-12-14 20:21:03 -08:00
|
|
|
$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
|
2018-12-30 19:04:34 -08:00
|
|
|
$class.__newindex =
|
2018-12-14 20:21:03 -08:00
|
|
|
for ($its $key = $value):
|
|
|
|
if $members.$key:
|
|
|
|
rawset $its $key $value
|
2018-11-17 14:38:05 -08:00
|
|
|
..else:
|
2018-12-14 20:21:03 -08:00
|
|
|
barf "Cannot set \$key, it's not one of the allowed member fields."
|
2018-11-19 17:37:37 -08:00
|
|
|
|
2018-12-30 19:04:34 -08:00
|
|
|
set $class's metatable to {
|
|
|
|
.__tostring = ($class -> $class.__type), .__call = (
|
|
|
|
for ($class with $initial_values):
|
|
|
|
if ($initial_values == (nil)): return $class
|
|
|
|
set $initial_values's metatable to $class
|
|
|
|
if $initial_values.set_up:
|
|
|
|
$initial_values, set up
|
|
|
|
return $initial_values
|
|
|
|
)
|
|
|
|
}
|
2018-11-19 17:37:37 -08:00
|
|
|
|
2018-11-17 14:38:05 -08:00
|
|
|
if ((initialize)'s meaning):
|
2018-12-14 20:21:03 -08:00
|
|
|
initialize $class
|
|
|
|
for $stub = $metamethod in $METAMETHOD_MAP:
|
2018-12-30 19:04:34 -08:00
|
|
|
if $class.($stub, as lua id):
|
|
|
|
$class.$metamethod = $class.($stub, as lua id)
|
2018-11-19 17:37:37 -08:00
|
|
|
|
2018-12-14 20:21:03 -08:00
|
|
|
return $class
|
2018-11-17 14:38:05 -08:00
|
|
|
|
2018-12-30 19:04:34 -08:00
|
|
|
[
|
2018-12-14 20:21:03 -08:00
|
|
|
a $classname is a thing with $members $class_body
|
|
|
|
an $classname is a thing with $members $class_body
|
2018-12-30 19:04:34 -08:00
|
|
|
] all compile to:
|
|
|
|
$class_id = ($classname.stub, as lua id)
|
2018-12-14 20:21:03 -08:00
|
|
|
if $class_body:
|
|
|
|
$body_lua = ($class_body as lua)
|
2018-12-30 19:04:34 -08:00
|
|
|
$body_lua, remove free vars [$class_id]
|
|
|
|
$body_lua, declare locals
|
2018-11-19 17:37:37 -08:00
|
|
|
|
2018-12-30 19:04:34 -08:00
|
|
|
return
|
|
|
|
Lua ("
|
2018-12-14 20:21:03 -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 "")\
|
2018-11-17 14:38:05 -08:00
|
|
|
..)
|
2018-12-30 19:04:34 -08:00
|
|
|
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-12-30 19:04:34 -08:00
|
|
|
[a $classname is a thing $class_body, an $classname is a thing] all parse as
|
2018-12-14 20:21:03 -08:00
|
|
|
a $classname is a thing with (nil) $class_body
|