2019-01-15 15:53:31 -08:00
|
|
|
#!/usr/bin/env nomsu -V6.15.13.8
|
2018-11-11 15:37:57 -08:00
|
|
|
#
|
|
|
|
A library for simple object oriented programming.
|
2019-01-16 16:38:15 -08:00
|
|
|
|
|
|
|
use "core/metaprogramming"
|
|
|
|
use "core/operators"
|
|
|
|
use "core/control_flow"
|
|
|
|
use "core/collections"
|
|
|
|
use "core/errors"
|
|
|
|
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
|
2018-11-06 15:15:21 -08:00
|
|
|
test:
|
2019-01-16 21:33:02 -08:00
|
|
|
(an Empty) is (a thing) with []
|
|
|
|
(a Buffer) is (a thing) with [$bits]:
|
|
|
|
$self.is_a_buffer = (yes)
|
|
|
|
($self, set up) means:
|
|
|
|
$bits or= []
|
|
|
|
($self, as text) means ($bits, joined)
|
|
|
|
[($self, add $bit), ($self, append $bit)] all mean: $bits, add $bit
|
|
|
|
assume $(a Buffer).is_a_buffer
|
2019-01-15 18:32:11 -08:00
|
|
|
$b = (a Buffer)
|
2019-01-16 21:33:02 -08:00
|
|
|
assume (type of $b) == "a Buffer"
|
|
|
|
assume ($b is "a Buffer")
|
2019-01-15 18:32:11 -08:00
|
|
|
assume $b.is_a_buffer
|
|
|
|
assume "\$b" == ""
|
|
|
|
assume ($b, as text) == ""
|
2019-01-16 21:33:02 -08:00
|
|
|
$b = (a Buffer {.bits = ["x"]})
|
2019-01-16 16:38:15 -08:00
|
|
|
$b,
|
2019-01-15 18:32:11 -08:00
|
|
|
add "y"
|
|
|
|
append "z"
|
|
|
|
assume "\$b" == "xyz"
|
2019-01-16 21:33:02 -08:00
|
|
|
assume $b == (a Buffer {.bits = ["x", "y", "z"]})
|
|
|
|
assume $b != (a Buffer {.bits = []})
|
|
|
|
(a Comma Buffer) is (a Buffer) with [$bits]:
|
|
|
|
($self, as text) means ($bits, joined with ",")
|
|
|
|
($self, number of commas) means ((#$bits) - 1)
|
2019-01-15 18:32:11 -08:00
|
|
|
$csv = (a Comma Buffer)
|
|
|
|
assume $csv.is_a_buffer
|
|
|
|
assume "\$csv" == ""
|
|
|
|
$csv, add "x"
|
|
|
|
$csv, add "y"
|
|
|
|
assume "\$csv" == "x,y"
|
|
|
|
assume ($csv, number of commas) == 1
|
2019-01-16 21:33:02 -08:00
|
|
|
(a Vec) is (a thing) with [$x, $y]:
|
|
|
|
($self, + $other) means (Vec ($x + $other.x) ($y + $other.y))
|
|
|
|
($self, length) means (sqrt ($x * $x + $y * $y))
|
|
|
|
(Vec $x $y) means (a Vec {.x = $x, .y = $y})
|
|
|
|
assume ((Vec 1 2) + (Vec 10 10)) == (Vec 11 12)
|
|
|
|
assume (((Vec 1 2) + (Vec 10 10)) != (Vec 0 0))
|
|
|
|
assume (Vec 3 4, length) == 5
|
2018-11-19 17:37:37 -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
|
|
|
|
2019-01-01 17:15:51 -08:00
|
|
|
$($ as text like a dict) = ({}'s metatable).__tostring
|
2019-01-15 15:53:31 -08:00
|
|
|
external:
|
2019-01-16 21:33:02 -08:00
|
|
|
($parent class named $classname $(initialize $)) means:
|
2019-01-15 15:53:31 -08:00
|
|
|
$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
|
|
|
|
|
|
|
|
set $class's metatable to {
|
2019-01-16 16:38:15 -08:00
|
|
|
.__index = $parent, .__tostring = ($class -> $class.__type)
|
2019-01-15 15:53:31 -08:00
|
|
|
.__call =
|
|
|
|
for ($class with $initial_values):
|
2019-01-16 21:33:02 -08:00
|
|
|
$initial_values or= {}
|
2019-01-15 15:53:31 -08:00
|
|
|
set $initial_values's metatable to $class
|
|
|
|
if $initial_values.set_up:
|
|
|
|
$initial_values, set up
|
|
|
|
return $initial_values
|
|
|
|
}
|
|
|
|
|
|
|
|
if $(initialize $):
|
|
|
|
initialize $class
|
|
|
|
for $stub = $metamethod in $METAMETHOD_MAP:
|
|
|
|
if $class.($stub, as lua id):
|
|
|
|
$class.$metamethod = $class.($stub, as lua id)
|
|
|
|
|
|
|
|
return $class
|
2019-01-16 21:33:02 -08:00
|
|
|
|
|
|
|
$(a thing) = ((nil) class named "thing")
|
|
|
|
|
|
|
|
($classname is $parent with $vars $class_body) compiles to:
|
|
|
|
unless ($vars.type == "List"):
|
|
|
|
at $vars fail ("
|
|
|
|
Compile error: This is not a list of variables.
|
|
|
|
")
|
2019-01-15 15:53:31 -08:00
|
|
|
$class_id = ($classname.stub, as lua id)
|
2019-01-16 21:33:02 -08:00
|
|
|
if $class_body:
|
|
|
|
$class_body =
|
|
|
|
$class_body with vars {
|
|
|
|
: for $v in $vars:
|
|
|
|
add ($v as lua expr, text) =
|
2019-01-18 14:22:17 -08:00
|
|
|
"IndexChain" tree with ("Var" tree with "self")
|
|
|
|
"Index" tree with ("Text" tree with $v.1)
|
2019-01-16 21:33:02 -08:00
|
|
|
}
|
2019-01-15 15:53:31 -08:00
|
|
|
$lua =
|
|
|
|
Lua ("
|
2019-01-16 21:33:02 -08:00
|
|
|
\$class_id = _1_class_named(\($parent as lua), \(quote $classname.stub)\(
|
2019-01-15 15:53:31 -08:00
|
|
|
(
|
|
|
|
Lua ("
|
|
|
|
, function(\$class_id)
|
2019-01-16 21:33:02 -08:00
|
|
|
local self = \$class_id
|
2019-01-15 18:10:56 -08:00
|
|
|
\($class_body as lua)
|
2019-01-15 15:53:31 -08:00
|
|
|
end
|
|
|
|
")
|
|
|
|
) if $class_body else ""
|
|
|
|
))
|
|
|
|
")
|
2019-01-16 21:33:02 -08:00
|
|
|
$lua, add free vars [$class_id]
|
2019-01-15 15:53:31 -08:00
|
|
|
return $lua
|