EinsDreiDreiSieben/mods/leads/knots.lua

166 lines
5.3 KiB
Lua

--[[
Leads — Adds leads for transporting animals to Minetest.
Copyright © 2023, Silver Sandstone <@SilverSandstone@craftodon.social>
Permission is hereby granted, free of charge, to any person obtaining a
copy of this software and associated documentation files (the "Software"),
to deal in the Software without restriction, including without limitation
the rights to use, copy, modify, merge, publish, distribute, sublicense,
and/or sell copies of the Software, and to permit persons to whom the
Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
DEALINGS IN THE SOFTWARE.
]]
--- Knot entity definition.
-- @module knots
local S = leads.S;
--- Ties a lead to a post.
-- @type KnotEntity
leads.KnotEntity = {};
leads.KnotEntity.description = S'Lead Knot';
leads.KnotEntity._leads_immobile = true;
leads.KnotEntity._leads_leashable = true;
leads.KnotEntity.initial_properties =
{
visual = 'mesh';
visual_size = vector.new(10, 10, 10);
mesh = 'leads_lead_knot.obj';
textures = {'leads_lead_knot.png'};
physical = false;
selectionbox = {-3/16, -4/16, -3/16, 3/16, 4/16, 3/16};
};
--- Spawns or loads a knot.
function leads.KnotEntity:on_activate(staticdata, dtime_s)
self.num_connections = 0;
local data = minetest.deserialize(staticdata);
if data then
self.num_connections = data.num_connections or 0;
end;
self.object:set_armor_groups{fleshy = 0};
end;
--- Steps the knot.
function leads.KnotEntity:on_step(dtime, moveresult)
if self.num_connections <= 0 then
self.object:remove();
return;
end;
local node = minetest.get_node(self.object:get_pos());
if not leads.is_knottable(node.name) then
self.object:remove();
return;
end;
end;
--- Returns the knot's state as a table.
function leads.KnotEntity:get_staticdata()
local data = {num_connections = self.num_connections};
return minetest.serialize(data);
end;
--- Handles the knot being punched.
function leads.KnotEntity:on_punch(puncher, time_from_last_punch, tool_capabilities, dir, damage)
-- Check protection:
if leads.settings.respect_protection and not minetest.check_player_privs(puncher, 'protection_bypass') then
local pos = self.object:get_pos():round();
local name = puncher and puncher:get_player_name() or '';
if minetest.is_protected(pos, name) then
minetest.record_protection_violation(pos, name);
return true;
end;
end;
-- Check if the puncher is holding Shift, and get a list of leads if necessary:
local break_leads = puncher and puncher:get_player_control().sneak;
local connected_leads;
if break_leads then
connected_leads = {};
for lead in leads.find_connected_leads(self.object, true, true) do
table.insert(connected_leads, lead);
end;
else
minetest.sound_play(leads.sounds.remove, {pos = self.object:get_pos()}, true);
end;
-- Transfer all connected leads to the puncher:
if puncher then
self:transfer_leads(puncher);
end;
-- Remove leads if holding Shift:
if break_leads then
for __, lead in ipairs(connected_leads) do
lead:get_luaentity():break_lead(puncher);
end;
end;
-- Prevent the player from breaking the post:
local name = puncher and puncher:get_player_name();
if name and name ~= '' then
leads.util.block_player_interaction(name, 0.25);
end;
-- Remove this knot:
self.object:remove();
return true;
end;
--- Handles the knot being right-clicked.
function leads.KnotEntity:on_rightclick(clicker)
local pos = self.object:get_pos();
leads.knot(clicker, pos);
end;
--- Handles the knot being interacted with while holding a lead item.
function leads.KnotEntity:_leads_on_interact(itemstack, user, pointed_thing, is_punch)
if is_punch then
self:on_punch(user);
return true, nil;
end;
return false, nil;
end;
--- Transfers all leads attached to this knot to another object.
-- @param leader [ObjectRef] The new leader.
function leads.KnotEntity:transfer_leads(leader)
for lead, is_leader in leads.find_connected_leads(self.object, true, true) do
local entity = lead:get_luaentity();
if not is_leader then
entity:reverse();
end;
entity:set_leader(leader);
end;
end;
--- Called when a lead is added.
function leads.KnotEntity:_leads_lead_add(lead, is_leader)
self.num_connections = self.num_connections + 1;
end;
--- Called when a lead is removed.
function leads.KnotEntity:_leads_lead_remove(lead, is_leader)
self.num_connections = self.num_connections - 1;
end;
minetest.register_entity('leads:knot', leads.KnotEntity);