-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathtableHooker.lua
127 lines (123 loc) · 3.4 KB
/
tableHooker.lua
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
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
module(..., package.seeall)
--==================================================
local ListenerNode = {}
ListenerNode.__index = ListenerNode
ListenerNode.__newindex = ListenerNode
function ListenerNode.new(parent, k)
local node = { parent = parent, children = {}, listeners = {}}
if parent then parent.children[k] = node end
setmetatable(node, ListenerNode)
return node
end
function ListenerNode:getOrCreateChild(path)
local node = self
for k in string.gmatch(path, '[^%.]+') do
local child = node.children[k]
if not child then child = ListenerNode.new(node, k) end
node = child
end
return node
end
function ListenerNode:getChild(path)
local node = self
for k in string.gmatch(path, '[^%.]+') do
node = node.children[k]
if not node then break end
end
return node
end
function ListenerNode:collectSelfAndParents(set)
local node = self
while node do
set[node] = true
node = node.parent
end
end
function ListenerNode:collectSelfAndChildren(set)
set[self] = true
for _, v in pairs(self.children) do
v:collectSelfAndChildren(set)
end
end
function ListenerNode:addListener(f)
self.listeners[f] = true
end
function ListenerNode:removeListener(f)
self.listeners[f] = nil
end
function ListenerNode:invokeListeners(set)
for f in pairs(self.listeners) do set[f] = true end
end
--==================================================
local g_mt = {}
local function _getOrCreateProxy(parentProxy, key, listenerNode, realT)
local r = parentProxy and parentProxy.__children[key]
if not r then
r = { __children = {}, __listenerNode = listenerNode,}
if parentProxy then
parentProxy.__children[key] = r
r.__root = parentProxy.__root
else
r.__root = r
r.__activeListenerNodes = {}
r.__enableAutoInvokeListeners = true
end
end
r.__realT = realT
setmetatable(r, g_mt)
return r
end
g_mt.__index = function(t, k)
local v = t.__realT[k]
if type(v) == 'table' then
v = _getOrCreateProxy(t, k, t.__listenerNode:getOrCreateChild(k), v)
end
return v
end
g_mt.__newindex = function(t, k, v)
if t.__realT[k] ~= v then
t.__realT[k] = v
local node = t.__listenerNode
node:collectSelfAndParents(t.__root.__activeListenerNodes)
node = node:getChild(k)
if node then
node:collectSelfAndChildren(t.__root.__activeListenerNodes)
end
if t.__root.__enableAutoInvokeListeners then
invokeActiveListeners(t.__root)
end
end
end
g_mt.__mode = 'v'
function addListener(t, path, f)
t.__listenerNode:getOrCreateChild(path):addListener(f)
end
function removeListener(t, path, f)
t.__listenerNode:getOrCreateChild(path):removeListener(f)
end
function invokeActiveListeners(t)
local listeners = {}
for node, _ in pairs(t.__activeListenerNodes) do
node:invokeListeners(listeners)
end
t.__activeListenerNodes = {}
for f in pairs(listeners) do
f()
end
end
function enableAutoInvokeListeners(t, enable)
t.__enableAutoInvokeListeners = enable
end
function _pairs(t)
local k = nil
return function()
k = next(t.__realT, k)
return k, t[k]
end
end
function hook(t)
return _getOrCreateProxy(nil, '', ListenerNode.new(), t)
end
function unhook(t)
return t.__realT
end