2019-03-02 14:24:48 +03:00
|
|
|
--[[
|
|
|
|
|
|
|
|
TechAge
|
|
|
|
=======
|
|
|
|
|
|
|
|
Copyright (C) 2019 Joachim Stolberg
|
|
|
|
|
2019-08-22 21:49:47 +03:00
|
|
|
GPL v3
|
2019-03-02 14:24:48 +03:00
|
|
|
See LICENSE.txt for more information
|
|
|
|
|
|
|
|
Basis functions for inter-node communication
|
|
|
|
|
|
|
|
]]--
|
|
|
|
|
|
|
|
--- for lazy programmers
|
|
|
|
local S = function(pos) if pos then return minetest.pos_to_string(pos) end end
|
|
|
|
local P = minetest.string_to_pos
|
|
|
|
local M = minetest.get_meta
|
|
|
|
|
2020-02-15 00:59:16 +03:00
|
|
|
local check_cart = minecart.check_cart
|
2019-03-02 14:24:48 +03:00
|
|
|
|
2020-01-31 21:55:10 +03:00
|
|
|
local function deserialize(s)
|
|
|
|
local tbl = {}
|
|
|
|
for line in s:gmatch("[^;]+") do
|
|
|
|
local num, spos = unpack(string.split(line, "="))
|
|
|
|
tbl[num] = {pos = minetest.string_to_pos(spos)}
|
|
|
|
end
|
|
|
|
return tbl
|
|
|
|
end
|
|
|
|
|
|
|
|
local function serialize(data)
|
|
|
|
local tbl = {}
|
|
|
|
for k,v in pairs(data) do
|
|
|
|
tbl[#tbl+1] = k.."="..minetest.pos_to_string(v.pos)
|
|
|
|
end
|
|
|
|
return table.concat(tbl, ";")
|
|
|
|
end
|
|
|
|
|
2019-03-02 14:24:48 +03:00
|
|
|
------------------------------------------------------------------
|
|
|
|
-- Data base storage
|
|
|
|
-------------------------------------------------------------------
|
|
|
|
local storage = minetest.get_mod_storage()
|
|
|
|
local NextNumber = minetest.deserialize(storage:get_string("NextNumber")) or 1
|
2020-01-31 21:55:10 +03:00
|
|
|
local Version = minetest.deserialize(storage:get_string("Version")) or 2
|
|
|
|
local Number2Pos
|
|
|
|
if Version == 1 then
|
|
|
|
Version = 2
|
|
|
|
Number2Pos = minetest.deserialize(storage:get_string("Number2Pos")) or {}
|
|
|
|
else
|
|
|
|
Number2Pos = deserialize(storage:get_string("Number2Pos"))
|
|
|
|
end
|
2019-03-02 14:24:48 +03:00
|
|
|
|
|
|
|
local function update_mod_storage()
|
2020-01-31 21:55:10 +03:00
|
|
|
local t = minetest.get_us_time()
|
2019-03-02 14:24:48 +03:00
|
|
|
minetest.log("action", "[TechAge] Store data...")
|
|
|
|
storage:set_string("NextNumber", minetest.serialize(NextNumber))
|
|
|
|
storage:set_string("Version", minetest.serialize(Version))
|
2020-01-31 21:55:10 +03:00
|
|
|
storage:set_string("Number2Pos", serialize(Number2Pos))
|
2019-03-02 14:24:48 +03:00
|
|
|
-- store data each hour
|
|
|
|
minetest.after(60*59, update_mod_storage)
|
2020-01-31 21:55:10 +03:00
|
|
|
t = minetest.get_us_time() - t
|
|
|
|
minetest.log("action", "[TechAge] Data stored. t="..t.."us")
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
|
|
|
|
minetest.register_on_shutdown(function()
|
|
|
|
update_mod_storage()
|
|
|
|
end)
|
|
|
|
|
|
|
|
-- store data after one hour
|
|
|
|
minetest.after(60*59, update_mod_storage)
|
|
|
|
|
|
|
|
-- Key2Number will be generated at runtine
|
|
|
|
local Key2Number = {}
|
|
|
|
|
|
|
|
-------------------------------------------------------------------
|
|
|
|
-- Local helper functions
|
|
|
|
-------------------------------------------------------------------
|
|
|
|
|
2019-06-08 23:57:01 +03:00
|
|
|
local function in_list(list, x)
|
|
|
|
for _, v in ipairs(list) do
|
|
|
|
if v == x then return true end
|
|
|
|
end
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
2019-03-02 14:24:48 +03:00
|
|
|
-- Localize functions to avoid table lookups (better performance).
|
|
|
|
local string_split = string.split
|
|
|
|
local NodeDef = techage.NodeDef
|
|
|
|
local Tube = techage.Tube
|
|
|
|
|
|
|
|
-- Determine position related node number for addressing purposes
|
|
|
|
local function get_number(pos)
|
|
|
|
local key = minetest.hash_node_position(pos)
|
|
|
|
if not Key2Number[key] then
|
|
|
|
Key2Number[key] = NextNumber
|
|
|
|
NextNumber = NextNumber + 1
|
|
|
|
end
|
|
|
|
return string.format("%u", Key2Number[key])
|
|
|
|
end
|
|
|
|
|
|
|
|
local function generate_Key2Number()
|
|
|
|
local key
|
|
|
|
for num,item in pairs(Number2Pos) do
|
|
|
|
key = minetest.hash_node_position(item.pos)
|
|
|
|
Key2Number[key] = num
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
local function not_protected(pos, placer_name, clicker_name)
|
|
|
|
local meta = minetest.get_meta(pos)
|
|
|
|
if meta then
|
2019-08-24 00:35:37 +03:00
|
|
|
if placer_name and not minetest.is_protected(pos, placer_name) then
|
|
|
|
if clicker_name == nil or placer_name == clicker_name then
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
if not minetest.is_protected(pos, clicker_name) then
|
2019-03-02 14:24:48 +03:00
|
|
|
return true
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
|
|
|
local function register_lbm(name, nodenames)
|
|
|
|
minetest.register_lbm({
|
2019-05-21 01:05:53 +03:00
|
|
|
label = "[TechAge] Node update",
|
2019-03-02 14:24:48 +03:00
|
|
|
name = name.."update",
|
|
|
|
nodenames = nodenames,
|
|
|
|
run_at_every_load = true,
|
|
|
|
action = function(pos, node)
|
2019-06-08 23:57:01 +03:00
|
|
|
if NodeDef[node.name] and NodeDef[node.name].on_node_load then
|
2020-01-26 01:15:44 +03:00
|
|
|
NodeDef[node.name].on_node_load(pos, node)
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
end
|
|
|
|
})
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
local DirToSide = {"B", "R", "F", "L", "D", "U"}
|
|
|
|
|
|
|
|
local function dir_to_side(dir, param2)
|
|
|
|
if dir < 5 then
|
|
|
|
dir = (((dir - 1) - (param2 % 4)) % 4) + 1
|
|
|
|
end
|
|
|
|
return DirToSide[dir]
|
|
|
|
end
|
|
|
|
|
|
|
|
local SideToDir = {B=1, R=2, F=3, L=4, D=5, U=6}
|
|
|
|
|
|
|
|
local function side_to_dir(side, param2)
|
|
|
|
local dir = SideToDir[side]
|
|
|
|
if dir < 5 then
|
|
|
|
dir = (((dir - 1) + (param2 % 4)) % 4) + 1
|
|
|
|
end
|
|
|
|
return dir
|
|
|
|
end
|
|
|
|
|
2019-03-09 21:26:15 +03:00
|
|
|
techage.side_to_outdir = side_to_dir
|
|
|
|
|
2019-03-10 13:36:00 +03:00
|
|
|
function techage.side_to_indir(side, param2)
|
|
|
|
return tubelib2.Turn180Deg[side_to_dir(side, param2)]
|
|
|
|
end
|
|
|
|
|
2019-05-01 13:01:24 +03:00
|
|
|
local function get_next_node(pos, out_dir)
|
|
|
|
local res, npos, node = Tube:compatible_node(pos, out_dir)
|
|
|
|
local in_dir = tubelib2.Turn180Deg[out_dir]
|
2019-06-08 23:57:01 +03:00
|
|
|
return res, npos, in_dir, node.name
|
2019-05-01 13:01:24 +03:00
|
|
|
end
|
|
|
|
|
2019-03-02 22:51:16 +03:00
|
|
|
local function get_dest_node(pos, out_dir)
|
|
|
|
local spos, in_dir = Tube:get_connected_node_pos(pos, out_dir)
|
|
|
|
local _,node = Tube:get_node(spos)
|
2019-06-08 23:57:01 +03:00
|
|
|
return spos, in_dir, node.name
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
|
|
|
|
local function item_handling_node(name)
|
|
|
|
local node_def = name and NodeDef[name]
|
|
|
|
if node_def then
|
|
|
|
return node_def.on_pull_item or node_def.on_push_item or node_def.is_pusher
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-02-15 00:59:16 +03:00
|
|
|
local function is_air_like(name)
|
|
|
|
local ndef = minetest.registered_nodes[name]
|
|
|
|
if ndef and ndef.buildable_to then
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
2019-03-02 14:24:48 +03:00
|
|
|
-------------------------------------------------------------------
|
|
|
|
-- API helper functions
|
|
|
|
-------------------------------------------------------------------
|
|
|
|
|
|
|
|
-- Function returns { pos, name } for the node on the given position number.
|
|
|
|
function techage.get_node_info(dest_num)
|
|
|
|
if Number2Pos[dest_num] then
|
|
|
|
return Number2Pos[dest_num]
|
|
|
|
end
|
|
|
|
return nil
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Function returns the node number from the given position or
|
|
|
|
-- nil, if no node number for this position is assigned.
|
|
|
|
function techage.get_node_number(pos)
|
|
|
|
local key = minetest.hash_node_position(pos)
|
|
|
|
local num = Key2Number[key]
|
|
|
|
if num then
|
|
|
|
num = string.format("%u", num)
|
|
|
|
if Number2Pos[num] and Number2Pos[num].name then
|
|
|
|
return num
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return nil
|
|
|
|
end
|
|
|
|
|
2020-01-26 01:15:44 +03:00
|
|
|
function techage.get_pos(pos, side)
|
|
|
|
local node = techage.get_node_lvm(pos)
|
|
|
|
local dir = nil
|
|
|
|
if node.name ~= "air" and node.name ~= "ignore" then
|
|
|
|
dir = side_to_dir(side, node.param2)
|
|
|
|
end
|
|
|
|
return tubelib2.get_pos(pos, dir)
|
|
|
|
end
|
|
|
|
|
2019-03-02 14:24:48 +03:00
|
|
|
-- Function is used for available nodes with lost numbers, only.
|
|
|
|
function techage.get_new_number(pos, name)
|
|
|
|
-- store position
|
|
|
|
local number = get_number(pos)
|
|
|
|
Number2Pos[number] = {
|
|
|
|
pos = pos,
|
|
|
|
name = name,
|
|
|
|
}
|
|
|
|
return number
|
|
|
|
end
|
|
|
|
|
2019-08-07 23:26:38 +03:00
|
|
|
-- extract ident and value from strings like "ident=value"
|
|
|
|
function techage.ident_value(s)
|
|
|
|
local ident, value = unpack(string.split(s, "=", true, 1))
|
|
|
|
return (ident or ""):trim(), (value or ""):trim()
|
|
|
|
end
|
|
|
|
|
2019-03-02 14:24:48 +03:00
|
|
|
-------------------------------------------------------------------
|
|
|
|
-- Node construction/destruction functions
|
|
|
|
-------------------------------------------------------------------
|
|
|
|
|
|
|
|
-- Add node to the techage lists.
|
|
|
|
-- Function determines and returns the node position number,
|
|
|
|
-- needed for message communication.
|
|
|
|
function techage.add_node(pos, name)
|
|
|
|
if item_handling_node(name) then
|
|
|
|
Tube:after_place_node(pos)
|
|
|
|
end
|
|
|
|
-- store position
|
|
|
|
local number = get_number(pos)
|
|
|
|
Number2Pos[number] = {
|
|
|
|
pos = pos,
|
|
|
|
name = name,
|
|
|
|
}
|
|
|
|
return number
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Function removes the node from the techage lists.
|
|
|
|
function techage.remove_node(pos)
|
|
|
|
local number = get_number(pos)
|
|
|
|
local name
|
|
|
|
if Number2Pos[number] then
|
|
|
|
name = Number2Pos[number].name
|
2020-01-31 21:55:10 +03:00
|
|
|
Number2Pos[number].name = nil
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
if item_handling_node(name) then
|
|
|
|
Tube:after_dig_node(pos)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
-------------------------------------------------------------------
|
|
|
|
-- Node register function
|
|
|
|
-------------------------------------------------------------------
|
|
|
|
|
|
|
|
-- Register node for techage communication
|
|
|
|
-- Call this function only at load time!
|
2019-06-08 23:57:01 +03:00
|
|
|
-- Param names: List of node names like {"techage:pusher_off", "techage:pusher_on"}
|
2019-03-02 14:24:48 +03:00
|
|
|
-- Param node_definition: A table according to:
|
|
|
|
-- {
|
2020-03-22 13:01:44 +03:00
|
|
|
-- on_pull_item = func(pos, in_dir, num, (opt.) item_name),
|
2019-03-09 21:26:15 +03:00
|
|
|
-- on_push_item = func(pos, in_dir, item),
|
|
|
|
-- on_unpull_item = func(pos, in_dir, item),
|
2019-08-25 19:08:30 +03:00
|
|
|
-- on_recv_message = func(pos, src, topic, payload),
|
2019-03-02 14:24:48 +03:00
|
|
|
-- on_node_load = func(pos), -- LBM function
|
2019-06-08 23:57:01 +03:00
|
|
|
-- on_transfer = func(pos, in_dir, topic, payload),
|
2019-03-02 14:24:48 +03:00
|
|
|
-- }
|
2019-06-08 23:57:01 +03:00
|
|
|
function techage.register_node(names, node_definition)
|
2019-03-02 14:24:48 +03:00
|
|
|
-- store facedir table for all known node names
|
2019-06-08 23:57:01 +03:00
|
|
|
for _,n in ipairs(names) do
|
|
|
|
NodeDef[n] = node_definition
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
if node_definition.on_pull_item or node_definition.on_push_item or
|
|
|
|
node_definition.is_pusher then
|
2019-06-08 23:57:01 +03:00
|
|
|
Tube:add_secondary_node_names(names)
|
2019-03-02 14:24:48 +03:00
|
|
|
|
2019-06-08 23:57:01 +03:00
|
|
|
for _,n in ipairs(names) do
|
2019-03-02 14:24:48 +03:00
|
|
|
techage.KnownNodes[n] = true
|
|
|
|
end
|
|
|
|
end
|
|
|
|
-- register LBM
|
|
|
|
if node_definition.on_node_load then
|
2019-06-08 23:57:01 +03:00
|
|
|
register_lbm(names[1], names)
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
-------------------------------------------------------------------
|
|
|
|
-- Send message functions
|
|
|
|
-------------------------------------------------------------------
|
|
|
|
|
2019-08-07 23:26:38 +03:00
|
|
|
function techage.not_protected(number, placer_name, clicker_name)
|
|
|
|
if Number2Pos[number] and Number2Pos[number].name then
|
|
|
|
local data = Number2Pos[number]
|
|
|
|
if data.pos then
|
|
|
|
return not_protected(data.pos, placer_name, clicker_name)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
2019-08-24 00:35:37 +03:00
|
|
|
-- Check the given list of numbers.
|
|
|
|
-- Returns true if number(s) is/are valid, point to real nodes and
|
|
|
|
-- and the nodes are not protected for the given player_name.
|
|
|
|
function techage.check_numbers(numbers, placer_name)
|
|
|
|
if numbers then
|
|
|
|
for _,num in ipairs(string_split(numbers, " ")) do
|
|
|
|
if not techage.not_protected(num, placer_name, nil) then
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
2019-08-25 19:08:30 +03:00
|
|
|
function techage.send_multi(src, numbers, topic, payload)
|
2020-02-29 21:20:54 +03:00
|
|
|
--print("send_multi", src, numbers, topic)
|
2019-03-02 14:24:48 +03:00
|
|
|
for _,num in ipairs(string_split(numbers, " ")) do
|
|
|
|
if Number2Pos[num] and Number2Pos[num].name then
|
|
|
|
local data = Number2Pos[num]
|
2019-08-07 23:26:38 +03:00
|
|
|
if data.pos and NodeDef[data.name] and NodeDef[data.name].on_recv_message then
|
2019-08-25 19:08:30 +03:00
|
|
|
NodeDef[data.name].on_recv_message(data.pos, src, topic, payload)
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-08-25 19:08:30 +03:00
|
|
|
function techage.send_single(src, number, topic, payload)
|
2020-02-29 21:20:54 +03:00
|
|
|
--print("send_single", src, number, topic)
|
2019-03-02 14:24:48 +03:00
|
|
|
if Number2Pos[number] and Number2Pos[number].name then
|
|
|
|
local data = Number2Pos[number]
|
2019-06-25 21:35:51 +03:00
|
|
|
if data.pos and NodeDef[data.name] and NodeDef[data.name].on_recv_message then
|
2019-08-25 19:08:30 +03:00
|
|
|
return NodeDef[data.name].on_recv_message(data.pos, src, topic, payload)
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
end
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
2019-06-08 23:57:01 +03:00
|
|
|
-- The destination node location is either:
|
|
|
|
-- A) a destination position, specified by pos
|
|
|
|
-- B) a neighbor position, specified by caller pos/outdir, or pos/side
|
|
|
|
-- C) a tubelib2 network connection, specified by caller pos/outdir, or pos/side
|
|
|
|
-- outdir is one of: 1..6
|
|
|
|
-- side is one of: "B", "R", "F", "L", "D", "U"
|
|
|
|
-- network is a tuebelib2 network instance
|
|
|
|
-- opt: nodenames is a table of valid the callee node names
|
|
|
|
function techage.transfer(pos, outdir, topic, payload, network, nodenames)
|
|
|
|
-- determine out-dir
|
|
|
|
if outdir and type(outdir) == "string" then
|
2019-06-09 01:25:25 +03:00
|
|
|
local param2 = techage.get_node_lvm(pos).param2
|
2019-06-08 23:57:01 +03:00
|
|
|
outdir = side_to_dir(outdir, param2)
|
|
|
|
end
|
|
|
|
-- determine destination pos
|
|
|
|
local dpos, indir
|
|
|
|
if network then
|
|
|
|
dpos, indir = network:get_connected_node_pos(pos, outdir)
|
|
|
|
else
|
2019-06-10 22:31:58 +03:00
|
|
|
dpos, indir = tubelib2.get_pos(pos, outdir), outdir
|
2019-06-08 23:57:01 +03:00
|
|
|
end
|
|
|
|
-- check node name
|
2019-06-09 01:25:25 +03:00
|
|
|
local name = techage.get_node_lvm(dpos).name
|
2019-06-08 23:57:01 +03:00
|
|
|
if nodenames and not in_list(nodenames, name) then
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
-- call "on_transfer"
|
|
|
|
local ndef = NodeDef[name]
|
|
|
|
if ndef and ndef.on_transfer then
|
|
|
|
return ndef.on_transfer(dpos, indir, topic, payload)
|
|
|
|
end
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
2019-03-02 14:24:48 +03:00
|
|
|
-------------------------------------------------------------------
|
|
|
|
-- Client side Push/Pull item functions
|
|
|
|
-------------------------------------------------------------------
|
|
|
|
|
2020-03-22 13:01:44 +03:00
|
|
|
function techage.pull_items(pos, out_dir, num, item_name)
|
2019-03-02 22:51:16 +03:00
|
|
|
local npos, in_dir, name = get_dest_node(pos, out_dir)
|
2019-05-01 13:01:24 +03:00
|
|
|
if npos and NodeDef[name] and NodeDef[name].on_pull_item then
|
2020-03-22 13:01:44 +03:00
|
|
|
return NodeDef[name].on_pull_item(npos, in_dir, num, item_name)
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-04-13 14:20:38 +03:00
|
|
|
function techage.push_items(pos, out_dir, stack, idx)
|
2019-03-02 22:51:16 +03:00
|
|
|
local npos, in_dir, name = get_dest_node(pos, out_dir)
|
2019-05-01 13:01:24 +03:00
|
|
|
if npos and NodeDef[name] and NodeDef[name].on_push_item then
|
2020-04-13 14:20:38 +03:00
|
|
|
return NodeDef[name].on_push_item(npos, in_dir, stack, idx)
|
2020-02-15 00:59:16 +03:00
|
|
|
elseif is_air_like(name) or check_cart(npos) then
|
2019-03-02 14:24:48 +03:00
|
|
|
minetest.add_item(npos, stack)
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
2019-03-09 21:26:15 +03:00
|
|
|
function techage.unpull_items(pos, out_dir, stack)
|
2019-03-02 22:51:16 +03:00
|
|
|
local npos, in_dir, name = get_dest_node(pos, out_dir)
|
2019-05-01 13:01:24 +03:00
|
|
|
if npos and NodeDef[name] and NodeDef[name].on_unpull_item then
|
2019-03-09 21:26:15 +03:00
|
|
|
return NodeDef[name].on_unpull_item(npos, in_dir, stack)
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
2019-05-01 13:01:24 +03:00
|
|
|
-------------------------------------------------------------------
|
2019-05-01 13:01:57 +03:00
|
|
|
-- Client side Push/Pull item functions for hopper like nodes
|
2019-05-01 13:01:24 +03:00
|
|
|
-- (nodes with no tube support)
|
|
|
|
-------------------------------------------------------------------
|
|
|
|
|
|
|
|
function techage.neighbour_pull_items(pos, out_dir, num)
|
|
|
|
local res, npos, in_dir, name = get_next_node(pos, out_dir)
|
|
|
|
if res and NodeDef[name] and NodeDef[name].on_pull_item then
|
|
|
|
return NodeDef[name].on_pull_item(npos, in_dir, num)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
function techage.neighbour_push_items(pos, out_dir, stack)
|
|
|
|
local res, npos, in_dir, name = get_next_node(pos, out_dir)
|
|
|
|
if res and NodeDef[name] and NodeDef[name].on_push_item then
|
|
|
|
return NodeDef[name].on_push_item(npos, in_dir, stack)
|
|
|
|
elseif name == "air" then
|
|
|
|
minetest.add_item(npos, stack)
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
|
|
|
function techage.neighbour_unpull_items(pos, out_dir, stack)
|
|
|
|
local res, npos, in_dir, name = get_next_node(pos, out_dir)
|
|
|
|
if res and NodeDef[name] and NodeDef[name].on_unpull_item then
|
|
|
|
return NodeDef[name].on_unpull_item(npos, in_dir, stack)
|
|
|
|
end
|
|
|
|
return false
|
|
|
|
end
|
2019-03-02 14:24:48 +03:00
|
|
|
|
|
|
|
-------------------------------------------------------------------
|
|
|
|
-- Server side helper functions
|
|
|
|
-------------------------------------------------------------------
|
|
|
|
|
2019-04-27 18:42:27 +03:00
|
|
|
-- Get the given number of items from the inv. The position within the list
|
2020-04-13 14:20:38 +03:00
|
|
|
-- is incremented each time so that different item stacks will be considered.
|
2019-03-02 14:24:48 +03:00
|
|
|
-- Returns nil if ItemList is empty.
|
2020-04-13 14:20:38 +03:00
|
|
|
function techage.get_items(pos, inv, listname, num)
|
2019-03-02 14:24:48 +03:00
|
|
|
if inv:is_empty(listname) then
|
|
|
|
return nil
|
|
|
|
end
|
|
|
|
local size = inv:get_size(listname)
|
2020-04-13 14:20:38 +03:00
|
|
|
local mem = techage.get_mem(pos)
|
|
|
|
mem.ta_startpos = mem.ta_startpos or 1
|
|
|
|
for idx = mem.ta_startpos, mem.ta_startpos+size do
|
2019-03-02 14:24:48 +03:00
|
|
|
idx = (idx % size) + 1
|
|
|
|
local items = inv:get_stack(listname, idx)
|
|
|
|
if items:get_count() > 0 then
|
|
|
|
local taken = items:take_item(num)
|
|
|
|
inv:set_stack(listname, idx, items)
|
2020-04-13 14:20:38 +03:00
|
|
|
mem.ta_startpos = idx
|
2019-03-02 14:24:48 +03:00
|
|
|
return taken
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return nil
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Put the given stack into the given ItemList.
|
|
|
|
-- Function returns false if ItemList is full.
|
2020-04-13 14:20:38 +03:00
|
|
|
function techage.put_items(inv, listname, item, idx)
|
|
|
|
if idx and inv and idx <= inv:get_size(listname) then
|
|
|
|
local stack = inv:get_stack(listname, idx)
|
|
|
|
if stack:item_fits(item) then
|
|
|
|
stack:add_item(item)
|
|
|
|
inv:set_stack(listname, idx, stack)
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
else
|
|
|
|
if inv and inv:room_for_item(listname, item) then
|
|
|
|
inv:add_item(listname, item)
|
|
|
|
return true
|
|
|
|
end
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
-- Return "full", "loaded", or "empty" depending
|
|
|
|
-- on the inventory load.
|
|
|
|
-- Full is returned, when no empty stack is available.
|
|
|
|
function techage.get_inv_state(inv, listname)
|
|
|
|
local state
|
|
|
|
if inv:is_empty(listname) then
|
|
|
|
state = "empty"
|
|
|
|
else
|
|
|
|
local list = inv:get_list(listname)
|
|
|
|
state = "full"
|
|
|
|
for _, item in ipairs(list) do
|
|
|
|
if item:is_empty() then
|
|
|
|
return "loaded"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return state
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
-------------------------------------------------------------------------------
|
|
|
|
-- Data Maintenance
|
|
|
|
-------------------------------------------------------------------------------
|
|
|
|
local function data_maintenance()
|
|
|
|
minetest.log("info", "[TechAge] Data maintenance started")
|
2020-01-31 21:55:10 +03:00
|
|
|
-- Remove unused positions
|
|
|
|
local tbl = table.copy(Number2Pos)
|
2019-03-02 14:24:48 +03:00
|
|
|
Number2Pos = {}
|
2020-01-31 21:55:10 +03:00
|
|
|
for num,item in pairs(tbl) do
|
|
|
|
local name = techage.get_node_lvm(item.pos).name
|
|
|
|
if NodeDef[name] then
|
2019-03-02 14:24:48 +03:00
|
|
|
Number2Pos[num] = item
|
2020-01-31 21:55:10 +03:00
|
|
|
-- add node names which are not stored as file
|
|
|
|
Number2Pos[num].name = name
|
2019-03-02 14:24:48 +03:00
|
|
|
end
|
|
|
|
end
|
|
|
|
minetest.log("info", "[TechAge] Data maintenance finished")
|
|
|
|
end
|
|
|
|
|
|
|
|
generate_Key2Number()
|
|
|
|
|
2020-01-31 21:55:10 +03:00
|
|
|
minetest.after(2, data_maintenance)
|
2019-03-02 14:24:48 +03:00
|
|
|
|
|
|
|
|