#!/usr/bin/env nomsu -V2.5.5.5 # This file contains the implementation of an Object-Oriented programming system. test: object "Dog": (class Dog).genus = "Canus" method [initialize %] (%.barks or= 0) method [bark, woof]: %barks = ("Bark!" for % in 1 to (me).barks) return (%barks joined with " ") method [get pissed off] ((me).barks += 1) %d = (new Dog {barks:2}) as %d: assume ((me) == %d) assume ((me).barks == 2) assume ((bark) == "Bark! Bark!") assume ((woof) == "Bark! Bark!") get pissed off assume ((me).barks == 3) assume ((bark) == "Bark! Bark! Bark!") assume ((me).genus == "Canus") assume ("\(%d.class)" == "Dog") assume (%d.genus == "Canus") assume (%d.barks == 3) as (new Dog) (assume ((me).barks == 0) or barf "Default initializer failed") as (new Dog {barks:1}) (assume ((bark) == "Bark!")) action [foo] (as (new Dog {barks:23}) (return (me).barks)) assume ((foo) == 23) or barf "Oops, \(foo) != 23" as (new Dog {barks:101}): try (as (new Dog {barks:8}) (barf)) and if it succeeds (barf) assume ((me).barks == 101) or barf ".." Error in nested 'as % %' failed to properly reset 'self' object "Corgi" extends (class Dog): method [sploot] "splooted" method [bark, woof]: %barks = ("Yip!" for % in 1 to (me).barks) return (%barks joined with " ") %corg = (new Corgi) assume (%corg.barks == 0) as (new Corgi {barks:1}): assume ((sploot) == "splooted") or barf "subclass method failed" assume ((bark) == "Yip!") or barf "inheritance failed" assume ((woof) == "Yip!") as (new Dog {barks:2}): assume ((bark) == "Bark! Bark!") compile [@, me] to (Lua value "self") compile [method %actions %body] to: %lua = (\(local action \[%actions.1] %body) as lua) declare locals in %lua for % in %actions: to %lua write "\n\(\%class as lua id).\(% as lua id) = \(%actions.1 as lua id)" %lua = (..) Lua ".." do -- Method: \(%actions.(1).stub) \%lua end return %lua ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ parse [as %instance %body] as (..) result of: %old_self = (me) (me) = %instance try %body and if it barfs %msg: (me) = %old_self barf %msg ..or if it succeeds: (me) = %old_self compile [object %classname extends %parent %class_body] to: %class = (\%class as lua id) return (..) Lua ".." do local \%class = {name=\(%classname as lua expr)} setmetatable(\%class, { __index=\(%parent as lua expr), __tostring=function(cls) return cls.name end, __call=function(cls, inst) inst = setmetatable(inst or {}, cls) if cls.A_initialize_1 then cls.A_initialize_1(inst) end return inst end, }) _ENV["A"..string.as_lua_id("new "..\%class.name)] = \%class _ENV["A"..string.as_lua_id("new "..\%class.name.." 1")] = \%class _ENV["A"..string.as_lua_id("class "..\%class.name)] = function() return \%class end \%class.__index = \%class \%class.class = \%class \(%class_body as lua statements) \%class.__tostring = \%class["A"..string.as_lua_id("as text")] or function(inst) return inst.name..getmetatable(dict{}).__tostring(inst) end end ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ parse [object %classname %class_body] as (..) object %classname extends (nil) %class_body