mirror of
				https://github.com/luanti-org/luanti.git
				synced 2025-10-28 22:25:20 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			240 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			Lua
		
	
	
	
	
	
			
		
		
	
	
			240 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			Lua
		
	
	
	
	
	
| -- Minetest: builtin/item_s.lua
 | |
| -- The distinction of what goes here is a bit tricky, basically it's everything
 | |
| -- that does not (directly or indirectly) need access to ServerEnvironment,
 | |
| -- Server or writable access to IGameDef on the engine side.
 | |
| -- (The '_s' stands for standalone.)
 | |
| 
 | |
| local builtin_shared = ...
 | |
| 
 | |
| --
 | |
| -- Item definition helpers
 | |
| --
 | |
| 
 | |
| function core.inventorycube(img1, img2, img3)
 | |
| 	img2 = img2 or img1
 | |
| 	img3 = img3 or img1
 | |
| 	return "[inventorycube"
 | |
| 			.. "{" .. img1:gsub("%^", "&")
 | |
| 			.. "{" .. img2:gsub("%^", "&")
 | |
| 			.. "{" .. img3:gsub("%^", "&")
 | |
| end
 | |
| 
 | |
| function core.dir_to_facedir(dir, is6d)
 | |
| 	--account for y if requested
 | |
| 	if is6d and math.abs(dir.y) > math.abs(dir.x) and math.abs(dir.y) > math.abs(dir.z) then
 | |
| 
 | |
| 		--from above
 | |
| 		if dir.y < 0 then
 | |
| 			if math.abs(dir.x) > math.abs(dir.z) then
 | |
| 				if dir.x < 0 then
 | |
| 					return 19
 | |
| 				else
 | |
| 					return 13
 | |
| 				end
 | |
| 			else
 | |
| 				if dir.z < 0 then
 | |
| 					return 10
 | |
| 				else
 | |
| 					return 4
 | |
| 				end
 | |
| 			end
 | |
| 
 | |
| 		--from below
 | |
| 		else
 | |
| 			if math.abs(dir.x) > math.abs(dir.z) then
 | |
| 				if dir.x < 0 then
 | |
| 					return 15
 | |
| 				else
 | |
| 					return 17
 | |
| 				end
 | |
| 			else
 | |
| 				if dir.z < 0 then
 | |
| 					return 6
 | |
| 				else
 | |
| 					return 8
 | |
| 				end
 | |
| 			end
 | |
| 		end
 | |
| 
 | |
| 	--otherwise, place horizontally
 | |
| 	elseif math.abs(dir.x) > math.abs(dir.z) then
 | |
| 		if dir.x < 0 then
 | |
| 			return 3
 | |
| 		else
 | |
| 			return 1
 | |
| 		end
 | |
| 	else
 | |
| 		if dir.z < 0 then
 | |
| 			return 2
 | |
| 		else
 | |
| 			return 0
 | |
| 		end
 | |
| 	end
 | |
| end
 | |
| 
 | |
| -- Table of possible dirs
 | |
| local facedir_to_dir = {
 | |
| 	vector.new( 0,  0,  1),
 | |
| 	vector.new( 1,  0,  0),
 | |
| 	vector.new( 0,  0, -1),
 | |
| 	vector.new(-1,  0,  0),
 | |
| 	vector.new( 0, -1,  0),
 | |
| 	vector.new( 0,  1,  0),
 | |
| }
 | |
| -- Mapping from facedir value to index in facedir_to_dir.
 | |
| local facedir_to_dir_map = {
 | |
| 	[0]=1, 2, 3, 4,
 | |
| 	5, 2, 6, 4,
 | |
| 	6, 2, 5, 4,
 | |
| 	1, 5, 3, 6,
 | |
| 	1, 6, 3, 5,
 | |
| 	1, 4, 3, 2,
 | |
| }
 | |
| function core.facedir_to_dir(facedir)
 | |
| 	return facedir_to_dir[facedir_to_dir_map[facedir % 32]]
 | |
| end
 | |
| 
 | |
| function core.dir_to_fourdir(dir)
 | |
| 	if math.abs(dir.x) > math.abs(dir.z) then
 | |
| 		if dir.x < 0 then
 | |
| 			return 3
 | |
| 		else
 | |
| 			return 1
 | |
| 		end
 | |
| 	else
 | |
| 		if dir.z < 0 then
 | |
| 			return 2
 | |
| 		else
 | |
| 			return 0
 | |
| 		end
 | |
| 	end
 | |
| end
 | |
| 
 | |
| function core.fourdir_to_dir(fourdir)
 | |
| 	return facedir_to_dir[facedir_to_dir_map[fourdir % 4]]
 | |
| end
 | |
| 
 | |
| function core.dir_to_wallmounted(dir)
 | |
| 	if math.abs(dir.y) > math.max(math.abs(dir.x), math.abs(dir.z)) then
 | |
| 		if dir.y < 0 then
 | |
| 			return 1
 | |
| 		else
 | |
| 			return 0
 | |
