mirror of
https://github.com/luanti-org/luanti.git
synced 2025-10-24 13:25:21 +02:00
240 lines
5.1 KiB
Lua
240 lines
5.1 KiB
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),
|
|
vector.new( 0, 1, 0),
|
|
vector.new( 0, -1, 0),
|
|
}
|
|
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 paramtype2 == "color" then
|
|
return param2
|
|
elseif paramtype2 == "colorfacedir" then
|
|
return math.floor(param2 / 32) * 32
|
|
elseif paramtype2 == "color4dir" then
|
|
return math.floor(param2 / 4) * 4
|
|
elseif paramtype2 == "colorwallmounted" then
|
|
return math.floor(param2 / 8) * 8
|
|
elseif paramtype2 == "colordegrotate" then
|
|
return math.floor(param2 / 32) * 32
|
|
else
|
|
return nil
|
|
end
|
|
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
|