-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathmetadata.lua
389 lines (373 loc) · 12.3 KB
/
metadata.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
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
mineunit("common/misc_helpers")
local function assert_invlist_index(index, size)
assert(type(index) == "number" and math.floor(index) == index, "InvList:set_stack: Invalid InvList stack index")
assert(index > 0, "InvList:set_stack: InvList stack index should never be less than 1")
assert(index <= size, "InvList:set_stack: Invalid InvList stack index larger than InvList size")
end
-- TODO: This should be moved to mineunit.utils with configurable failure behavior
local function get_integer(value, msg)
local number = tonumber(value)
if number == nil then
error(msg.."Expected number, got: "..tostring(value))
elseif type(number) ~= "number" then
mineunit:warning(msg.."Expected number, got: "..type(value))
end
local integer = math.floor(number)
if number ~= integer then
mineunit:warning(msg.."Expected integer, got: "..tostring(value))
end
return integer
end
local function invlist_tostring(list)
local items = {}
for _, item in ipairs(list) do
table.insert(items, tostring(item))
end
return ('InvList({%s}, %d)'):format(table.concat(items, ","), #list)
end
--
-- InvRef
--
local InvRef = {}
-- * `is_empty(listname)`: return `true` if list is empty
function InvRef:is_empty(listname)
assert.is_string(listname, "InvRef:is_empty listname must be string. Got "..type(listname))
if self._empty[listname] ~= nil then
return self._empty[listname]
elseif self._lists[listname] then
for _, stack in ipairs(self._lists[listname]) do
if not stack:is_empty() then
self._empty[listname] = false
return false
end
end
self._empty[listname] = true
end
return true
end
-- * `get_size(listname)`: get size of a list
function InvRef:get_size(listname)
return self._sizes[listname] or 0
end
-- * `set_size(listname, size)`: set size of a list
-- * returns `false` on error (e.g. invalid `listname` or `size`)
function InvRef:set_size(listname, size)
local newsize = get_integer(size, "InvRef:set_size: ")
if newsize < 0 then
mineunit:warning("InvRef:set_size: Invalid size: "..tostring(size))
return false
elseif newsize == 0 then
self._lists[listname] = nil
self._empty[listname] = nil
newsize = nil
elseif newsize == self._sizes[listname] then
return true
elseif not self._lists[listname] then
self._lists[listname] = {}
setmetatable(self._lists[listname], { __tostring = invlist_tostring })
self._empty[listname] = true
for index = 1, newsize do
self._lists[listname][index] = ItemStack()
end
elseif newsize > self._sizes[listname] then
for index = self._sizes[listname], newsize do
self._lists[listname][index] = ItemStack()
end
elseif newsize < self._sizes[listname] then
self._empty[listname] = self._empty[listname] and true or nil
for index = self._sizes[listname], newsize, -1 do
self._lists[listname][index] = nil
end
end
self._sizes[listname] = newsize
return true
end
-- * `get_width(listname)`: get width of a list
function InvRef:get_width(listname)
error("NOT IMPLEMENTED")
end
-- * `set_width(listname, width)`: set width of list; currently used for crafting
function InvRef:set_width(listname, width)
error("NOT IMPLEMENTED")
end
-- * `get_stack(listname, i)`: get a copy of stack index `i` in list
function InvRef:get_stack(listname, index)
local list = self:get_list(listname)
assert(list, "InvRef:set_stack: Invalid inventory list " .. tostring(list))
assert_invlist_index(index, self:get_size(listname))
return list[index] and ItemStack(list[index]) or ItemStack()
end
-- * `set_stack(listname, i, stack)`: copy `stack` to index `i` in list
function InvRef:set_stack(listname, index, stack)
local list = self._lists[listname]
assert(list, "InvRef:set_stack: Invalid inventory list " .. tostring(list))
assert_invlist_index(index, self._sizes[listname])
-- Either clone or create itemstack, both cases are needed here. References should not be added to lists.
stack = ItemStack(stack)
local input_empty = stack:is_empty()
local target_empty = self._empty[listname] or list[index]:is_empty()
if input_empty and target_empty then
-- Both input and target stacks are empty, skip inventory update
return
elseif not target_empty then
if input_empty then
self._empty[listname] = self._sizes[listname] == 1 and true or nil
end
elseif not input_empty then
self._empty[listname] = false
end
list[index] = stack
end
-- * `get_list(listname)`: return full list
function InvRef:get_list(listname)
if not self._lists[listname] then
mineunit:warning("InvRef:get_list list not found: "..tostring(listname))
return nil
end
local result = {}
for index, stack in ipairs(self._lists[listname]) do
result[index] = ItemStack(stack)
end
return result
end
-- * `set_list(listname, list)`: set full list (size will not change)
function InvRef:set_list(listname, list)
assert.is_string(listname, "InvRef:set_list expected `listname` to be string, got "..type(listname))
assert.is_table(list, "InvRef:set_list expected `list` to be table, got " .. type(list))
assert.is_table(self._lists[listname], "InvRef:set_list list does not exist "..tostring(listname))
assert.is_table(list, "InvRef:set_stack: Invalid inventory list " .. tostring(list))
-- Update list if input or target contains anything
if not self._empty[listname] or next(list) then
local empty = true
local target = self._lists[listname]
for index = 1, self._sizes[listname] do
-- Either clone or create itemstack, both cases are needed here. References should not be added to lists.
local stack = ItemStack(list[index])
if stack:is_empty() then
if not target[index]:is_empty() then
target[index] = stack
end
else
empty = false
target[index] = stack
end
end
self._empty[listname] = empty
end
end
-- * `get_lists()`: returns list of inventory lists
function InvRef:get_lists()
local results = {}
for listname,list in pairs(self._lists) do
results[listname] = table.copy(list)
end
return results
end
-- * `set_lists(lists)`: sets inventory lists (size will not change)
function InvRef:set_lists(lists)
for listname,list in pairs(lists) do
self:set_list(listname, list)
end
end
-- * `add_item(listname, stack)`: add item somewhere in list, returns leftover
function InvRef:add_item(listname, stack)
stack = ItemStack(stack)
if not stack:is_empty() then
local list = self._lists[listname]
local count = stack:get_count()
assert(list, "InvRef:add_item: Invalid inventory list " .. tostring(listname))
for index = 1, self:get_size(listname) do
-- Try to add items into stack, get and check leftover stack
stack = list[index]:add_item(stack)
if stack:is_empty() then
break
end
end
if stack:get_count() < count then
self._empty[listname] = false
end
end
return stack
end
-- * `room_for_item(listname, stack)`: returns `true` if the stack of items can be fully added to the list
function InvRef:room_for_item(listname, stack)
for _, slot in ipairs(self:get_list(listname)) do
if slot:item_fits(stack) then
return true
end
end
return false
end
-- * `contains_item(listname, stack, [match_meta])`: returns `true` if
-- the stack of items can be fully taken from the list.
-- If `match_meta` is false, only the items names are compared
-- (default: `false`).
function InvRef:contains_item(listname, stack, match_meta)
local list = self:get_list(listname)
assert(list, "InvRef:contains_item: Invalid inventory list " .. tostring(listname))
stack = type(stack) == "string" and ItemStack(stack) or stack
local name = stack:get_name()
local count = stack:get_count()
local meta1 = match_meta and stack:get_meta():to_table().fields or nil
for _, liststack in ipairs(list) do
if liststack:get_name() == name and liststack:get_count() >= count then
if not match_meta then
return true
end
local meta2 = liststack:get_meta():to_table().fields
local fieldcount = 0
local matching = true
for k,v in pairs(meta1) do
if not v == meta2[k] then
matching = false
break
end
fieldcount = fieldcount + 1
end
if matching and mineunit.utils.count(meta2) == fieldcount then
return true
end
end
end
return false
end
-- * `remove_item(listname, stack)`: take as many items as specified from the
-- list, returns the items that were actually removed (as an `ItemStack`)
-- -- note that any item metadata is ignored, so attempting to remove a specific
-- unique item this way will likely remove the wrong one -- to do that use
-- `set_stack` with an empty `ItemStack`.
function InvRef:remove_item(listname, stack)
error("NOT IMPLEMENTED")
end
-- * `get_location()`: returns a location compatible to
-- `minetest.get_inventory(location)`.
-- * returns `{type="undefined"}` in case location is not known
function InvRef:get_location()
error("NOT IMPLEMENTED")
end
mineunit.export_object(InvRef, {
name = "InvRef",
constructor = function(self, value)
local obj
if value == nil then
obj = {
_lists = {},
_sizes = {},
_empty = {},
}
elseif mineunit.utils.type(value) == "InvRef" then
obj = table.copy(value)
else
error("TYPE NOT IMPLEMENTED: " .. type(value))
end
setmetatable(obj, InvRef)
return obj
end,
})
--
-- MetaDataRef
--
local MetaDataRef = {}
function MetaDataRef:contains(key) return self._data[key] ~= nil end
function MetaDataRef:get(key) return self._data[key] end
function MetaDataRef:set_string(key, value)
value = (value ~= nil and value ~= "") and tostring(value) or nil
if self._data[key] ~= value then
self._data[key] = value
end
end
function MetaDataRef:get_string(key) return self._data[key] or "" end
function MetaDataRef:set_int(key, value) self:set_string(key, math.floor(value)) end
function MetaDataRef:get_int(key) return math.floor(tonumber(self._data[key]) or 0) end
function MetaDataRef:set_float(key, value) self:set_string(key, value) end
function MetaDataRef:get_float(key) return tonumber(self._data[key]) or 0 end
function MetaDataRef:to_table()
local fields = {}
for key, value in pairs(self._data) do
fields[key] = value
end
return { fields = fields }
end
function MetaDataRef:from_table(t)
assert(type(t) == "table", "MetaDataRef:from_table expects table as argument")
if t.fields then
assert(type(t.fields) == "table", "MetaDataRef:from_table expects table fields to be table")
self._data = table.copy(t.fields)
end
end
function MetaDataRef:equals(other) error("NOT IMPLEMENTED") end
-- TODO: Allows `same` assertions but in corner cases makes mod code to return true where engine would return false.
-- Requires either overriding luassert `same` (nice for users) or only allowing special assertions (not so nice).
function MetaDataRef:__eq(other)
if mineunit.utils.type(other) == "MetaDataRef" then
local fieldcount = 0
for key, value in pairs(self._data) do
if other._data[key] ~= value then
return false
end
fieldcount = fieldcount + 1
end
return mineunit.utils.count(other._data) == fieldcount
end
return false
end
mineunit.export_object(MetaDataRef, {
name = "MetaDataRef",
constructor = function(self, value)
local obj
if value == nil then
obj = {}
elseif mineunit.utils.type(value) == "MetaDataRef" then
obj = table.copy(value)
else
print(value)
error("TYPE NOT IMPLEMENTED: " .. type(value))
end
obj._data = obj._data or {}
setmetatable(obj, MetaDataRef)
return obj
end,
})
--
-- NodeMetaRef
--
local NodeMetaRef = table.copy(MetaDataRef)
function NodeMetaRef:get_inventory() return self._inventory end
function NodeMetaRef:mark_as_private(...) mineunit:info("NodeMetaRef:mark_as_private", ...) end
function NodeMetaRef:to_table()
local result = MetaDataRef.to_table(self)
result.inventory = self:get_inventory():get_lists()
return result
end
function NodeMetaRef:from_table(t)
assert(type(t) == "table", "NodeMetaRef:from_table expects table as argument")
if t.fields then
MetaDataRef.from_table(self, t)
end
if t.inventory then
assert(type(t.inventory) == "table", "NodeMetaRef:from_table expects table inventory to be table")
local inv = self:get_inventory()
for listname, list in pairs(t.inventory) do
inv:set_size(listname, #list)
inv:set_list(listname, list)
end
end
end
mineunit.export_object(NodeMetaRef, {
name = "NodeMetaRef",
constructor = function(self, value)
local obj
if value == nil then
obj = {}
elseif type(value) == "table" then
obj = table.copy(value)
else
print(value)
error("TYPE NOT IMPLEMENTED: " .. type(value))
end
obj._data = obj._data or {}
obj._inventory = obj._inventory or InvRef()
setmetatable(obj, NodeMetaRef)
return obj
end,
})