local time = tonumber(minetest.setting_get("remove_items")) if not time then time = 1800 -- set to 30 minutes the time before remove a dropped item end unwalkable_nodes = {} minetest.after(0, function() for itemname, node in pairs(minetest.registered_nodes) do if node.walkable == false then table.insert(unwalkable_nodes, 1, itemname) end end end) local get_flowing_dir = function(self) local pos = self.object:getpos() local param2 = minetest.get_node(pos).param2 for i,d in ipairs({-1, 1, -1, 1}) do if i<3 then pos.x = pos.x+d else pos.z = pos.z+d end local name = minetest.get_node(pos).name local par2 = minetest.get_node(pos).param2 if name == "default:water_flowing" and par2 < param2 then return pos end if i<3 then pos.x = pos.x-d else pos.z = pos.z-d end end return nil end minetest.register_entity(":__builtin:item", { initial_properties = { hp_max = 1, physical = true, collisionbox = {-0.175, -0.175, -0.175, 0.175, 0.175, 0.175}, collide_with_objects = false, visual = "sprite", visual_size = {x=0.5, y=0.5}, textures = {""}, spritediv = {x=1, y=1}, initial_sprite_basepos = {x=0, y=0}, is_visible = false, timer = 0, }, itemstring = "", physical_state = true, set_item = function(self, itemstring) self.itemstring = itemstring local stack = ItemStack(itemstring) local itemtable = stack:to_table() local itemname = nil if itemtable then itemname = stack:to_table().name end local item_texture = nil local item_type = "" if minetest.registered_items[itemname] then item_texture = minetest.registered_items[itemname].inventory_image item_type = minetest.registered_items[itemname].type end local prop = { is_visible = true, visual = "wielditem", textures = {(itemname or "unknown_item.png")}, visual_size = {x=0.175, y=0.175}, automatic_rotate = math.pi * 0.5, } self.object:set_properties(prop) end, get_staticdata = function(self) --return self.itemstring return minetest.serialize({ itemstring = self.itemstring, always_collect = self.always_collect, timer = self.timer, }) end, on_activate = function(self, staticdata, dtime_s) if string.sub(staticdata, 1, string.len("return")) == "return" then local data = minetest.deserialize(staticdata) if data and type(data) == "table" then self.itemstring = data.itemstring self.always_collect = data.always_collect self.timer = data.timer if not self.timer then self.timer = 0 end self.timer = self.timer+dtime_s end else self.itemstring = staticdata end self.object:set_armor_groups({immortal=1}) self.object:setvelocity({x=0, y=2, z=0}) self.object:setacceleration({x=0, y=-10, z=0}) self:set_item(self.itemstring) end, on_step = function(self, dtime) if not self.timer then self.timer = 0 end self.timer = self.timer + dtime if self.itemstring == "" or (time ~= 0 and (self.timer > time)) then self.object:remove() return end local p = self.object:getpos() local name = minetest.get_node(p).name if not minetest.registered_nodes[name] then return end if minetest.registered_nodes[name].damage_per_second > 0 or name == "maptools:igniter" then minetest.sound_play("builtin_item_lava", {pos = p, gain = 0.5}) self.object:remove() return end --[[ if name == "default:water_source" then self.object:setacceleration({x = 0, y = 4, z = 0}) else self.object:setacceleration({x = 0, y = -10, z = 0}) end --]] if minetest.registered_nodes[name].liquidtype == "flowing" then local vec = get_flowing_dir(self) if vec then local v = self.object:getvelocity() if v and vec.x-p.x > 0 then self.object:setacceleration({x = 0, y = 0, z = 0}) self.object:setvelocity({x = 1, y = -0.22, z = 0}) elseif v and vec.x-p.x < 0 then self.object:setacceleration({x = 0, y = 0, z = 0}) self.object:setvelocity({x = -1, y = -0.22, z = 0}) elseif v and vec.z-p.z > 0 then self.object:setacceleration({x = 0, y = 0, z = 0}) self.object:setvelocity({x = 0, y = -0.22, z = 1}) elseif v and vec.z-p.z < 0 then self.object:setacceleration({x = 0, y = 0, z = 0}) self.object:setvelocity({x = 0, y = -0.22, z = -1}) end self.object:setacceleration({x = 0, y = -10, z = 0}) self.physical_state = true self.object:set_properties({ physical = true }) return end end p.y = p.y - 0.3 local nn = minetest.get_node(p).name -- If node is not registered or node is walkably solid. if not minetest.registered_nodes[nn] or minetest.registered_nodes[nn].walkable then if self.physical_state then self.object:setvelocity({x=0,y=0,z=0}) self.object:setacceleration({x=0, y=0, z=0}) self.physical_state = false self.object:set_properties({ physical = false }) end else if not self.physical_state then self.object:setvelocity({x=0,y=0,z=0}) self.object:setacceleration({x=0, y=-10, z=0}) self.physical_state = true self.object:set_properties({ physical = true }) end end -- Eject if not walkable local upnode = minetest.get_node({x = p.x, y = math.ceil(p.y), z = p.z}).name if minetest.registered_nodes[upnode] and minetest.registered_nodes[upnode].walkable then local minp, maxp = {x=p.x-1, y=math.ceil(p.y), z=p.z-1}, {x=p.x+1, y=math.ceil(p.y)+1, z=p.z+1} local nodes = minetest.find_nodes_in_area(minp, maxp, unwalkable_nodes) if table.getn(nodes) > 0 then self.object:setpos(nodes[math.random(1,#nodes)]) end end end, --[[ This causes a duplication glitch if a player walks upon an item and clicks on it at the same time. on_punch = function(self, hitter) if self.itemstring ~= "" then local left = hitter:get_inventory():add_item("main", self.itemstring) if not left:is_empty() then self.itemstring = left:to_string() return end end self.object:remove() end, --]] }) if minetest.setting_get("log_mods") then minetest.log("action", "[builtin_item] loaded.") end