mirror of
https://github.com/kikito/middleclass.git
synced 2024-11-25 02:44:20 +00:00
another fix to the readme. I forgot to remove the specs dir, doing so now
This commit is contained in:
parent
10c330a161
commit
015d905a14
@ -19,8 +19,8 @@ h1. Features
|
||||
* @Class.superclass@ returns its super class
|
||||
* Subclassing:
|
||||
** @class(name)@ creates a subclass of @Object@
|
||||
** @class(name, Superclass)@ creates a subclass of the class @Class@
|
||||
** @Class:subclass(name)@ also creates a subclass of the class @Class@
|
||||
** @class(name, Superclass)@ creates a subclass of the class @SuperClass@
|
||||
** @SuperClass:subclass(name)@ also creates a subclass of the class @SuperClass@
|
||||
* Instantiation:
|
||||
** Classes can define an @initialize@ method for initializing new instances. They can accept an arbitrary number of params.
|
||||
** Instances are created by doing @Class:new(params)@ or also @Class(params)@
|
||||
|
@ -1,280 +0,0 @@
|
||||
require('MiddleClass')
|
||||
|
||||
context( 'Object', function()
|
||||
|
||||
context( 'When creating a direct subclass of Object', function()
|
||||
|
||||
context( 'using Object:subclass("name")', function()
|
||||
local MyClass = Object:subclass('MyClass')
|
||||
|
||||
test( 'should have its name correctly set up', function()
|
||||
assert_equal(MyClass.name, 'MyClass')
|
||||
end)
|
||||
|
||||
test( 'should have Object as its superclass', function()
|
||||
assert_equal(MyClass.superclass, Object)
|
||||
end)
|
||||
end)
|
||||
|
||||
context( 'When no name is given', function()
|
||||
test( 'should throw an error', function()
|
||||
assert_false( pcall(Object.subclass, Object) )
|
||||
end)
|
||||
end)
|
||||
|
||||
end)
|
||||
|
||||
context( 'An instance attribute', function()
|
||||
local Person = class('Person')
|
||||
function Person:initialize(name)
|
||||
super.initialize(self)
|
||||
self.name = name
|
||||
end
|
||||
|
||||
local AgedPerson = class('AgedPerson', Person)
|
||||
function AgedPerson:initialize(name, age)
|
||||
super.initialize(self, name)
|
||||
self.age = age
|
||||
end
|
||||
|
||||
test('should be available after being initialized', function()
|
||||
local bob = Person:new('bob')
|
||||
assert_equal(bob.name, 'bob')
|
||||
end)
|
||||
|
||||
test('should be available after being initialized by a superclass', function()
|
||||
local pete = AgedPerson:new('pete', 31)
|
||||
assert_equal(pete.name, 'pete')
|
||||
assert_equal(pete.age, 31)
|
||||
end)
|
||||
end)
|
||||
|
||||
context( 'An instance method', function()
|
||||
local A = class('A')
|
||||
function A:foo() return 'foo' end
|
||||
function A:bar() return 'bar' end
|
||||
|
||||
local B = class('B', A)
|
||||
function B:foo() return 'baz' end
|
||||
|
||||
local a = A:new()
|
||||
local b = B:new()
|
||||
|
||||
test('should be available for any instance', function()
|
||||
assert_equal(a:foo(), 'foo')
|
||||
end)
|
||||
|
||||
test('should be inheritable', function()
|
||||
assert_equal(b:bar(), 'bar')
|
||||
end)
|
||||
|
||||
test('should be overridable', function()
|
||||
assert_equal(b:foo(), 'baz')
|
||||
end)
|
||||
end)
|
||||
|
||||
context( 'A super call', function()
|
||||
local Level0 = Object:subclass('Level0')
|
||||
function Level0:initialize() self.type = self:getType() end
|
||||
function Level0:getType() return 'level0' end
|
||||
function Level0:getNumber() return 10 end
|
||||
|
||||
local Level1 = Level0:subclass('Level1')
|
||||
function Level1:initialize() super.initialize(self) end
|
||||
function Level1:getType() return 'level1' end
|
||||
|
||||
local Level2 = Level1:subclass('Level2')
|
||||
function Level2:initialize() super.initialize(self) end
|
||||
function Level2:getType() return 'level2' end
|
||||
-- Calling super.getNumber(self) on a Level2 object skips Level1
|
||||
-- (since it's not overriden here) and calls Level0:getNumber()
|
||||
function Level2:getNumber() return super.getNumber(self) + 1 end
|
||||
|
||||
local level0 = Level0:new()
|
||||
local level1 = Level1:new()
|
||||
local level2 = Level2:new()
|
||||
|
||||
test('should jump accross classes when not defined on the middle one', function()
|
||||
assert_equal(level2:getNumber(), 11)
|
||||
end)
|
||||
|
||||
test('should use the appropiate versions of each method on every level', function()
|
||||
assert_equal(level0.type, 'level0')
|
||||
assert_equal(level1.type, 'level1')
|
||||
assert_equal(level2.type, 'level2')
|
||||
end)
|
||||
end)
|
||||
|
||||
context( 'A class attribute', function()
|
||||
local A = class('A')
|
||||
A.foo = 'foo'
|
||||
|
||||
local B = class('B', A)
|
||||
|
||||
test('should be available after being initialized', function()
|
||||
assert_equal(A.foo, 'foo')
|
||||
end)
|
||||
|
||||
test('should be available for subclasses', function()
|
||||
assert_equal(B.foo, 'foo')
|
||||
end)
|
||||
|
||||
test('should be overridable by subclasses, without affecting the superclasses', function()
|
||||
B.foo = 'chunky bacon'
|
||||
assert_equal(B.foo, 'chunky bacon')
|
||||
assert_equal(A.foo, 'foo')
|
||||
end)
|
||||
end)
|
||||
|
||||
context( 'A class method', function()
|
||||
local A = class('A')
|
||||
function A.foo(theClass) return 'foo' end
|
||||
|
||||
local B = class('B', A)
|
||||
|
||||
test('should be available after being initialized', function()
|
||||
assert_equal(A:foo(), 'foo')
|
||||
end)
|
||||
|
||||
test('should be available for subclasses', function()
|
||||
assert_equal(B:foo(), 'foo')
|
||||
end)
|
||||
|
||||
test('should be overridable by subclasses, without affecting the superclasses', function()
|
||||
function B.foo(theClass) return 'chunky bacon' end
|
||||
assert_equal(B:foo(), 'chunky bacon')
|
||||
assert_equal(A:foo(), 'foo')
|
||||
end)
|
||||
end)
|
||||
|
||||
context( 'A Mixin', function()
|
||||
|
||||
local Class1 = class('Class1')
|
||||
local Mixin = {}
|
||||
function Mixin:included(theClass) theClass.includesMixin = true end
|
||||
function Mixin:foo() return 'foo' end
|
||||
function Mixin:bar() return 'bar' end
|
||||
Class1:include(Mixin)
|
||||
|
||||
Class2 = class('Class2', Class1)
|
||||
function Class2:foo() return 'baz' end
|
||||
|
||||
test('should invoke the "included" method when included', function()
|
||||
assert_true(Class1.includesMixin)
|
||||
end)
|
||||
|
||||
test('should have all its functions (except "included") copied to its target class', function()
|
||||
assert_equal(Class1:foo(), 'foo')
|
||||
assert_equal(Class1.included, nil)
|
||||
end)
|
||||
|
||||
test('should make its functions available to subclasses', function()
|
||||
assert_equal(Class2:bar(), 'bar')
|
||||
end)
|
||||
|
||||
test('should allow overriding of methods on subclasses', function()
|
||||
assert_equal(Class2:foo(), 'baz')
|
||||
end)
|
||||
|
||||
end)
|
||||
|
||||
context( 'Metamethods', function()
|
||||
|
||||
context('Custom Metamethods', function()
|
||||
-- Tests all metamethods. Note that __len is missing (lua makes table length unoverridable)
|
||||
-- I'll use a() to note the length of vector "a" (I would have preferred to use #a, but it's not possible)
|
||||
-- I'll be using 'a' instead of 'self' on this example since it is shorter
|
||||
local Vector= class('Vector')
|
||||
function Vector.initialize(a,x,y,z) a.x, a.y, a.z = x,y,z end
|
||||
function Vector.__tostring(a) return a.class.name .. '[' .. a.x .. ',' .. a.y .. ',' .. a.z .. ']' end
|
||||
function Vector.__eq(a,b) return a.x==b.x and a.y==b.y and a.z==b.z end
|
||||
function Vector.__lt(a,b) return a() < b() end
|
||||
function Vector.__le(a,b) return a() <= b() end
|
||||
function Vector.__add(a,b) return Vector:new(a.x+b.x, a.y+b.y ,a.z+b.z) end
|
||||
function Vector.__sub(a,b) return Vector:new(a.x-b.x, a.y-b.y, a.z-b.z) end
|
||||
function Vector.__div(a,s) return Vector:new(a.x/s, a.y/s, a.z/s) end
|
||||
function Vector.__unm(a) return Vector:new(-a.x, -a.y, -a.z) end
|
||||
function Vector.__concat(a,b) return a.x*b.x+a.y*b.y+a.z*b.z end
|
||||
function Vector.__call(a) return math.sqrt(a.x*a.x+a.y*a.y+a.z*a.z) end
|
||||
function Vector.__pow(a,b)
|
||||
return Vector:new(a.y*b.z-a.z*b.y,a.z*b.x-a.x*b.z,a.x*b.y-a.y*b.x)
|
||||
end
|
||||
function Vector.__mul(a,b)
|
||||
if type(b)=="number" then return Vector:new(a.x*b, a.y*b, a.z*b) end
|
||||
if type(a)=="number" then return Vector:new(a*b.x, a*b.y, a*b.z) end
|
||||
end
|
||||
|
||||
local a = Vector:new(1,2,3)
|
||||
local b = Vector:new(2,4,6)
|
||||
|
||||
for metamethod,values in pairs({
|
||||
__tostring = { tostring(a), "Vector[1,2,3]" },
|
||||
__eq = { a, a},
|
||||
__lt = { a<b, true },
|
||||
__le = { a<=b, true },
|
||||
__add = { a+b, Vector(3,6,9) },
|
||||
__sub = { b-a, Vector(1,2,3) },
|
||||
__div = { b/2, Vector(1,2,3) },
|
||||
__unm = { -a, Vector(-1,-2,-3) },
|
||||
__concat = { a..b, 28 },
|
||||
__call = { a(), math.sqrt(14) },
|
||||
__pow = { a^b, Vector(0,0,0) },
|
||||
__mul = { 4*a, Vector(4,8,12) }
|
||||
}) do
|
||||
test(metamethod .. ' should work', function()
|
||||
assert_equal(values[1], values[2])
|
||||
end)
|
||||
end
|
||||
|
||||
context('Inherited Metamethods', function()
|
||||
local Vector2= class('Vector2', Vector)
|
||||
function Vector2:initialize(x,y,z) super.initialize(self,x,y,z) end
|
||||
|
||||
local c = Vector2:new(1,2,3)
|
||||
local d = Vector2:new(2,4,6)
|
||||
for metamethod,values in pairs({
|
||||
__tostring = { tostring(c), "Vector2[1,2,3]" },
|
||||
__eq = { c, c },
|
||||
__lt = { c<d, true },
|
||||
__le = { c<=d, true },
|
||||
__add = { c+d, Vector(3,6,9) },
|
||||
__sub = { d-c, Vector(1,2,3) },
|
||||
__div = { d/2, Vector(1,2,3) },
|
||||
__unm = { -c, Vector(-1,-2,-3) },
|
||||
__concat = { c..d, 28 },
|
||||
__call = { c(), math.sqrt(14) },
|
||||
__pow = { c^d, Vector(0,0,0) },
|
||||
__mul = { 4*c, Vector(4,8,12) }
|
||||
}) do
|
||||
test(metamethod .. ' should work', function()
|
||||
assert_equal(values[1], values[2])
|
||||
end)
|
||||
end
|
||||
end)
|
||||
|
||||
end)
|
||||
|
||||
context('Default Metamethods', function()
|
||||
local Peter = class('Peter')
|
||||
local peter = Peter()
|
||||
|
||||
context('A Class', function()
|
||||
test('should have a tostring metamethod', function()
|
||||
assert_equal(tostring(Peter), 'class Peter')
|
||||
end)
|
||||
test('should have a call metamethod', function()
|
||||
assert_true(instanceOf(Peter, peter))
|
||||
end)
|
||||
end)
|
||||
|
||||
context('An instance', function()
|
||||
test('should have a tostring metamethod, different from Object.__tostring', function()
|
||||
assert_not_equal(Peter.__tostring, Object.__tostring)
|
||||
assert_equal(tostring(peter), 'instance of Peter')
|
||||
end)
|
||||
end)
|
||||
end)
|
||||
|
||||
end)
|
||||
|
||||
end)
|
@ -1,38 +0,0 @@
|
||||
h1. MiddleClass specs
|
||||
|
||||
These specs have been implemented using "telescope":http://github.com/norman/telescope
|
||||
|
||||
They are a series of tests that verify that the current implementation of MiddleClass works as expected. It is recommended to run them after modifying MiddleClass.
|
||||
|
||||
h2. Installation dependencies for the specs
|
||||
|
||||
First step is installing telescope. The easiest way in ubuntu is installing luarocks and then telescope itself:
|
||||
|
||||
<pre>
|
||||
sudo apt-get install lua luarocks
|
||||
sudo luarocks build telescope --from=http://luarocks.luaforge.net/rocks-cvs/
|
||||
</pre>
|
||||
|
||||
The first command might ask about installing some dependencies; answer (y)es to that.
|
||||
|
||||
h2. Test execution
|
||||
|
||||
Open a console and change to the middleclass top-directory:
|
||||
|
||||
<pre>
|
||||
cd middleclass
|
||||
</pre>
|
||||
|
||||
Then execute @specs/run.lua@. It should be configured to start the testing automatically.
|
||||
<pre>
|
||||
specs/run.lua
|
||||
</pre>
|
||||
|
||||
If everything has been set up properly, you should see lost of testing being done, and, hopefully, passing.
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
@ -1,37 +0,0 @@
|
||||
require('MiddleClass')
|
||||
|
||||
context( 'class', function()
|
||||
|
||||
context( 'When creating a class', function()
|
||||
|
||||
context( 'using class("name")', function()
|
||||
local TheClass = class('TheClass')
|
||||
|
||||
test( 'should have the correct name', function()
|
||||
assert_equal(TheClass.name, 'TheClass')
|
||||
end)
|
||||
|
||||
test( 'should have Object as their superclass', function()
|
||||
assert_equal(TheClass.superclass, Object)
|
||||
end)
|
||||
end)
|
||||
|
||||
context( 'using class("name", AClass)', function()
|
||||
local TheSuperClass = class('TheSuperClass')
|
||||
local TheSubClass = class('TheSubClass', TheSuperClass)
|
||||
|
||||
test( 'should have the correct superclass', function()
|
||||
assert_equal(TheSubClass.superclass, TheSuperClass)
|
||||
end)
|
||||
end)
|
||||
|
||||
context( 'using no name', function()
|
||||
test( 'class() should throw an error', function()
|
||||
assert_error(class)
|
||||
end)
|
||||
end)
|
||||
|
||||
end)
|
||||
|
||||
|
||||
end)
|
@ -1,66 +0,0 @@
|
||||
require('MiddleClass')
|
||||
|
||||
context( 'includes', function()
|
||||
|
||||
context( 'Primitives', function()
|
||||
local o = Object:new()
|
||||
local primitives = {nil, 1, 'hello', {}, function() end}
|
||||
|
||||
for _,primitive in pairs(primitives) do
|
||||
local theType = type(primitive)
|
||||
context('A ' .. theType, function()
|
||||
|
||||
local f1 = function() return includes(Object, primitive) end
|
||||
local f2 = function() return includes(primitive, o) end
|
||||
local f3 = function() return includes(primitive, primitive) end
|
||||
|
||||
context('should not throw errors', function()
|
||||
test('includes(Object, '.. theType ..')', function()
|
||||
assert_not_error(f1)
|
||||
end)
|
||||
test('includes(' .. theType .. ', Object:new())', function()
|
||||
assert_not_error(f2)
|
||||
end)
|
||||
test('includes(' .. theType .. ',' .. theType ..')', function()
|
||||
assert_not_error(f3)
|
||||
end)
|
||||
end)
|
||||
|
||||
test('should make includes return false', function()
|
||||
assert_false(f1())
|
||||
assert_false(f2())
|
||||
assert_false(f3())
|
||||
end)
|
||||
|
||||
end)
|
||||
end -- for
|
||||
|
||||
end)
|
||||
|
||||
context( 'A class', function()
|
||||
|
||||
local Class1 = class('Class1')
|
||||
local Class2 = class('Class2', Class1)
|
||||
local Class3 = class('Class3', Class2)
|
||||
local UnrelatedClass = class('Unrelated')
|
||||
|
||||
local hasFoo = { foo=function() return 'foo' end }
|
||||
Class1:include(hasFoo)
|
||||
|
||||
test('should return true if it includes a mixin', function()
|
||||
assert_true(includes(hasFoo, Class1))
|
||||
end)
|
||||
|
||||
test('should return true if its superclass includes a mixin', function()
|
||||
assert_true(includes(hasFoo, Class2))
|
||||
assert_true(includes(hasFoo, Class3))
|
||||
end)
|
||||
|
||||
test('should return false otherwise', function()
|
||||
assert_false(includes(hasFoo, UnrelatedClass))
|
||||
end)
|
||||
|
||||
end)
|
||||
|
||||
|
||||
end)
|
@ -1,81 +0,0 @@
|
||||
require('MiddleClass')
|
||||
|
||||
context( 'instanceOf', function()
|
||||
|
||||
context( 'Primitives', function()
|
||||
local o = Object:new()
|
||||
local primitives = {nil, 1, 'hello', {}, function() end}
|
||||
|
||||
for _,primitive in pairs(primitives) do
|
||||
local theType = type(primitive)
|
||||
context('A ' .. theType, function()
|
||||
|
||||
local f1 = function() return instanceOf(Object, primitive) end
|
||||
local f2 = function() return instanceOf(primitive, o) end
|
||||
local f3 = function() return instanceOf(primitive, primitive) end
|
||||
|
||||
context('should not throw errors', function()
|
||||
test('instanceOf(Object, '.. theType ..')', function()
|
||||
assert_not_error(f1)
|
||||
end)
|
||||
test('instanceOf(' .. theType .. ', Object:new())', function()
|
||||
assert_not_error(f2)
|
||||
end)
|
||||
test('instanceOf(' .. theType .. ',' .. theType ..')', function()
|
||||
assert_not_error(f3)
|
||||
end)
|
||||
end)
|
||||
|
||||
test('should make instanceOf return false', function()
|
||||
assert_false(f1())
|
||||
assert_false(f2())
|
||||
assert_false(f3())
|
||||
end)
|
||||
|
||||
end)
|
||||
end -- for
|
||||
|
||||
end)
|
||||
|
||||
context( 'An instance', function()
|
||||
local Class1 = class('Class1')
|
||||
local Class2 = class('Class2', Class1)
|
||||
local Class3 = class('Class3', Class2)
|
||||
local UnrelatedClass = class('Unrelated')
|
||||
|
||||
local o1, o2, o3 = Class1:new(), Class2:new(), Class3:new()
|
||||
|
||||
test('should be instanceOf(Object)', function()
|
||||
assert_true(instanceOf(Object, o1))
|
||||
assert_true(instanceOf(Object, o2))
|
||||
assert_true(instanceOf(Object, o3))
|
||||
end)
|
||||
|
||||
test('should be instanceOf its class', function()
|
||||
assert_true(instanceOf(Class1, o1))
|
||||
assert_true(instanceOf(Class2, o2))
|
||||
assert_true(instanceOf(Class3, o3))
|
||||
end)
|
||||
|
||||
test('should be instanceOf its class\' superclasses', function()
|
||||
assert_true(instanceOf(Class1, o2))
|
||||
assert_true(instanceOf(Class1, o3))
|
||||
assert_true(instanceOf(Class2, o3))
|
||||
end)
|
||||
|
||||
test('should not be an instanceOf its class\' subclasses', function()
|
||||
assert_false(instanceOf(Class2, o1))
|
||||
assert_false(instanceOf(Class3, o1))
|
||||
assert_false(instanceOf(Class3, o2))
|
||||
end)
|
||||
|
||||
test('should not be an instanceOf an unrelated class', function()
|
||||
assert_false(instanceOf(UnrelatedClass, o1))
|
||||
assert_false(instanceOf(UnrelatedClass, o2))
|
||||
assert_false(instanceOf(UnrelatedClass, o3))
|
||||
end)
|
||||
|
||||
end)
|
||||
|
||||
|
||||
end)
|
@ -1,16 +0,0 @@
|
||||
#!/usr/bin/env lua
|
||||
|
||||
-- command should be specs/run.lua
|
||||
local command = debug.getinfo(1).source:match("@(.*)$"):gsub('\\', '/')
|
||||
|
||||
if command ~= 'specs/run.lua' then
|
||||
error('You must run the specs like this: specs/run.lua (or specs\\run.lua for windows)')
|
||||
end
|
||||
|
||||
-- This only works if command is spec/run.lua
|
||||
local config = {
|
||||
package_path = "package.path = './?.lua'",
|
||||
specs_dir= "specs/"
|
||||
}
|
||||
|
||||
os.execute( ("tsc -f --before=%q %s*_spec.lua "):format( config.package_path, config.specs_dir ) )
|
@ -1,73 +0,0 @@
|
||||
require('MiddleClass')
|
||||
|
||||
context( 'subclassOf', function()
|
||||
|
||||
context( 'Primitives', function()
|
||||
local primitives = {nil, 1, 'hello', {}, function() end}
|
||||
|
||||
for _,primitive in pairs(primitives) do
|
||||
local theType = type(primitive)
|
||||
context('A ' .. theType, function()
|
||||
|
||||
local f1 = function() return subclassOf(Object, primitive) end
|
||||
local f2 = function() return subclassOf(primitive, o) end
|
||||
local f3 = function() return subclassOf(primitive, primitive) end
|
||||
|
||||
context('should not throw errors', function()
|
||||
test('subclassOf(Object, '.. theType ..')', function()
|
||||
assert_not_error(f1)
|
||||
end)
|
||||
test('subclassOf(' .. theType .. ', Object:new())', function()
|
||||
assert_not_error(f2)
|
||||
end)
|
||||
test('subclassOf(' .. theType .. ',' .. theType ..')', function()
|
||||
assert_not_error(f3)
|
||||
end)
|
||||
end)
|
||||
|
||||
test('should make subclassOf return false', function()
|
||||
assert_false(f1())
|
||||
assert_false(f2())
|
||||
assert_false(f3())
|
||||
end)
|
||||
|
||||
end)
|
||||
end -- for
|
||||
end)
|
||||
|
||||
context( 'Any class (except Object)', function()
|
||||
local Class1 = class('Class1')
|
||||
local Class2 = class('Class2', Class1)
|
||||
local Class3 = class('Class3', Class2)
|
||||
local UnrelatedClass = class('Unrelated')
|
||||
|
||||
test('should be subclassOf(Object)', function()
|
||||
assert_true(subclassOf(Object, Class1))
|
||||
assert_true(subclassOf(Object, Class2))
|
||||
assert_true(subclassOf(Object, Class3))
|
||||
end)
|
||||
|
||||
test('should be subclassOf its direct superclass', function()
|
||||
assert_true(subclassOf(Class1, Class2))
|
||||
assert_true(subclassOf(Class2, Class3))
|
||||
end)
|
||||
|
||||
test('should be subclassOf its ancestors', function()
|
||||
assert_true(subclassOf(Class1, Class3))
|
||||
end)
|
||||
|
||||
test('should not be an subclassOf its class\' subclasses', function()
|
||||
assert_false(subclassOf(Class2, Class1))
|
||||
assert_false(subclassOf(Class3, Class1))
|
||||
assert_false(subclassOf(Class3, Class2))
|
||||
end)
|
||||
|
||||
test('should not be an subclassOf an unrelated class', function()
|
||||
assert_false(subclassOf(UnrelatedClass, Class1))
|
||||
assert_false(subclassOf(UnrelatedClass, Class2))
|
||||
assert_false(subclassOf(UnrelatedClass, Class3))
|
||||
end)
|
||||
|
||||
end)
|
||||
|
||||
end)
|
Loading…
Reference in New Issue
Block a user