2016-11-26 08:20:23 +01:00
|
|
|
local hudHandlers = {}
|
|
|
|
|
2019-12-28 02:43:12 +01:00
|
|
|
---plants to place in openfarming
|
|
|
|
local plants = {
|
2020-10-14 22:33:09 +02:00
|
|
|
["farming:beetroot"]="air", ["farming:blueberries"]="air", ["farming:cabbage"]="air",
|
|
|
|
["farming:carrot"]="air", ["farming:chili_pepper"]="air", ["farming:coffee_beans"]="air",
|
|
|
|
["farming:corn"]="air", ["farming:cucumber"]="air", ["farming:garlic_clove"]="air",
|
|
|
|
["farming:melon_slice"]="air", ["farming:onion"]="air", ["default:papyrus"]="air",
|
|
|
|
["farming:pea_pod"]="air", ["farming:peppercorn"]="air", ["farming:pineapple_top"]="air",
|
2019-12-28 02:43:12 +01:00
|
|
|
["farming:potato"]="air", ["farming:pumpkin_slice"]="air", ["farming:raspberries"]="air",
|
2020-10-14 22:33:09 +02:00
|
|
|
["farming:rhubarb"]="air",
|
|
|
|
["farming:seed_barley"]="air", ["farming:seed_cotton"]="air", ["farming:seed_hemp"]="air",
|
|
|
|
["farming:seed_mint"]="air", ["farming:seed_oat"]="air", ["farming:seed_rice"]="air",
|
|
|
|
["farming:seed_rye"]="air", ["farming:seed_wheat"]="air",
|
|
|
|
["farming:tomato"]="air",
|
|
|
|
["farming:trellis"]="air", ["farming:grapes"]="farming:trellis",
|
|
|
|
["farming:beanpole"]="air", ["farming:beans"]="farming:beanpole",
|
|
|
|
["morefarming:seed_wildcarrot"]="air", ["morefarming:seed_teosinte"]="air",
|
|
|
|
["morefarming:seed_carrot"]="air", ["morefarming:seed_corn"]="air",
|
2019-12-28 02:43:12 +01:00
|
|
|
}
|
2019-04-02 05:28:50 +02:00
|
|
|
|
2024-03-22 22:26:12 +01:00
|
|
|
areas.registered_protection_conditions = {}
|
2019-04-02 05:28:50 +02:00
|
|
|
areas.registered_on_adds = {}
|
|
|
|
areas.registered_on_removes = {}
|
|
|
|
areas.registered_on_moves = {}
|
|
|
|
|
2024-03-22 22:26:12 +01:00
|
|
|
areas.callback_origins = {}
|
|
|
|
|
|
|
|
function areas:registerProtectionCondition(func)
|
|
|
|
table.insert(areas.registered_protection_conditions, func)
|
|
|
|
local debug_info = debug.getinfo(func, "S")
|
|
|
|
areas.callback_origins[func] = {
|
|
|
|
mod = core.get_current_modname() or "??",
|
|
|
|
source = debug_info.short_src or "??",
|
|
|
|
line = debug_info.linedefined or "??"
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2019-04-02 05:28:50 +02:00
|
|
|
function areas:registerOnAdd(func)
|
|
|
|
table.insert(areas.registered_on_adds, func)
|
|
|
|
end
|
|
|
|
|
|
|
|
function areas:registerOnRemove(func)
|
|
|
|
table.insert(areas.registered_on_removes, func)
|
|
|
|
end
|
|
|
|
|
|
|
|
function areas:registerOnMove(func)
|
|
|
|
table.insert(areas.registered_on_moves, func)
|
|
|
|
end
|
|
|
|
|
2016-11-26 08:20:23 +01:00
|
|
|
--- Adds a function as a HUD handler, it will be able to add items to the Areas HUD element.
|
|
|
|
function areas:registerHudHandler(handler)
|
|
|
|
table.insert(hudHandlers, handler)
|
|
|
|
end
|
|
|
|
|
|
|
|
function areas:getExternalHudEntries(pos)
|
|
|
|
local areas = {}
|
|
|
|
for _, func in pairs(hudHandlers) do
|
|
|
|
func(pos, areas)
|
|
|
|
end
|
|
|
|
return areas
|
|
|
|
end
|
2013-09-03 01:16:14 +02:00
|
|
|
|
2015-07-09 11:42:13 +02:00
|
|
|
--- Returns a list of areas that include the provided position.
|
2014-01-16 02:10:24 +01:00
|
|
|
function areas:getAreasAtPos(pos)
|
2015-07-09 11:42:13 +02:00
|
|
|
local res = {}
|
2016-11-26 08:20:23 +01:00
|
|
|
|
2015-07-09 11:42:13 +02:00
|
|
|
if self.store then
|
|
|
|
local a = self.store:get_areas_for_pos(pos, false, true)
|
|
|
|
for store_id, store_area in pairs(a) do
|
|
|
|
local id = tonumber(store_area.data)
|
|
|
|
res[id] = self.areas[id]
|
|
|
|
end
|
|
|
|
else
|
|
|
|
local px, py, pz = pos.x, pos.y, pos.z
|
|
|
|
for id, area in pairs(self.areas) do
|
|
|
|
local ap1, ap2 = area.pos1, area.pos2
|
|
|
|
if
|
|
|
|
(px >= ap1.x and px <= ap2.x) and
|
|
|
|
(py >= ap1.y and py <= ap2.y) and
|
|
|
|
(pz >= ap1.z and pz <= ap2.z) then
|
|
|
|
res[id] = area
|
|
|
|
end
|
2014-01-16 02:10:24 +01:00
|
|
|
end
|
|
|
|
end
|
2015-07-09 11:42:13 +02:00
|
|
|
return res
|
|
|
|
end
|
|
|
|
|
|
|
|
--- Returns areas that intersect with the passed area.
|
|
|
|
function areas:getAreasIntersectingArea(pos1, pos2)
|
|
|
|
local res = {}
|
|
|
|
if self.store then
|
|
|
|
local a = self.store:get_areas_in_area(pos1, pos2,
|
|
|
|
true, false, true)
|
|
|
|
for store_id, store_area in pairs(a) do
|
|
|
|
local id = tonumber(store_area.data)
|
|
|
|
res[id] = self.areas[id]
|
|
|
|
end
|
|
|
|
else
|
|
|
|
self:sortPos(pos1, pos2)
|
|
|
|
local p1x, p1y, p1z = pos1.x, pos1.y, pos1.z
|
|
|
|
local p2x, p2y, p2z = pos2.x, pos2.y, pos2.z
|
|
|
|
for id, area in pairs(self.areas) do
|
|
|
|
local ap1, ap2 = area.pos1, area.pos2
|
|
|
|
if
|
|
|
|
(ap1.x <= p2x and ap2.x >= p1x) and
|
|
|
|
(ap1.y <= p2y and ap2.y >= p1y) and
|
|
|
|
(ap1.z <= p2z and ap2.z >= p1z) then
|
|
|
|
-- Found an intersecting area.
|
|
|
|
res[id] = area
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return res
|
2014-01-16 02:10:24 +01:00
|
|
|
end
|
|
|
|
|
2024-09-11 22:00:47 +02:00
|
|
|
-- Returns smallest area at position and its id or nil.
|
|
|
|
-- If multiple areas have the same volume, larger id takes precedence.
|
|
|
|
function areas:getSmallestAreaAtPos(pos)
|
|
|
|
local smallest_area, smallest_id, volume
|
|
|
|
local smallest_volume = math.huge
|
|
|
|
for id, area in pairs(self:getAreasAtPos(pos)) do
|
|
|
|
volume = (area.pos2.x - area.pos1.x + 1)
|
|
|
|
* (area.pos2.y - area.pos1.y + 1)
|
|
|
|
* (area.pos2.z - area.pos1.z + 1)
|
|
|
|
if smallest_volume >= volume then
|
|
|
|
smallest_area = area
|
|
|
|
smallest_id = id
|
|
|
|
smallest_volume = volume
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return smallest_area, smallest_id
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Checks if the area is unprotected, open, owned by player
|
|
|
|
-- or player is part of faction of smallest area at position.
|
2013-09-03 01:16:14 +02:00
|
|
|
function areas:canInteract(pos, name)
|
2014-07-13 01:06:05 +02:00
|
|
|
if minetest.check_player_privs(name, self.adminPrivs) then
|
2013-09-03 01:16:14 +02:00
|
|
|
return true
|
|
|
|
end
|
2024-09-11 22:00:47 +02:00
|
|
|
local area = self:getSmallestAreaAtPos(pos)
|
|
|
|
-- No area, player owns it or area is open
|
|
|
|
if not area
|
|
|
|
or area.owner == name
|
|
|
|
or area.open
|
|
|
|
then
|
|
|
|
return true
|
2024-09-15 08:24:21 +02:00
|
|
|
elseif area.openfarming then
|
|
|
|
-- if area is openfarming
|
|
|
|
local player = minetest.get_player_by_name(name)
|
|
|
|
local node = minetest.get_node(pos).name
|
|
|
|
if player and minetest.registered_nodes[node] then
|
|
|
|
local wstack = player:get_wielded_item():get_name()
|
|
|
|
if wstack == "" then wstack = "hand" end
|
2019-12-28 02:43:12 +01:00
|
|
|
|
2024-09-15 08:24:21 +02:00
|
|
|
--on_dig
|
|
|
|
if minetest.get_item_group(node, "plant") == 1 and (wstack == "hand" or minetest.registered_tools[wstack]) then
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
|
|
|
|
--on_place
|
|
|
|
if plants[wstack] ~= nil and plants[wstack] == node then
|
|
|
|
return true
|
2019-12-28 02:43:12 +01:00
|
|
|
end
|
2022-07-09 02:49:12 +02:00
|
|
|
end
|
2024-09-15 08:24:21 +02:00
|
|
|
end
|
2024-09-11 22:00:47 +02:00
|
|
|
elseif areas.factions_available and area.faction_open then
|
|
|
|
if (factions.version or 0) < 2 then
|
|
|
|
local faction_name = factions.get_player_faction(name)
|
|
|
|
if faction_name then
|
2020-09-05 18:49:17 +02:00
|
|
|
for _, fname in ipairs(area.faction_open or {}) do
|
2024-09-11 22:00:47 +02:00
|
|
|
if faction_name == fname then
|
2020-09-05 18:49:17 +02:00
|
|
|
return true
|
|
|
|
end
|
|
|
|
end
|
2019-09-21 15:33:42 +02:00
|
|
|
end
|
2024-09-11 22:00:47 +02:00
|
|
|
else
|
|
|
|
for _, fname in ipairs(area.faction_open or {}) do
|
|
|
|
if factions.player_is_in_faction(fname, name) then
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
end
|
2013-09-03 01:16:14 +02:00
|
|
|
end
|
|
|
|
end
|
2024-09-11 22:00:47 +02:00
|
|
|
return false
|
2013-09-03 01:16:14 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
-- Returns a table (list) of all players that own an area
|
|
|
|
function areas:getNodeOwners(pos)
|
|
|
|
local owners = {}
|
2014-01-16 02:10:24 +01:00
|
|
|
for _, area in pairs(self:getAreasAtPos(pos)) do
|
|
|
|
table.insert(owners, area.owner)
|
2013-09-03 01:16:14 +02:00
|
|
|
end
|
|
|
|
return owners
|
|
|
|
end
|
|
|
|
|
2014-07-12 22:37:54 +02:00
|
|
|
--- Checks if the area intersects with an area that the player can't interact in.
|
|
|
|
-- Note that this fails and returns false when the specified area is fully
|
2014-11-14 19:21:09 +01:00
|
|
|
-- owned by the player, but with multiple protection zones, none of which
|
2014-07-12 22:37:54 +02:00
|
|
|
-- cover the entire checked area.
|
2020-03-22 21:25:05 +01:00
|
|
|
-- @param name (optional) Player name. If not specified checks for any intersecting areas.
|
2015-07-09 11:42:13 +02:00
|
|
|
-- @param allow_open Whether open areas should be counted as if they didn't exist.
|
2014-07-12 22:37:54 +02:00
|
|
|
-- @return Boolean indicating whether the player can interact in that area.
|
2015-07-09 11:42:13 +02:00
|
|
|
-- @return Un-owned intersecting area ID, if found.
|
2014-11-14 19:21:09 +01:00
|
|
|
function areas:canInteractInArea(pos1, pos2, name, allow_open)
|
2014-07-13 01:06:05 +02:00
|
|
|
if name and minetest.check_player_privs(name, self.adminPrivs) then
|
|
|
|
return true
|
|
|
|
end
|
2015-07-09 11:42:13 +02:00
|
|
|
self:sortPos(pos1, pos2)
|
|
|
|
|
|
|
|
-- Intersecting non-owned area ID, if found.
|
|
|
|
local blocking_area = nil
|
|
|
|
|
|
|
|
local areas = self:getAreasIntersectingArea(pos1, pos2)
|
|
|
|
for id, area in pairs(areas) do
|
|
|
|
-- First check for a fully enclosing owned area.
|
|
|
|
-- A little optimization: isAreaOwner isn't necessary
|
|
|
|
-- here since we're iterating over all relevant areas.
|
|
|
|
if area.owner == name and
|
|
|
|
self:isSubarea(pos1, pos2, id) then
|
|
|
|
return true
|
2014-07-12 22:37:54 +02:00
|
|
|
end
|
2015-07-09 11:42:13 +02:00
|
|
|
|
|
|
|
-- Then check for intersecting non-owned (blocking) areas.
|
|
|
|
-- We don't bother with this check if we've already found a
|
|
|
|
-- blocking area, as the check is somewhat expensive.
|
|
|
|
-- The area blocks if the area is closed or open areas aren't
|
|
|
|
-- acceptable to the caller, and the area isn't owned.
|
|
|
|
-- Note: We can't return directly here, because there might be
|
|
|
|
-- an exclosing owned area that we haven't gotten to yet.
|
|
|
|
if not blocking_area and
|
|
|
|
(not allow_open or not area.open) and
|
|
|
|
(not name or not self:isAreaOwner(id, name)) then
|
|
|
|
blocking_area = id
|
2014-07-12 22:37:54 +02:00
|
|
|
end
|
|
|
|
end
|
2015-07-09 11:42:13 +02:00
|
|
|
|
|
|
|
if blocking_area then
|
|
|
|
return false, blocking_area
|
|
|
|
end
|
|
|
|
|
|
|
|
-- There are no intersecting areas or they are only partially
|
|
|
|
-- intersecting areas and they are all owned by the player.
|
2014-07-12 22:37:54 +02:00
|
|
|
return true
|
|
|
|
end
|