| 		end
 | |
| 	elseif math.abs(dir.x) > math.abs(dir.z) then
 | |
| 		if dir.x < 0 then
 | |
| 			return 3
 | |
| 		else
 | |
| 			return 2
 | |
| 		end
 | |
| 	else
 | |
| 		if dir.z < 0 then
 | |
| 			return 5
 | |
| 		else
 | |
| 			return 4
 | |
| 		end
 | |
| 	end
 | |
| end
 | |
| 
 | |
| -- table of dirs in wallmounted order
 | |
| local wallmounted_to_dir = {
 | |
| 	[0] = vector.new( 0,  1,  0),
 | |
| 	vector.new( 0, -1,  0),
 | |
| 	vector.new( 1,  0,  0),
 | |
| 	vector.new(-1,  0,  0),
 | |
| 	vector.new( 0,  0,  1),
 | |
| 	vector.new( 0,  0, -1),
 | |
| }
 | |
| function core.wallmounted_to_dir(wallmounted)
 | |
| 	return wallmounted_to_dir[wallmounted % 8]
 | |
| end
 | |
| 
 | |
| function core.dir_to_yaw(dir)
 | |
| 	return -math.atan2(dir.x, dir.z)
 | |
| end
 | |
| 
 | |
| function core.yaw_to_dir(yaw)
 | |
| 	return vector.new(-math.sin(yaw), 0, math.cos(yaw))
 | |
| end
 | |
| 
 | |
| function core.is_colored_paramtype(ptype)
 | |
| 	return (ptype == "color") or (ptype == "colorfacedir") or
 | |
| 		(ptype == "color4dir") or (ptype == "colorwallmounted") or
 | |
| 		(ptype == "colordegrotate")
 | |
| end
 | |
| 
 | |
| function core.strip_param2_color(param2, paramtype2)
 | |
| 	if not core.is_colored_paramtype(paramtype2) then
 | |
| 		return nil
 | |
| 	end
 | |
| 	if paramtype2 == "colorfacedir" then
 | |
| 		param2 = math.floor(param2 / 32) * 32
 | |
| 	elseif paramtype2 == "color4dir" then
 | |
| 		param2 = math.floor(param2 / 4) * 4
 | |
| 	elseif paramtype2 == "colorwallmounted" then
 | |
| 		param2 = math.floor(param2 / 8) * 8
 | |
| 	elseif paramtype2 == "colordegrotate" then
 | |
| 		param2 = math.floor(param2 / 32) * 32
 | |
| 	end
 | |
| 	-- paramtype2 == "color" requires no modification.
 | |
| 	return param2
 | |
| end
 | |
| 
 | |
| -- Content ID caching
 | |
| 
 | |
| local old_get_content_id = core.get_content_id
 | |
| local old_get_name_from_content_id = core.get_name_from_content_id
 | |
| 
 | |
| local name2content = setmetatable({}, {
 | |
| 	__index = function(self, name)
 | |
| 		return old_get_content_id(name)
 | |
| 	end,
 | |
| })
 | |
| 
 | |
| local content2name = setmetatable({}, {
 | |
| 	__index = function(self, id)
 | |
| 		return old_get_name_from_content_id(id)
 | |
| 	end,
 | |
| })
 | |
| 
 | |
| function core.get_content_id(name)
 | |
| 	return name2content[name]
 | |
| end
 | |
| 
 | |
| function core.get_name_from_content_id(id)
 | |
| 	return content2name[id]
 | |
| end
 | |
| 
 | |
| -- Cache content IDs after they have stopped changing.
 | |
| function builtin_shared.cache_content_ids()
 | |
| 	for name in pairs(core.registered_nodes) do
 | |
| 		local id = old_get_content_id(name)
 | |
| 		name2content[name] = id
 | |
| 		content2name[id] = name
 | |
| 	end
 | |
| 	-- unknown is not in the registered node list.
 | |
| 	local unknown_name = old_get_name_from_content_id(core.CONTENT_UNKNOWN)
 | |
| 	name2content[unknown_name] = core.CONTENT_UNKNOWN
 | |
| 	content2name[core.CONTENT_UNKNOWN] = unknown_name
 | |
| 
 | |
| 	for name in pairs(core.registered_aliases) do
 | |
| 		if core.registered_nodes[name] then
 | |
| 			name2content[name] = old_get_content_id(name)
 | |
| 		end
 | |
| 	end
 | |
| end
 | |
| 
 | |
| if core.set_read_node and core.set_push_node then
 | |
| 	local function read_node(node)
 | |
| 		return name2content[node.name], node.param1, node.param2
 | |
| 	end
 | |
| 	core.set_read_node(read_node)
 | |
| 	core.set_read_node = nil
 | |
| 
 | |
| 	local function push_node(content, param1, param2)
 | |
| 		return {name = content2name[content], param1 = param1, param2 = param2}
 | |
| 	end
 | |
| 	core.set_push_node(push_node)
 | |
| 	core.set_push_node = nil
 | |
| end
 |