2018-10-05 21:55:40 +02:00
|
|
|
--[[
|
|
|
|
|
|
|
|
Tube Library 2
|
|
|
|
==============
|
|
|
|
|
|
|
|
Copyright (C) 2017-2018 Joachim Stolberg
|
|
|
|
|
|
|
|
LGPLv2.1+
|
|
|
|
See LICENSE.txt for more information
|
|
|
|
|
|
|
|
tube_api.lua
|
|
|
|
|
|
|
|
]]--
|
|
|
|
|
2018-10-27 17:53:07 +02:00
|
|
|
-- Version for compatibility checks, see readme.md/history
|
2018-10-28 18:45:07 +01:00
|
|
|
tubelib2.version = 0.3
|
2018-10-27 17:53:07 +02:00
|
|
|
|
2018-10-05 21:55:40 +02:00
|
|
|
-- for lazy programmers
|
|
|
|
local M = minetest.get_meta
|
|
|
|
|
2018-10-27 17:53:07 +02:00
|
|
|
local Dir2Str = {"north", "east", "south", "west", "down", "up"}
|
|
|
|
|
|
|
|
function tubelib2.dir_to_string(dir)
|
|
|
|
return Dir2Str[dir]
|
|
|
|
end
|
2018-10-05 21:55:40 +02:00
|
|
|
|
|
|
|
local function Tbl(list)
|
|
|
|
local tbl = {}
|
|
|
|
for _,item in ipairs(list) do
|
|
|
|
tbl[item] = true
|
|
|
|
end
|
|
|
|
return tbl
|
|
|
|
end
|
|
|
|
|
2018-10-28 18:45:07 +01:00
|
|
|
local function after_place_node(self, pos)
|
|
|
|
-- Check all valid positions
|
|
|
|
for dir = 1,6 do
|
|
|
|
if self.allowed_6d_dirs[dir] then
|
|
|
|
self:delete_tube_meta_data(pos, dir)
|
|
|
|
local npos, d1, d2, num = self:add_tube_dir(pos, dir)
|
|
|
|
if npos then
|
|
|
|
self.clbk_after_place_tube(self:tube_data_to_table(npos, d1, d2, num))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
local function after_dig_node(self, pos)
|
|
|
|
-- Check all valid positions
|
|
|
|
for dir = 1,6 do
|
|
|
|
if self.allowed_6d_dirs[dir] then
|
|
|
|
self:delete_tube_meta_data(pos, dir)
|
|
|
|
local npos, d1, d2, num = self:del_tube_dir(pos, dir)
|
|
|
|
if npos then
|
|
|
|
self.clbk_after_place_tube(self:tube_data_to_table(npos, d1, d2, num))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-10-05 21:55:40 +02:00
|
|
|
--
|
|
|
|
-- API Functions
|
|
|
|
--
|
|
|
|
|
|
|
|
-- Tubelib2 Class
|
|
|
|
tubelib2.Tube = {}
|
|
|
|
local Tube = tubelib2.Tube
|
|
|
|
|
|
|
|
function Tube:new(attr)
|
|
|
|
local o = {
|
|
|
|
allowed_6d_dirs = attr.allowed_6d_dirs or {true, true, true, true, true, true},
|
|
|
|
max_tube_length = attr.max_tube_length or 1000,
|
|
|
|
primary_node_names = Tbl(attr.primary_node_names or {}),
|
|
|
|
secondary_node_names = Tbl(attr.secondary_node_names or {}),
|
|
|
|
show_infotext = attr.show_infotext or false,
|
2018-10-15 20:05:41 +02:00
|
|
|
clbk_after_place_tube = attr.after_place_tube,
|
2018-10-06 21:46:38 +02:00
|
|
|
pairingList = {}, -- teleporting nodes
|
2018-10-05 21:55:40 +02:00
|
|
|
}
|
|
|
|
setmetatable(o, self)
|
|
|
|
self.__index = self
|
|
|
|
return o
|
|
|
|
end
|
|
|
|
|
2018-10-06 13:29:47 +02:00
|
|
|
-- Register (foreign) tubelib compatible nodes.
|
2018-10-05 21:55:40 +02:00
|
|
|
function Tube:add_secondary_node_names(names)
|
|
|
|
for _,name in ipairs(names) do
|
|
|
|
self.secondary_node_names[name] = true
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-10-20 19:19:44 +02:00
|
|
|
-- Check if node at given position is a tube node
|
|
|
|
-- If dir == nil then node_pos = pos
|
|
|
|
-- Function returns the new pos or nil
|
|
|
|
function Tube:primary_node(pos, dir)
|
|
|
|
local npos, node = self:get_next_node(pos, dir)
|
|
|
|
local _,_,num_conn = self:decode_param2(node.param2)
|
|
|
|
if self.primary_node_names[node.name] then
|
|
|
|
return npos
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Check if node at given position is a secondary node
|
|
|
|
-- If dir == nil then node_pos = pos
|
|
|
|
-- Function returns the new pos or nil
|
|
|
|
function Tube:secondary_node(pos, dir)
|
|
|
|
local npos, node = self:get_next_node(pos, dir)
|
|
|
|
if self.secondary_node_names[node.name] then
|
|
|
|
return npos
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-10-27 17:53:07 +02:00
|
|
|
-- Used to determine the node side to the tube connection.
|
|
|
|
-- Function returns the first found dir value
|
|
|
|
-- to a primary node.
|
|
|
|
function Tube:get_primary_dir(pos)
|
|
|
|
-- Check all valid positions
|
|
|
|
for dir = 1,6 do
|
|
|
|
if self:primary_node(pos, dir) then
|
|
|
|
return dir
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-10-14 21:28:31 +02:00
|
|
|
-- From source node to destination node via tubes.
|
|
|
|
-- pos is the source node position, dir the output dir
|
|
|
|
-- The returned pos is the destination position, dir
|
|
|
|
-- is the direction into the destination node.
|
2018-10-05 21:55:40 +02:00
|
|
|
function Tube:get_connected_node_pos(pos, dir)
|
2018-10-14 21:28:31 +02:00
|
|
|
local node = {}
|
|
|
|
if self:is_tube_head(pos, dir, node) then
|
|
|
|
local npos, ndir = self:get_peer_tube_head(node)
|
|
|
|
return vector.add(npos, tubelib2.Dir6dToVector[ndir or 0]), ndir
|
|
|
|
end
|
|
|
|
return vector.add(pos, tubelib2.Dir6dToVector[dir or 0]), dir
|
|
|
|
end
|
2018-10-05 21:55:40 +02:00
|
|
|
|
2018-10-27 17:53:07 +02:00
|
|
|
function Tube:beside(pos1, pos2)
|
|
|
|
local pos = vector.subtract(pos1, pos2)
|
|
|
|
local res = pos.x + pos.y + pos.z
|
|
|
|
return res == 1 or res == -1
|
|
|
|
end
|
|
|
|
|
2018-10-06 13:29:47 +02:00
|
|
|
-- From tube head to tube head.
|
2018-10-14 21:28:31 +02:00
|
|
|
-- pos is the tube head position, dir is the direction into the head node.
|
|
|
|
-- The returned pos is the peer head position, dir
|
|
|
|
-- is the direction out of the peer head node.
|
|
|
|
function Tube:get_tube_end_pos(pos, dir)
|
|
|
|
local node = {}
|
|
|
|
if self:is_tube_head(pos, nil, node) then
|
|
|
|
return self:get_peer_tube_head(node)
|
|
|
|
end
|
|
|
|
return pos, dir
|
2018-10-05 21:55:40 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
|
2018-10-27 17:53:07 +02:00
|
|
|
-- To be called after a secondary node is placed.
|
2018-10-15 20:05:41 +02:00
|
|
|
function Tube:after_place_node(pos, dir1, dir2)
|
2018-10-28 18:45:07 +01:00
|
|
|
if not dir1 and not dir2 then
|
|
|
|
after_place_node(self, pos)
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
2018-10-14 21:28:31 +02:00
|
|
|
self:delete_tube_meta_data(pos, dir1, dir2)
|
|
|
|
|
|
|
|
if dir1 then
|
|
|
|
local npos, d1, d2, num = self:add_tube_dir(pos, dir1)
|
|
|
|
if npos then
|
2018-10-15 20:05:41 +02:00
|
|
|
self.clbk_after_place_tube(self:tube_data_to_table(npos, d1, d2, num))
|
2018-10-14 21:28:31 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
if dir2 then
|
|
|
|
local npos, d1, d2, num = self:add_tube_dir(pos, dir2)
|
|
|
|
if npos then
|
2018-10-15 20:05:41 +02:00
|
|
|
self.clbk_after_place_tube(self:tube_data_to_table(npos, d1, d2, num))
|
2018-10-14 21:28:31 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-10-27 17:53:07 +02:00
|
|
|
|
|
|
|
-- To be called after a tube/primary node is placed.
|
2018-10-15 20:05:41 +02:00
|
|
|
function Tube:after_place_tube(pos, placer, pointed_thing)
|
2018-10-05 21:55:40 +02:00
|
|
|
local preferred_pos, fdir = self:get_player_data(placer, pointed_thing)
|
|
|
|
local dir1, dir2, num_tubes = self:determine_tube_dirs(pos, preferred_pos, fdir)
|
|
|
|
if dir1 == nil then
|
2018-10-15 20:05:41 +02:00
|
|
|
return false
|
2018-10-05 21:55:40 +02:00
|
|
|
end
|
|
|
|
|
2018-10-14 21:28:31 +02:00
|
|
|
self:delete_tube_meta_data(pos, dir1, dir2)
|
2018-10-05 21:55:40 +02:00
|
|
|
|
2018-10-15 20:05:41 +02:00
|
|
|
self.clbk_after_place_tube(self:tube_data_to_table(pos, dir1, dir2, num_tubes))
|
|
|
|
|
2018-10-05 21:55:40 +02:00
|
|
|
if num_tubes >= 1 then
|
|
|
|
local npos, d1, d2, num = self:add_tube_dir(pos, dir1)
|
|
|
|
if npos then
|
2018-10-15 20:05:41 +02:00
|
|
|
self.clbk_after_place_tube(self:tube_data_to_table(npos, d1, d2, num))
|
2018-10-05 21:55:40 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
if num_tubes >= 2 then
|
|
|
|
local npos, d1, d2, num = self:add_tube_dir(pos, dir2)
|
|
|
|
if npos then
|
2018-10-15 20:05:41 +02:00
|
|
|
self.clbk_after_place_tube(self:tube_data_to_table(npos, d1, d2, num))
|
2018-10-05 21:55:40 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-10-15 20:05:41 +02:00
|
|
|
return true
|
2018-10-05 21:55:40 +02:00
|
|
|
end
|
|
|
|
|
2018-10-14 21:28:31 +02:00
|
|
|
-- To be called after a secondary node is removed.
|
2018-10-15 20:05:41 +02:00
|
|
|
function Tube:after_dig_node(pos, dir1, dir2)
|
2018-10-28 18:45:07 +01:00
|
|
|
if not dir1 and not dir2 then
|
|
|
|
after_dig_node(self, pos)
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
2018-10-14 21:28:31 +02:00
|
|
|
self:delete_tube_meta_data(pos, dir1, dir2)
|
|
|
|
|
|
|
|
local npos, d1, d2, num = self:del_tube_dir(pos, dir1)
|
|
|
|
if npos then
|
2018-10-15 20:05:41 +02:00
|
|
|
self.clbk_after_place_tube(self:tube_data_to_table(npos, d1, d2, num))
|
2018-10-14 21:28:31 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
npos, d1, d2, num = self:del_tube_dir(pos, dir2)
|
|
|
|
if npos then
|
2018-10-15 20:05:41 +02:00
|
|
|
self.clbk_after_place_tube(self:tube_data_to_table(npos, d1, d2, num))
|
2018-10-14 21:28:31 +02:00
|
|
|
end
|
|
|
|
end
|
2018-10-05 21:55:40 +02:00
|
|
|
|
2018-10-06 13:29:47 +02:00
|
|
|
-- To be called after a tube node is removed.
|
2018-10-15 20:05:41 +02:00
|
|
|
function Tube:after_dig_tube(pos, oldnode, oldmetadata)
|
2018-10-05 21:55:40 +02:00
|
|
|
local dir1, dir2, num_tubes = self:decode_param2(oldnode.param2)
|
|
|
|
|
2018-10-14 21:28:31 +02:00
|
|
|
self:delete_tube_meta_data(pos, dir1, dir2, oldmetadata)
|
|
|
|
|
2018-10-05 21:55:40 +02:00
|
|
|
local npos, d1, d2, num = self:del_tube_dir(pos, dir1)
|
|
|
|
if npos then
|
2018-10-15 20:05:41 +02:00
|
|
|
self.clbk_after_place_tube(self:tube_data_to_table(npos, d1, d2, num))
|
2018-10-05 21:55:40 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
npos, d1, d2, num = self:del_tube_dir(pos, dir2)
|
|
|
|
if npos then
|
2018-10-15 20:05:41 +02:00
|
|
|
self.clbk_after_place_tube(self:tube_data_to_table(npos, d1, d2, num))
|
2018-10-05 21:55:40 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-10-06 13:29:47 +02:00
|
|
|
|
|
|
|
-- To be called from a repair tool in the case of a "WorldEdit" corrupted tube line.
|
2018-10-14 21:28:31 +02:00
|
|
|
function Tube:tool_repair_tubes(pos)
|
2018-10-06 13:29:47 +02:00
|
|
|
local d1, d2 = self:get_tube_dirs(pos)
|
2018-10-14 21:28:31 +02:00
|
|
|
if d1 ~= 0 then
|
2018-10-06 21:46:38 +02:00
|
|
|
self:set_2_conn_tube(pos)
|
|
|
|
local npos1, dir1, cnt1 = self:repair_tube_line(pos, d1)
|
|
|
|
local npos2, dir2, cnt2 = self:repair_tube_line(pos, d2)
|
|
|
|
self:add_meta_data(npos1, npos2, dir1, dir2, cnt1+cnt2+1)
|
|
|
|
return npos1, npos2, dir1, dir2, cnt1, cnt2
|
|
|
|
end
|
2018-10-05 21:55:40 +02:00
|
|
|
end
|
|
|
|
|
2018-10-06 13:29:47 +02:00
|
|
|
|
|
|
|
-- To be called from a repair tool in the case, tube nodes are "unbreakable".
|
2018-10-14 21:28:31 +02:00
|
|
|
function Tube:tool_remove_tube(pos, sound)
|
2018-10-05 21:55:40 +02:00
|
|
|
local dir1, dir2 = self:get_tube_dirs(pos)
|
2018-10-14 21:28:31 +02:00
|
|
|
if dir1 ~= 0 then
|
2018-10-05 21:55:40 +02:00
|
|
|
minetest.sound_play({
|
|
|
|
name=sound},{
|
|
|
|
gain=1,
|
|
|
|
max_hear_distance=5,
|
|
|
|
loop=false})
|
|
|
|
minetest.remove_node(pos)
|
2018-10-20 19:19:44 +02:00
|
|
|
|
|
|
|
self:delete_tube_meta_data(pos, dir1, dir2)
|
|
|
|
|
|
|
|
local npos, d1, d2, num = self:del_tube_dir(pos, dir1)
|
|
|
|
if npos then
|
|
|
|
self.clbk_after_place_tube(self:tube_data_to_table(npos, d1, d2, num))
|
|
|
|
end
|
|
|
|
|
|
|
|
npos, d1, d2, num = self:del_tube_dir(pos, dir2)
|
|
|
|
if npos then
|
|
|
|
self.clbk_after_place_tube(self:tube_data_to_table(npos, d1, d2, num))
|
|
|
|
end
|
2018-10-05 21:55:40 +02:00
|
|
|
end
|
|
|
|
end
|
2018-10-06 21:46:38 +02:00
|
|
|
|
|
|
|
function Tube:prepare_pairing(pos, tube_dir, sFormspec)
|
|
|
|
local meta = M(pos)
|
|
|
|
meta:set_int("tube_dir", tube_dir)
|
2018-10-14 21:28:31 +02:00
|
|
|
|
|
|
|
-- break the connection
|
|
|
|
self:delete_tube_meta_data(pos, tube_dir)
|
|
|
|
|
2018-10-06 21:46:38 +02:00
|
|
|
meta:set_string("channel", nil)
|
|
|
|
meta:set_string("infotext", "Unconnected")
|
|
|
|
meta:set_string("formspec", sFormspec)
|
|
|
|
end
|
|
|
|
|
|
|
|
function Tube:pairing(pos, channel)
|
|
|
|
if self.pairingList[channel] and pos ~= self.pairingList[channel] then
|
|
|
|
-- store peer position on both nodes
|
|
|
|
local peer_pos = self.pairingList[channel]
|
2018-10-14 21:28:31 +02:00
|
|
|
|
|
|
|
local tube_dir1 = self:store_teleport_data(pos, peer_pos)
|
|
|
|
local tube_dir2 = self:store_teleport_data(peer_pos, pos)
|
|
|
|
|
|
|
|
self:delete_tube_meta_data(pos, tube_dir1)
|
|
|
|
self:delete_tube_meta_data(peer_pos, tube_dir2)
|
|
|
|
|
2018-10-06 21:46:38 +02:00
|
|
|
self.pairingList[channel] = nil
|
|
|
|
return true
|
|
|
|
else
|
|
|
|
self.pairingList[channel] = pos
|
|
|
|
local meta = M(pos)
|
|
|
|
meta:set_string("channel", channel)
|
|
|
|
meta:set_string("infotext", "Unconnected ("..channel..")")
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-10-14 21:28:31 +02:00
|
|
|
function Tube:stop_pairing(pos, oldmetadata, sFormspec)
|
2018-10-06 21:46:38 +02:00
|
|
|
-- unpair peer node
|
2018-10-14 21:28:31 +02:00
|
|
|
if oldmetadata and oldmetadata.fields and oldmetadata.fields.tele_pos then
|
2018-10-27 17:53:07 +02:00
|
|
|
local tele_pos = minetest.string_to_pos(oldmetadata.fields.tele_pos)
|
2018-10-14 21:28:31 +02:00
|
|
|
local peer_meta = M(tele_pos)
|
2018-10-06 21:46:38 +02:00
|
|
|
if peer_meta then
|
2018-10-14 21:28:31 +02:00
|
|
|
self:delete_tube_meta_data(tele_pos, peer_meta:get_int("tube_dir"))
|
|
|
|
|
2018-10-06 21:46:38 +02:00
|
|
|
peer_meta:set_string("channel", nil)
|
|
|
|
peer_meta:set_string("tele_pos", nil)
|
|
|
|
peer_meta:set_string("formspec", sFormspec)
|
|
|
|
peer_meta:set_string("infotext", "Unconnected")
|
|
|
|
end
|
|
|
|
end
|
2018-10-14 21:28:31 +02:00
|
|
|
|
|
|
|
if oldmetadata and oldmetadata.fields then
|
|
|
|
self:delete_tube_meta_data(pos, tonumber(oldmetadata.fields.tube_dir or 0), nil, oldmetadata)
|
|
|
|
end
|
2018-10-06 21:46:38 +02:00
|
|
|
end
|