2013-09-15 21:14:00 +00:00
|
|
|
local class = require 'middleclass'
|
2013-09-15 21:18:08 +00:00
|
|
|
local Object = class.Object
|
2011-08-12 08:33:58 +00:00
|
|
|
|
2013-12-05 00:15:28 +00:00
|
|
|
local function is_lua_5_2_compatible()
|
|
|
|
return type(rawlen) == 'function'
|
|
|
|
end
|
|
|
|
|
2015-10-12 11:39:35 +00:00
|
|
|
local function is_lua_5_3_compatible()
|
|
|
|
return type(string.unpack) == 'function'
|
|
|
|
end
|
|
|
|
|
2015-11-01 16:09:15 +00:00
|
|
|
if is_lua_5_2_compatible() then
|
|
|
|
require 'spec/metamethods_lua_5_2'
|
|
|
|
end
|
2011-08-12 08:33:58 +00:00
|
|
|
|
2015-11-01 16:09:15 +00:00
|
|
|
if is_lua_5_3_compatible() then
|
|
|
|
require 'spec.metamethods_lua_5_3'
|
|
|
|
end
|
|
|
|
|
|
|
|
describe('Metamethods', function()
|
2013-09-17 18:55:50 +00:00
|
|
|
describe('Custom Metamethods', function()
|
2015-10-12 11:49:50 +00:00
|
|
|
local Vector, a, b
|
|
|
|
before_each(function()
|
|
|
|
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
|
2015-10-31 15:50:11 +00:00
|
|
|
function Vector.__add(a,b) return a.class:new(a.x+b.x, a.y+b.y ,a.z+b.z) end
|
|
|
|
function Vector.__sub(a,b) return a.class:new(a.x-b.x, a.y-b.y, a.z-b.z) end
|
|
|
|
function Vector.__div(a,s) return a.class:new(a.x/s, a.y/s, a.z/s) end
|
|
|
|
function Vector.__unm(a) return a.class:new(-a.x, -a.y, -a.z) end
|
2015-10-12 11:49:50 +00:00
|
|
|
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)
|
2015-10-31 16:16:10 +00:00
|
|
|
return a.class: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)
|
2015-10-12 11:49:50 +00:00
|
|
|
end
|
|
|
|
function Vector.__mul(a,b)
|
2015-10-31 15:50:11 +00:00
|
|
|
if type(b)=="number" then return a.class:new(a.x*b, a.y*b, a.z*b) end
|
|
|
|
if type(a)=="number" then return b.class:new(a*b.x, a*b.y, a*b.z) end
|
2015-10-12 11:49:50 +00:00
|
|
|
end
|
2015-11-19 10:59:19 +00:00
|
|
|
Vector.__metatable = "metatable of a vector"
|
2015-11-19 11:10:38 +00:00
|
|
|
Vector.__mode = "k"
|
2015-10-31 15:50:11 +00:00
|
|
|
|
2015-10-12 11:49:50 +00:00
|
|
|
a = Vector:new(1,2,3)
|
|
|
|
b = Vector:new(2,4,6)
|
|
|
|
end)
|
2011-08-12 08:33:58 +00:00
|
|
|
|
2015-10-12 21:45:32 +00:00
|
|
|
it('implements __tostring', function()
|
|
|
|
assert.equal(tostring(a), "Vector[1,2,3]")
|
|
|
|
end)
|
2015-10-12 11:39:35 +00:00
|
|
|
|
2015-10-12 21:45:32 +00:00
|
|
|
it('implements __eq', function()
|
|
|
|
assert.equal(a, a)
|
|
|
|
end)
|
|
|
|
|
|
|
|
it('implements __lt', function()
|
|
|
|
assert.is_true(a < b)
|
|
|
|
end)
|
|
|
|
|
|
|
|
it('implements __le', function()
|
|
|
|
assert.is_true(a <= b)
|
|
|
|
end)
|
|
|
|
|
|
|
|
it('implements __add', function()
|
|
|
|
assert.equal(a+b, Vector(3,6,9))
|
|
|
|
end)
|
|
|
|
|
|
|
|
it('implements __sub', function()
|
|
|
|
assert.equal(b-a, Vector(1,2,3))
|
|
|
|
end)
|
|
|
|
|
|
|
|
it('implements __div', function()
|
|
|
|
assert.equal(b/2, Vector(1,2,3))
|
|
|
|
end)
|
|
|
|
|
|
|
|
it('implements __concat', function()
|
|
|
|
assert.equal(a..b, 28)
|
|
|
|
end)
|
|
|
|
|
|
|
|
it('implements __call', function()
|
|
|
|
assert.equal(a(), math.sqrt(14))
|
|
|
|
end)
|
|
|
|
|
|
|
|
it('implements __pow', function()
|
|
|
|
assert.equal(a^b, Vector(0,0,0))
|
|
|
|
end)
|
|
|
|
|
|
|
|
it('implements __mul', function()
|
|
|
|
assert.equal(4*a, Vector(4,8,12))
|
|
|
|
end)
|
|
|
|
|
2015-11-19 10:59:19 +00:00
|
|
|
it('implements __metatable', function()
|
|
|
|
assert.equal("metatable of a vector", getmetatable(a))
|
|
|
|
end)
|
|
|
|
|
2015-11-19 11:10:38 +00:00
|
|
|
it('implements __mode', function()
|
|
|
|
a[{}] = true
|
|
|
|
collectgarbage()
|
|
|
|
for k in pairs(a) do assert.not_table(k) end
|
|
|
|
end)
|
|
|
|
|
2015-10-12 21:45:32 +00:00
|
|
|
--[[
|
|
|
|
it('implements __index', function()
|
|
|
|
assert.equal(b[1], 3)
|
|
|
|
end)
|
|
|
|
--]]
|
2013-12-05 00:15:28 +00:00
|
|
|
|
2013-09-17 18:55:50 +00:00
|
|
|
describe('Inherited Metamethods', function()
|
2015-10-12 21:45:32 +00:00
|
|
|
local Vector2, c, d
|
|
|
|
before_each(function()
|
|
|
|
Vector2= class('Vector2', Vector)
|
|
|
|
function Vector2:initialize(x,y,z) Vector.initialize(self,x,y,z) end
|
|
|
|
|
|
|
|
c = Vector2:new(1,2,3)
|
|
|
|
d = Vector2:new(2,4,6)
|
|
|
|
end)
|
|
|
|
|
|
|
|
it('implements __tostring', function()
|
|
|
|
assert.equal(tostring(c), "Vector2[1,2,3]")
|
|
|
|
end)
|
|
|
|
|
|
|
|
it('implements __eq', function()
|
|
|
|
assert.equal(c, c)
|
|
|
|
end)
|
|
|
|
|
|
|
|
it('implements __lt', function()
|
|
|
|
assert.is_true(c < d)
|
|
|
|
end)
|
|
|
|
|
|
|
|
it('implements __le', function()
|
|
|
|
assert.is_true(c <= d)
|
|
|
|
end)
|
|
|
|
|
|
|
|
it('implements __add', function()
|
2015-10-31 16:16:10 +00:00
|
|
|
assert.equal(c+d, Vector2(3,6,9))
|
2015-10-12 21:45:32 +00:00
|
|
|
end)
|
|
|
|
|
|
|
|
it('implements __sub', function()
|
2015-10-31 16:16:10 +00:00
|
|
|
assert.equal(d-c, Vector2(1,2,3))
|
2015-10-12 21:45:32 +00:00
|
|
|
end)
|
|
|
|
|
|
|
|
it('implements __div', function()
|
2015-10-31 16:16:10 +00:00
|
|
|
assert.equal(d/2, Vector2(1,2,3))
|
2015-10-12 21:45:32 +00:00
|
|
|
end)
|
|
|
|
|
|
|
|
it('implements __concat', function()
|
|
|
|
assert.equal(c..d, 28)
|
|
|
|
end)
|
|
|
|
|
|
|
|
it('implements __call', function()
|
|
|
|
assert.equal(c(), math.sqrt(14))
|
|
|
|
end)
|
|
|
|
|
|
|
|
it('implements __pow', function()
|
2015-10-31 16:16:10 +00:00
|
|
|
assert.equal(c^d, Vector2(0,0,0))
|
2015-10-12 21:45:32 +00:00
|
|
|
end)
|
|
|
|
|
|
|
|
it('implements __mul', function()
|
2015-10-31 16:16:10 +00:00
|
|
|
assert.equal(4*c, Vector2(4,8,12))
|
2015-10-12 21:45:32 +00:00
|
|
|
end)
|
2015-11-19 10:13:04 +00:00
|
|
|
|
2015-11-19 10:59:19 +00:00
|
|
|
it('implements __metatable', function()
|
|
|
|
assert.equal("metatable of a vector", getmetatable(c))
|
|
|
|
end)
|
|
|
|
|
2015-11-19 11:10:38 +00:00
|
|
|
it('implements __mode', function()
|
|
|
|
c[{}] = true
|
|
|
|
collectgarbage()
|
|
|
|
for k in pairs(c) do assert.not_table(k) end
|
|
|
|
end)
|
|
|
|
|
2015-11-23 12:42:49 +00:00
|
|
|
it('allows inheriting further', function()
|
|
|
|
local Vector3 = class('Vector3', Vector2)
|
|
|
|
local e = Vector3(1,2,3)
|
|
|
|
local f = Vector3(3,4,5)
|
|
|
|
assert.equal(e+f, Vector3(4,6,8))
|
|
|
|
end)
|
|
|
|
|
2015-11-19 10:13:04 +00:00
|
|
|
describe('Updates', function()
|
|
|
|
it('overrides __add', function()
|
|
|
|
Vector2.__add = function(a, b) return Vector.__add(a, b)/2 end
|
|
|
|
assert.equal(c+d, Vector2(1.5,3,4.5))
|
|
|
|
end)
|
|
|
|
|
|
|
|
it('updates __add', function()
|
|
|
|
Vector.__add = Vector.__sub
|
|
|
|
assert.equal(c+d, Vector2(-1,-2,-3))
|
|
|
|
end)
|
|
|
|
|
|
|
|
it('does not update __add after overriding', function()
|
|
|
|
Vector2.__add = function(a, b) return Vector.__add(a, b)/2 end
|
|
|
|
Vector.__add = Vector.__sub
|
|
|
|
assert.equal(c+d, Vector2(-0.5,-1,-1.5))
|
|
|
|
end)
|
|
|
|
|
|
|
|
it('reverts __add override', function()
|
|
|
|
Vector2.__add = function(a, b) return Vector.__add(a, b)/2 end
|
|
|
|
Vector2.__add = nil
|
|
|
|
assert.equal(c+d, Vector2(3,6,9))
|
|
|
|
end)
|
|
|
|
end)
|
2011-08-12 08:33:58 +00:00
|
|
|
end)
|
|
|
|
end)
|
|
|
|
|
2015-11-24 15:48:17 +00:00
|
|
|
describe('Custom __index and __newindex', function()
|
|
|
|
describe('Tables', function()
|
|
|
|
local Proxy, fallback, p
|
|
|
|
before_each(function()
|
|
|
|
Proxy = class('Proxy')
|
|
|
|
fallback = {foo = 'bar', common = 'fallback'}
|
|
|
|
Proxy.__index = fallback
|
|
|
|
Proxy.__newindex = fallback
|
|
|
|
Proxy.common = 'class'
|
|
|
|
p = Proxy()
|
|
|
|
end)
|
|
|
|
|
|
|
|
it('uses __index', function()
|
|
|
|
assert.equal(p.foo, 'bar')
|
|
|
|
end)
|
|
|
|
|
|
|
|
it('does not use __index when field exists in class', function()
|
|
|
|
assert.equal(p.common, 'class')
|
|
|
|
end)
|
|
|
|
|
|
|
|
it('uses __newindex', function()
|
|
|
|
p.key = 'value'
|
|
|
|
assert.equal(fallback.key, 'value')
|
|
|
|
end)
|
|
|
|
|
2015-11-28 15:10:35 +00:00
|
|
|
it('uses __newindex when field exists in class', function()
|
2015-11-24 15:48:17 +00:00
|
|
|
p.common = 'value'
|
2015-11-28 15:10:35 +00:00
|
|
|
assert.equal(p.common, 'class')
|
2015-11-24 15:48:17 +00:00
|
|
|
assert.equal(Proxy.common, 'class')
|
2015-11-28 15:10:35 +00:00
|
|
|
assert.equal(fallback.common, 'value')
|
2015-11-24 15:48:17 +00:00
|
|
|
end)
|
|
|
|
end)
|
|
|
|
|
|
|
|
describe('Functions', function()
|
|
|
|
local Namespace, Rectangle, r
|
|
|
|
before_each(function()
|
|
|
|
Namespace = class('Namespace')
|
|
|
|
function Namespace:__index(name)
|
|
|
|
local getter = self.class[name.."Getter"]
|
|
|
|
if getter then return getter(self) end
|
|
|
|
end
|
|
|
|
function Namespace:__newindex(name, value)
|
|
|
|
local setter = self.class[name.."Setter"]
|
|
|
|
if setter then setter(self, value) else rawset(self, name, value) end
|
|
|
|
end
|
|
|
|
Rectangle = class('Rectangle', Namespace)
|
|
|
|
function Rectangle:initialize(x, y, scale)
|
|
|
|
self._scale, self.x, self.y = 1, x, y
|
|
|
|
self.scale = scale
|
|
|
|
end
|
|
|
|
function Rectangle:scaleGetter() return self._scale end
|
|
|
|
function Rectangle:scaleSetter(v)
|
|
|
|
self.x = self.x*v/self._scale
|
|
|
|
self.y = self.y*v/self._scale
|
|
|
|
self._scale = v
|
|
|
|
end
|
|
|
|
function Rectangle:areaGetter() return self.x * self.y end
|
|
|
|
r = Rectangle(3, 4, 2)
|
|
|
|
end)
|
|
|
|
|
|
|
|
it('uses setter', function()
|
|
|
|
assert.equal(r.x, 6)
|
|
|
|
assert.equal(r.y, 8)
|
|
|
|
r.scale = 3
|
|
|
|
assert.equal(r.x, 9)
|
|
|
|
assert.equal(r.y, 12)
|
|
|
|
end)
|
|
|
|
|
|
|
|
it('uses getters', function()
|
|
|
|
assert.equal(r.scale, 2)
|
|
|
|
assert.equal(r.area, 48)
|
|
|
|
end)
|
|
|
|
|
|
|
|
it('updates inherited __index', function()
|
|
|
|
function Namespace.__index() return 42 end
|
|
|
|
assert.equal(r.area, 42)
|
|
|
|
function Rectangle.__index() return 24 end
|
|
|
|
assert.equal(r.area, 24)
|
|
|
|
function Namespace.__index() return 96 end
|
|
|
|
assert.equal(r.area, 24)
|
|
|
|
Rectangle.__index = nil
|
|
|
|
assert.equal(r.area, 96)
|
|
|
|
end)
|
|
|
|
end)
|
|
|
|
end)
|
|
|
|
|
2013-09-17 18:55:50 +00:00
|
|
|
describe('Default Metamethods', function()
|
2011-08-12 08:33:58 +00:00
|
|
|
|
|
|
|
local Peter, peter
|
|
|
|
|
2013-09-17 18:55:50 +00:00
|
|
|
before_each(function()
|
2011-08-12 08:33:58 +00:00
|
|
|
Peter = class('Peter')
|
|
|
|
peter = Peter()
|
|
|
|
end)
|
|
|
|
|
2013-09-17 18:55:50 +00:00
|
|
|
describe('A Class', function()
|
|
|
|
it('has a call metamethod properly set', function()
|
|
|
|
assert.is_true(peter:isInstanceOf(Peter))
|
2011-08-12 08:33:58 +00:00
|
|
|
end)
|
2013-09-17 18:55:50 +00:00
|
|
|
it('has a tostring metamethod properly set', function()
|
|
|
|
assert.equal(tostring(Peter), 'class Peter')
|
2011-08-12 08:33:58 +00:00
|
|
|
end)
|
|
|
|
end)
|
|
|
|
|
2013-09-17 18:55:50 +00:00
|
|
|
describe('An instance', function()
|
|
|
|
it('has a tostring metamethod, returning a different result from Object.__tostring', function()
|
|
|
|
assert.equal(tostring(peter), 'instance of class Peter')
|
2011-08-12 08:33:58 +00:00
|
|
|
end)
|
|
|
|
end)
|
|
|
|
end)
|
|
|
|
|
|
|
|
end)
|