summaryrefslogtreecommitdiffstats
path: root/aconf/object.lua
blob: dca59f6c33c2ec76fc0a38cbfba97f8eeb9e1380 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
--[[
Copyright (c) 2012-2015 Kaarle Ritvanen
See LICENSE file for license details
--]]

local M = {}

function M.class(base)
   local cls = {}

   local mt = {
      __call=function(self, ...)
		local obj = {}
		setmetatable(obj, {__index=cls, class=cls})
		obj:init(...)
		return obj
	     end
   }

   if not base and M.Object then base = M.Object end
   if base then
      cls._base = base
      mt.__index = base
   end

   return setmetatable(cls, mt)
end


M.Object = M.class()

function M.Object:init(...) end


function M.super(obj, cls)
   assert(M.isinstance(obj, cls))

   local mt = {}

   function mt.__index(t, k)
      local v = cls._base[k]
      if type(v) ~= 'function' then return v end
      return function(...)
		local arg = {...}
		arg[1] = obj
		return v(table.unpack(arg))
	     end
   end

   return setmetatable({}, mt)
end


function M.issubclass(cls1, cls2)
   if cls1 == cls2 then return true end
   if cls1._base then return M.issubclass(cls1._base, cls2) end
   return false
end

function M.isinstance(obj, cls)
   if not obj or type(obj) ~= 'table' then return false end
   local mt = getmetatable(obj)
   if not mt then return false end
   return type(obj) == 'table' and mt.class and M.issubclass(mt.class, cls)
end

function M.toinstance(obj, ...)
   return getmetatable(obj).class and obj or obj(...)
end


return M