--[[ TechAge ======= Copyright (C) 2020-2023 Joachim Stolberg AGPL v3 See LICENSE.txt for more information Block fly/move library ]]-- -- for lazy programmers local M = minetest.get_meta local P2S = function(pos) if pos then return minetest.pos_to_string(pos) end end local S2P = minetest.string_to_pos local S = techage.S local flylib = {} local function lvect_add_vec(lvect1, offs) if not lvect1 or not offs then return end local lvect2 = {} for _, v in ipairs(lvect1) do lvect2[#lvect2 + 1] = vector.add(v, offs) end return lvect2 end -- yaw in radiant local function rotate(v, yaw) local sinyaw = math.sin(yaw) local cosyaw = math.cos(yaw) return {x = v.x * cosyaw - v.z * sinyaw, y = v.y, z = v.x * sinyaw + v.z * cosyaw} end -- playername is needed for carts, to attach the player to the cart entity local function set_node(item, playername) local dest_pos = item.dest_pos local name = item.name or "air" local param2 = item.param2 or 0 local nvm = techage.get_nvm(item.base_pos) local node = techage.get_node_lvm(dest_pos) local ndef1 = minetest.registered_nodes[name] local ndef2 = minetest.registered_nodes[node.name] nvm.running = false M(item.base_pos):set_string("status", S("Stopped")) if ndef1 and ndef2 then if minecart.is_cart(name) and (minecart.is_rail(dest_pos, node.name) or minecart.is_cart(name)) then local player = playername and minetest.get_player_by_name(playername) minecart.place_and_start_cart(dest_pos, {name = name, param2 = param2}, item.cartdef, player) return elseif ndef2.buildable_to then local meta = M(dest_pos) if name ~= "techage:moveblock" then minetest.set_node(dest_pos, {name=name, param2=param2}) meta:from_table(item.metadata or {}) meta:set_string("ta_move_block", "") meta:set_int("ta_door_locked", 1) end return end local meta = M(dest_pos) if not meta:contains("ta_move_block") then meta:set_string("ta_move_block", minetest.serialize({name=name, param2=param2})) return end elseif ndef1 then if name ~= "techage:moveblock" then minetest.add_item(dest_pos, ItemStack(name)) end end end ------------------------------------------------------------------------------- -- Entity monitoring ------------------------------------------------------------------------------- local queue = {} local first = 0 local last = -1 local function push(item) last = last + 1 queue[last] = item end local function pop() if first > last then return end local item = queue[first] queue[first] = nil -- to allow garbage collection first = first + 1 return item end local function monitoring() local num = last - first + 1 for _ = 1, num do local item = pop() if item.ttl >= techage.SystemTime then -- still valud push(item) elseif item.ttl ~= 0 then set_node(item) end end minetest.after(1, monitoring) end minetest.after(1, monitoring) minetest.register_on_shutdown(function() local num = last - first + 1 for _ = 1, num do local item = pop() if item.ttl ~= 0 then set_node(item) end end end) local function monitoring_add_entity(item) item.ttl = techage.SystemTime + 1 push(item) end local function monitoring_del_entity(item) -- Mark as timed out item.ttl = 0 end local function monitoring_trigger_entity(item) item.ttl = techage.SystemTime + 1 end ------------------------------------------------------------------------------- -- to_path function for the fly/move path ------------------------------------------------------------------------------- local function strsplit(text) text = text:gsub("\r\n", "\n") text = text:gsub("\r", "\n") return string.split(text, "\n", true) end local function trim(s) return (s:gsub("^%s*(.-)%s*$", "%1")) end function flylib.distance(v) return math.abs(v.x) + math.abs(v.y) + math.abs(v.z) end function flylib.to_vector(s, max_dist) local x,y,z = unpack(string.split(s, ",")) x = tonumber(x) or 0 y = tonumber(y) or 0 z = tonumber(z) or 0 if x and y and z then if not max_dist or (math.abs(x) + math.abs(y) + math.abs(z)) <= max_dist then return {x = x, y = y, z = z} end end end function flylib.to_path(s, max_dist) local tPath local dist = 0 for _, line in ipairs(strsplit(s)) do line = trim(line) line = string.split(line, "--", true, 1)[1] or "" if line ~= "" then local v = flylib.to_vector(line) if v then dist = dist + flylib.distance(v) if not max_dist or dist <= max_dist then tPath = tPath or {} tPath[#tPath + 1] = v else return tPath, S("Error: Max. length of the flight route exceeded by @1 blocks !!", dist - max_dist) end else return tPath, S("Error: Invalid path !!") end end end return tPath end local function next_path_pos(pos, lpath, idx) local offs = lpath[idx] if offs then return vector.add(pos, offs) end end local function reverse_path(lpath) local lres = {} for i = #lpath, 1, -1 do lres[#lres + 1] = vector.multiply(lpath[i], -1) end return lres end local function dest_offset(lpath) local offs = {x=0, y=0, z=0} for i = 1,#lpath do offs = vector.add(offs, lpath[i]) end return offs end ------------------------------------------------------------------------------- -- Protect the doors from being opened by hand ------------------------------------------------------------------------------- local function new_on_rightclick(old_on_rightclick) return function(pos, node, clicker, itemstack, pointed_thing) if M(pos):contains("ta_door_locked") then return itemstack end if old_on_rightclick then return old_on_rightclick(pos, node, clicker, itemstack, pointed_thing) else return itemstack end end end function flylib.protect_door_from_being_opened(name) -- Change on_rightclick function. local ndef = minetest.registered_nodes[name] if ndef then local old_on_rightclick = ndef.on_rightclick minetest.override_item(ndef.name, { on_rightclick = new_on_rightclick(old_on_rightclick) }) end end ------------------------------------------------------------------------------- -- Entity / Move / Attach / Detach ------------------------------------------------------------------------------- local MIN_SPEED = 0.4 local MAX_SPEED = 8 local CORNER_SPEED = 4 local function calc_speed(v) return math.sqrt(v.x * v.x + v.y * v.y + v.z * v.z) end -- Only the ID ist stored, not the object local function get_object_id(object) for id, entity in pairs(minetest.luaentities) do if entity.object == object then return id end end end -- determine exact position of attached entities local function obj_pos(obj) local _, _, pos = obj:get_attach() if pos then pos = vector.divide(pos, 29) return vector.add(obj:get_pos(), pos) end end -- Check access conflicts with other mods local function lock_player(player) local meta = player:get_meta() if meta:get_int("player_physics_locked") == 0 then meta:set_int("player_physics_locked", 1) meta:set_string("player_physics_locked_by", "ta_flylib") return true end return false end local function unlock_player(player) local meta = player:get_meta() if meta:get_int("player_physics_locked") == 1 then if meta:get_string("player_physics_locked_by") == "ta_flylib" then meta:set_int("player_physics_locked", 0) meta:set_string("player_physics_locked_by", "") return true end end return false end local function detach_player(player) local pos = obj_pos(player) if pos then player:set_detach() player:set_properties({visual_size = {x=1, y=1}}) player:set_pos(pos) end -- TODO: move to save position end -- Attach player/mob to given parent object (block) local function attach_single_object(parent, obj, distance) local self = parent:get_luaentity() local res = obj:get_attach() if not res then -- not already attached local yaw if obj:is_player() then yaw = obj:get_look_horizontal() else yaw = obj:get_rotation().y end -- store for later use local offs = table.copy(distance) -- Calc entity rotation, which is relative to the parent's rotation local rot = parent:get_rotation() if self.param2 >= 20 then distance = rotate(distance, 2 * math.pi - rot.y) distance.y = -distance.y distance.x = -distance.x rot.y = rot.y - yaw elseif self.param2 < 4 then distance = rotate(distance, 2 * math.pi - rot.y) rot.y = rot.y - yaw end distance = vector.multiply(distance, 29) obj:set_attach(parent, "", distance, vector.multiply(rot, 180 / math.pi)) obj:set_properties({visual_size = {x=2.9, y=2.9}}) if obj:is_player() then if lock_player(obj) then table.insert(self.players, {name = obj:get_player_name(), offs = offs}) end else table.insert(self.entities, {objID = get_object_id(obj), offs = offs}) end end end -- Attach all objects around to the parent object -- offs is the search/attach position offset -- distance (optional) is the attach distance to the center of the entity local function attach_objects(pos, offs, parent, yoffs, distance) local pos1 = vector.add(pos, offs) for _, obj in pairs(minetest.get_objects_inside_radius(pos1, 0.9)) do -- keep relative object position distance = distance or vector.subtract(obj:get_pos(), pos) local entity = obj:get_luaentity() if entity then local mod = entity.name:gmatch("(.-):")() if techage.RegisteredMobsMods[mod] then distance.y = distance.y + yoffs attach_single_object(parent, obj, distance) end elseif obj:is_player() then attach_single_object(parent, obj, distance) end end end -- Detach all attached objects from the parent object local function detach_objects(pos, self) for _, item in ipairs(self.entities or {}) do local entity = minetest.luaentities[item.objID] if entity then local obj = entity.object obj:set_detach() obj:set_properties({visual_size = {x=1, y=1}}) local pos1 = vector.add(pos, item.offs) pos1.y = pos1.y - (self.yoffs or 0) obj:set_pos(pos1) end end for _, item in ipairs(self.players or {}) do local obj = minetest.get_player_by_name(item.name) if obj then obj:set_detach() obj:set_properties({visual_size = {x=1, y=1}}) local pos1 = vector.add(pos, item.offs) pos1.y = pos1.y + 0.1 obj:set_pos(pos1) unlock_player(obj) end end self.entities = {} self.players = {} end local function entity_to_node(pos, obj) local self = obj:get_luaentity() if self and self.item then local playername = self.players and self.players[1] and self.players[1].name detach_objects(pos, self) monitoring_del_entity(self.item) minetest.after(0.1, obj.remove, obj) set_node(self.item, playername) end end -- Create a node entitiy. -- * base_pos is controller block related -- * start_pos and dest_pos are entity positions local function node_to_entity(base_pos, start_pos, dest_pos) local meta = M(start_pos) local node, metadata, cartdef node = techage.get_node_lvm(start_pos) if minecart.is_cart(node.name) then cartdef = minecart.remove_cart(start_pos) elseif meta:contains("ta_move_block") then -- Move-block stored as metadata node = minetest.deserialize(meta:get_string("ta_move_block")) metadata = {} meta:set_string("ta_move_block", "") meta:set_string("ta_block_locked", "true") elseif not meta:contains("ta_block_locked") then -- Block with other metadata node = techage.get_node_lvm(start_pos) metadata = meta:to_table() minetest.after(0.1, minetest.remove_node, start_pos) else return end local obj = minetest.add_entity(start_pos, "techage:move_item") if obj then local self = obj:get_luaentity() local rot = techage.facedir_to_rotation(node.param2) obj:set_rotation(rot) obj:set_properties({wield_item=node.name}) obj:set_armor_groups({immortal=1}) -- To be able to revert to node self.param2 = node.param2 self.item = { name = node.name, param2 = node.param2, metadata = metadata or {}, dest_pos = dest_pos, base_pos = base_pos, cartdef = cartdef, } monitoring_add_entity(self.item) -- Prepare for attachments self.players = {} self.entities = {} -- Prepare for path walk self.path_idx = 1 return obj, self.item.cartdef ~= nil end end -- move block direction local function determine_dir(pos1, pos2) local vdist = vector.subtract(pos2, pos1) local ndist = vector.length(vdist) if ndist > 0 then return vector.divide(vdist, ndist) end return {x=0, y=0, z=0} end local function move_entity(obj, next_pos, dir, is_corner) local self = obj:get_luaentity() self.next_pos = next_pos self.dir = dir if is_corner then local vel = vector.multiply(dir, math.min(CORNER_SPEED, self.max_speed)) obj:set_velocity(vel) end local acc = vector.multiply(dir, self.max_speed / 2) obj:set_acceleration(acc) end local function moveon_entity(obj, self, pos1) local pos2 = next_path_pos(pos1, self.lmove, self.path_idx) if pos2 then self.path_idx = self.path_idx + 1 local dir = determine_dir(pos1, pos2) move_entity(obj, pos2, dir, true) return true end end minetest.register_entity("techage:move_item", { initial_properties = { pointable = true, makes_footstep_sound = true, static_save = false, collide_with_objects = false, physical = false, visual = "wielditem", wield_item = "default:dirt", visual_size = {x=0.67, y=0.67, z=0.67}, selectionbox = {-0.5, -0.5, -0.5, 0.5, 0.5, 0.5}, }, on_step = function(self, dtime, moveresult) local stop_obj = function(obj, self) local next_pos = self.next_pos obj:move_to(self.next_pos, true) obj:set_acceleration({x=0, y=0, z=0}) obj:set_velocity({x=0, y=0, z=0}) self.next_pos = nil self.old_dist = nil return next_pos end if self.next_pos then local obj = self.object local pos = obj:get_pos() local dist = vector.distance(pos, self.next_pos) local speed = calc_speed(obj:get_velocity()) self.old_dist = self.old_dist or dist if self.lmove and self.lmove[self.path_idx] then if dist < 1 or dist > self.old_dist then -- change of direction local next_pos = stop_obj(obj, self) if not moveon_entity(obj, self, next_pos) then minetest.after(0.5, entity_to_node, next_pos, obj) end return end elseif dist < 0.05 or dist > self.old_dist then -- Landing local next_pos = stop_obj(obj, self) local dest_pos = self.item.dest_pos or next_pos minetest.after(0.5, entity_to_node, dest_pos, obj) return end self.old_dist = dist -- Braking or limit max speed if speed > (dist * 2) or speed > self.max_speed then speed = math.min(speed, math.max(dist * 2, MIN_SPEED)) local vel = vector.multiply(self.dir,speed) obj:set_velocity(vel) obj:set_acceleration({x=0, y=0, z=0}) end monitoring_trigger_entity(self.item) end end, }) local function is_valid_dest(pos) local node = techage.get_node_lvm(pos) if techage.is_air_like(node.name) then return true end if minecart.is_rail(pos, node.name) or minecart.is_cart(node.name) then return true end if not M(pos):contains("ta_move_block") then return true end return false end local function is_simple_node(pos) local node = techage.get_node_lvm(pos) if not minecart.is_rail(pos, node.name) then if node.name == "air" then minetest.swap_node(pos, {name = "techage:moveblock", param2 = 0}) return true end local ndef = minetest.registered_nodes[node.name] return not techage.is_air_like(node.name) and techage.can_dig_node(node.name, ndef) or minecart.is_cart(node.name) end end -- Move node from 'pos1' to the destination, calculated by means of 'lmove' -- * pos and meta are controller block related -- * lmove is the movement as a list of `moves` -- * height is move block height as value between 0 and 1 and used to calculate the offset -- for the attached object (player). local function move_node(pos, meta, pos1, lmove, max_speed, height) local pos2 = next_path_pos(pos1, lmove, 1) local offs = dest_offset(lmove) local dest_pos = vector.add(pos1, offs) -- optional for non-player objects local yoffs = meta:get_float("offset") if pos2 then local dir = determine_dir(pos1, pos2) local obj, is_cart = node_to_entity(pos, pos1, dest_pos) if obj then if is_cart then attach_objects(pos1, 0, obj, yoffs, {x = 0, y = -0.4, z = 0}) else local offs = {x=0, y=height or 1, z=0} attach_objects(pos1, offs, obj, yoffs) if dir.y == 0 then if (dir.x ~= 0 and dir.z == 0) or (dir.x == 0 and dir.z ~= 0) then attach_objects(pos1, dir, obj, yoffs) end end end local self = obj:get_luaentity() self.path_idx = 2 self.lmove = lmove self.max_speed = max_speed self.yoffs = yoffs move_entity(obj, pos2, dir) return true else return false end end end -- Move the nodes from nvm.lpos1 to nvm.lpos2 -- * nvm.lpos1 is a list of nodes -- * lmove is the movement as a list of `moves` -- * pos, meta, and nvm are controller block related --- height is move block height as value between 0 and 1 and used to calculate the offset -- for the attached object (player). local function multi_move_nodes(pos, meta, nvm, lmove, max_speed, height, move2to1) local owner = meta:get_string("owner") techage.counting_add(owner, #lmove, #nvm.lpos1 * #lmove) for idx = 1, #nvm.lpos1 do local pos1 = nvm.lpos1[idx] local pos2 = nvm.lpos2[idx] --print("multi_move_nodes", idx, P2S(pos1), P2S(pos2)) if move2to1 then pos1, pos2 = pos2, pos1 end if not minetest.is_protected(pos1, owner) and not minetest.is_protected(pos2, owner) then if is_simple_node(pos1) and is_valid_dest(pos2) then if move_node(pos, meta, pos1, lmove, max_speed, height) == false then meta:set_string("status", S("No valid node at the start position")) return false end else if not is_simple_node(pos1) then meta:set_string("status", S("No valid node at the start position")) else meta:set_string("status", S("No valid destination position")) end return false end else if minetest.is_protected(pos1, owner) then meta:set_string("status", S("Start position is protected")) else meta:set_string("status", S("Destination position is protected")) end return false end end meta:set_string("status", S("Running")) return true end -- Move the nodes from lpos1 to lpos2. -- * lpos1 is a list of nodes -- * lpos2 = lpos1 + move -- * pos and meta are controller block related -- * height is move block height as value between 0 and 1 and used to calculate the offset -- for the attached object (player). local function move_nodes(pos, meta, lpos1, move, max_speed, height) local owner = meta:get_string("owner") lpos1 = lpos1 or {} techage.counting_add(owner, #lpos1) local lpos2 = {} for idx = 1, #lpos1 do local pos1 = lpos1[idx] local pos2 = vector.add(lpos1[idx], move) lpos2[idx] = pos2 if not minetest.is_protected(pos1, owner) and not minetest.is_protected(pos2, owner) then if is_simple_node(pos1) and is_valid_dest(pos2) then move_node(pos, meta, pos1, {move}, max_speed, height) else if not is_simple_node(pos1) then meta:set_string("status", S("No valid node at the start position")) else meta:set_string("status", S("No valid destination position")) end return false, lpos1 end else if minetest.is_protected(pos1, owner) then meta:set_string("status", S("Start position is protected")) else meta:set_string("status", S("Destination position is protected")) end return false, lpos1 end end meta:set_string("status", S("Running")) return true, lpos2 end -- move2to1 is the direction and is true for 'from pos2 to pos1' -- Move path and other data is stored as meta data of pos function flylib.move_to_other_pos(pos, move2to1) local meta = M(pos) local nvm = techage.get_nvm(pos) local lmove, err = flylib.to_path(meta:get_string("path")) or {} local max_speed = meta:contains("max_speed") and meta:get_int("max_speed") or MAX_SPEED local height = meta:contains("height") and meta:get_float("height") or 1 if err or nvm.running then return false end height = techage.in_range(height, 0, 1) max_speed = techage.in_range(max_speed, MIN_SPEED, MAX_SPEED) nvm.lpos1 = nvm.lpos1 or {} local offs = dest_offset(lmove) if move2to1 then lmove = reverse_path(lmove) end -- calc destination positions nvm.lpos2 = lvect_add_vec(nvm.lpos1, offs) nvm.running = multi_move_nodes(pos, meta, nvm, lmove, max_speed, height, move2to1) nvm.moveBA = nvm.running and not move2to1 return nvm.running end -- `move` the movement as a vector function flylib.move_to(pos, move) local meta = M(pos) local nvm = techage.get_nvm(pos) local height = techage.in_range(meta:contains("height") and meta:get_float("height") or 1, 0, 1) local max_speed = meta:contains("max_speed") and meta:get_int("max_speed") or MAX_SPEED if nvm.running then return false end nvm.running, nvm.lastpos = move_nodes(pos, meta, nvm.lastpos or nvm.lpos1, move, max_speed, height) return nvm.running end function flylib.reset_move(pos) local meta = M(pos) local nvm = techage.get_nvm(pos) local height = techage.in_range(meta:contains("height") and meta:get_float("height") or 1, 0, 1) local max_speed = meta:contains("max_speed") and meta:get_int("max_speed") or MAX_SPEED if nvm.running then return false end if nvm.lpos1 and nvm.lpos1[1] then local move = vector.subtract(nvm.lpos1[1], (nvm.lastpos or nvm.lpos1)[1]) nvm.running, nvm.lastpos = move_nodes(pos, meta, nvm.lastpos or nvm.lpos1, move, max_speed, height) return nvm.running end return false end -- pos is the controller block pos -- lpos is a list of node positions to be moved -- rot is one of "l", "r", "2l", "2r" function flylib.rotate_nodes(pos, lpos, rot) local meta = M(pos) local owner = meta:get_string("owner") -- cpos is the center pos local cpos = meta:contains("center") and flylib.to_vector(meta:get_string("center")) local lpos2 = techage.rotate_around_center(lpos, rot, cpos) local param2 local nodes2 = {} techage.counting_add(owner, #lpos * 2) for i, pos1 in ipairs(lpos) do local node = techage.get_node_lvm(pos1) if rot == "l" then param2 = techage.param2_turn_right(node.param2) elseif rot == "r" then param2 = techage.param2_turn_left(node.param2) else param2 = techage.param2_turn_right(techage.param2_turn_right(node.param2)) end if not minetest.is_protected(pos1, owner) and is_simple_node(pos1) then minetest.remove_node(pos1) nodes2[#nodes2 + 1] = {pos = lpos2[i], name = node.name, param2 = param2} end end for _,item in ipairs(nodes2) do if not minetest.is_protected(item.pos, owner) and is_valid_dest(item.pos) then minetest.add_node(item.pos, {name = item.name, param2 = item.param2}) end end return lpos2 end function flylib.exchange_node(pos, name, param2) local meta = M(pos) local move_block -- consider stored "objects" if meta:contains("ta_move_block") then move_block = meta:get_string("ta_move_block") end minetest.swap_node(pos, {name = name, param2 = param2}) if move_block then meta:set_string("ta_move_block", move_block) end end function flylib.remove_node(pos) local meta = M(pos) local move_block -- consider stored "objects" if meta:contains("ta_move_block") then move_block = meta:get_string("ta_move_block") end minetest.remove_node(pos) if move_block then local node = minetest.deserialize(move_block) minetest.add_node(pos, node) meta:set_string("ta_move_block", "") end end minetest.register_node("techage:moveblock", { description = "Techage Move Block", drawtype = "normal", tiles = {"techage_invisible.png"}, sunlight_propagates = true, walkable = false, pointable = false, diggable = false, buildable_to = true, floodable = true, is_ground_content = false, groups = {not_in_creative_inventory=1}, drop = "", }) minetest.register_on_joinplayer(function(player) unlock_player(player) end) minetest.register_on_leaveplayer(function(player) if unlock_player(player) then detach_player(player) end end) minetest.register_on_dieplayer(function(player) if unlock_player(player) then detach_player(player) end end) techage.flylib = flylib