2021-05-29 17:12:33 +03:00
local S = minetest.get_translator ( minetest.get_current_modname ( ) )
2019-03-08 02:46:35 +03:00
2018-05-29 14:25:25 +03:00
-- Core mcl_stairs API
2017-11-18 22:57:41 +03:00
-- Wrapper around mintest.pointed_thing_to_face_pos.
local function get_fpos ( placer , pointed_thing )
2019-04-01 16:16:33 +03:00
local finepos = minetest.pointed_thing_to_face_pos ( placer , pointed_thing )
return finepos.y % 1
2017-11-18 22:57:41 +03:00
end
2017-06-05 20:11:01 +03:00
local function place_slab_normal ( itemstack , placer , pointed_thing )
2017-07-24 21:10:13 +03:00
-- Use pointed node's on_rightclick function first, if present
local node = minetest.get_node ( pointed_thing.under )
if placer and not placer : get_player_control ( ) . sneak then
if minetest.registered_nodes [ node.name ] and minetest.registered_nodes [ node.name ] . on_rightclick then
return minetest.registered_nodes [ node.name ] . on_rightclick ( pointed_thing.under , node , placer , itemstack ) or itemstack
end
end
2017-06-05 20:11:01 +03:00
local p0 = pointed_thing.under
local p1 = pointed_thing.above
2021-04-17 10:26:37 +03:00
--local placer_pos = placer:get_pos()
2017-06-05 20:11:01 +03:00
2017-11-18 22:57:41 +03:00
local fpos = get_fpos ( placer , pointed_thing )
2017-06-05 20:11:01 +03:00
local place = ItemStack ( itemstack )
local origname = itemstack : get_name ( )
if p0.y - 1 == p1.y or ( fpos > 0 and fpos < 0.5 )
or ( fpos < - 0.5 and fpos > - 0.999999999 ) then
place : set_name ( origname .. " _top " )
end
local ret = minetest.item_place ( place , placer , pointed_thing , 0 )
ret : set_name ( origname )
return ret
end
local function place_stair ( itemstack , placer , pointed_thing )
2017-07-24 21:10:13 +03:00
-- Use pointed node's on_rightclick function first, if present
local node = minetest.get_node ( pointed_thing.under )
if placer and not placer : get_player_control ( ) . sneak then
if minetest.registered_nodes [ node.name ] and minetest.registered_nodes [ node.name ] . on_rightclick then
return minetest.registered_nodes [ node.name ] . on_rightclick ( pointed_thing.under , node , placer , itemstack ) or itemstack
end
end
2017-06-05 20:11:01 +03:00
local p0 = pointed_thing.under
local p1 = pointed_thing.above
local param2 = 0
2019-02-01 08:33:07 +03:00
local placer_pos = placer : get_pos ( )
2017-06-05 20:11:01 +03:00
if placer_pos then
param2 = minetest.dir_to_facedir ( vector.subtract ( p1 , placer_pos ) )
end
2017-11-20 05:26:34 +03:00
local fpos = get_fpos ( placer , pointed_thing )
2017-06-05 20:11:01 +03:00
if p0.y - 1 == p1.y or ( fpos > 0 and fpos < 0.5 )
or ( fpos < - 0.5 and fpos > - 0.999999999 ) then
param2 = param2 + 20
if param2 == 21 then
param2 = 23
elseif param2 == 23 then
param2 = 21
end
end
return minetest.item_place ( itemstack , placer , pointed_thing , param2 )
end
2018-05-29 14:25:25 +03:00
-- Register stairs.
2017-06-05 20:11:01 +03:00
-- Node will be called mcl_stairs:stair_<subname>
2020-09-08 14:06:52 +03:00
function mcl_stairs . register_stair ( subname , recipeitem , groups , images , description , sounds , blast_resistance , hardness , corner_stair_texture_override )
2017-12-08 21:03:20 +03:00
if recipeitem then
if not images then
images = minetest.registered_items [ recipeitem ] . tiles
end
if not groups then
groups = minetest.registered_items [ recipeitem ] . groups
end
if not sounds then
sounds = minetest.registered_items [ recipeitem ] . sounds
end
if not hardness then
hardness = minetest.registered_items [ recipeitem ] . _mcl_hardness
end
2020-09-08 14:06:52 +03:00
if not blast_resistance then
blast_resistance = minetest.registered_items [ recipeitem ] . _mcl_blast_resistance
end
2017-12-08 21:03:20 +03:00
end
2023-03-30 00:23:15 +03:00
groups.stair = 1
groups.building_block = 1
2023-11-03 10:48:59 +03:00
local image_table = { }
for i , image in ipairs ( images ) do
image_table [ i ] = type ( image ) == " string " and { name = image } or table.copy ( image )
image_table [ i ] . align_style = " world "
end
2017-06-05 20:11:01 +03:00
minetest.register_node ( " :mcl_stairs:stair_ " .. subname , {
description = description ,
2019-03-16 02:07:44 +03:00
_doc_items_longdesc = S ( " Stairs are useful to reach higher places by walking over them; jumping is not required. Placing stairs in a corner pattern will create corner stairs. Stairs placed on the ceiling or at the upper half of the side of a block will be placed upside down. " ) ,
2023-11-03 10:48:59 +03:00
drawtype = " nodebox " ,
tiles = image_table ,
2017-06-05 20:11:01 +03:00
paramtype = " light " ,
paramtype2 = " facedir " ,
is_ground_content = false ,
groups = groups ,
sounds = sounds ,
2023-11-03 10:48:59 +03:00
node_box = {
type = " fixed " ,
fixed = {
{ - 0.5 , - 0.5 , - 0.5 , 0.5 , 0 , 0.5 } ,
{ - 0.5 , 0 , 0 , 0.5 , 0.5 , 0.5 } ,
} ,
} ,
2017-06-05 20:11:01 +03:00
selection_box = {
type = " fixed " ,
fixed = {
{ - 0.5 , - 0.5 , - 0.5 , 0.5 , 0 , 0.5 } ,
{ - 0.5 , 0 , 0 , 0.5 , 0.5 , 0.5 } ,
} ,
} ,
collision_box = {
type = " fixed " ,
fixed = {
{ - 0.5 , - 0.5 , - 0.5 , 0.5 , 0 , 0.5 } ,
{ - 0.5 , 0 , 0 , 0.5 , 0.5 , 0.5 } ,
} ,
} ,
on_place = function ( itemstack , placer , pointed_thing )
if pointed_thing.type ~= " node " then
return itemstack
end
return place_stair ( itemstack , placer , pointed_thing )
end ,
2019-12-09 18:27:11 +03:00
on_rotate = function ( pos , node , user , mode , param2 )
-- Flip stairs vertically
if mode == screwdriver.ROTATE_AXIS then
local minor = node.param2
if node.param2 >= 20 then
minor = node.param2 - 20
if minor == 3 then
minor = 1
elseif minor == 1 then
minor = 3
end
node.param2 = minor
else
if minor == 3 then
minor = 1
elseif minor == 1 then
minor = 3
end
node.param2 = minor
node.param2 = node.param2 + 20
end
minetest.set_node ( pos , node )
2019-12-09 21:09:44 +03:00
return true
2019-12-09 18:27:11 +03:00
end
end ,
2020-09-08 14:06:52 +03:00
_mcl_blast_resistance = blast_resistance ,
2017-06-05 20:11:01 +03:00
_mcl_hardness = hardness ,
} )
if recipeitem then
minetest.register_craft ( {
2021-05-29 17:12:33 +03:00
output = " mcl_stairs:stair_ " .. subname .. " 4 " ,
2017-06-05 20:11:01 +03:00
recipe = {
{ recipeitem , " " , " " } ,
{ recipeitem , recipeitem , " " } ,
{ recipeitem , recipeitem , recipeitem } ,
} ,
} )
-- Flipped recipe
minetest.register_craft ( {
2021-05-29 17:12:33 +03:00
output = " mcl_stairs:stair_ " .. subname .. " 4 " ,
2017-06-05 20:11:01 +03:00
recipe = {
{ " " , " " , recipeitem } ,
{ " " , recipeitem , recipeitem } ,
{ recipeitem , recipeitem , recipeitem } ,
} ,
} )
2023-11-13 23:16:05 +03:00
-- Stonecutter recipe
mcl_stonecutter.register_recipe ( recipeitem , " mcl_stairs:stair_ " .. subname )
2017-06-05 20:11:01 +03:00
end
2018-05-29 14:25:25 +03:00
2018-05-29 16:17:15 +03:00
mcl_stairs.cornerstair . add ( " mcl_stairs:stair_ " .. subname , corner_stair_texture_override )
2017-06-05 20:11:01 +03:00
end
-- Slab facedir to placement 6d matching table
2021-04-17 10:26:37 +03:00
--local slab_trans_dir = {[0] = 8, 0, 2, 1, 3, 4}
2017-06-05 20:11:01 +03:00
-- Register slabs.
-- Node will be called mcl_stairs:slab_<subname>
-- double_description: NEW argument, not supported in Minetest Game
-- double_description: Description of double slab
2020-09-08 14:06:52 +03:00
function mcl_stairs . register_slab ( subname , recipeitem , groups , images , description , sounds , blast_resistance , hardness , double_description )
2017-06-05 20:11:01 +03:00
local lower_slab = " mcl_stairs:slab_ " .. subname
local upper_slab = lower_slab .. " _top "
local double_slab = lower_slab .. " _double "
2017-12-08 21:03:20 +03:00
if recipeitem then
if not images then
images = minetest.registered_items [ recipeitem ] . tiles
end
if not groups then
groups = minetest.registered_items [ recipeitem ] . groups
end
if not sounds then
sounds = minetest.registered_items [ recipeitem ] . sounds
end
if not hardness then
hardness = minetest.registered_items [ recipeitem ] . _mcl_hardness
end
2020-09-08 14:06:52 +03:00
if not blast_resistance then
blast_resistance = minetest.registered_items [ recipeitem ] . _mcl_blast_resistance
end
2017-12-08 21:03:20 +03:00
end
2017-06-05 20:11:01 +03:00
-- Automatically generate double slab description
if not double_description then
2019-03-08 02:46:35 +03:00
double_description = S ( " Double @1 " , description )
2017-06-05 20:11:01 +03:00
end
groups.slab = 1
groups.building_block = 1
2019-03-08 02:46:35 +03:00
local longdesc = S ( " Slabs are half as high as their full block counterparts and occupy either the lower or upper part of a block, depending on how it was placed. Slabs can be easily stepped on without needing to jump. When a slab is placed on another slab of the same type, a double slab is created. " )
2017-06-05 20:11:01 +03:00
local slabdef = {
description = description ,
_doc_items_longdesc = longdesc ,
drawtype = " nodebox " ,
tiles = images ,
paramtype = " light " ,
-- Facedir intentionally left out (see below)
is_ground_content = false ,
groups = groups ,
sounds = sounds ,
node_box = {
type = " fixed " ,
fixed = { - 0.5 , - 0.5 , - 0.5 , 0.5 , 0 , 0.5 } ,
} ,
on_place = function ( itemstack , placer , pointed_thing )
2024-07-31 01:11:22 +03:00
if not placer then return end
local above = pointed_thing.above
local under = pointed_thing.under
local anode = minetest.get_node ( above )
local unode = minetest.get_node ( under )
local adefs = minetest.registered_nodes [ anode.name ]
local udefs = minetest.registered_nodes [ unode.name ]
2017-06-05 20:11:01 +03:00
local wield_item = itemstack : get_name ( )
2024-07-31 01:11:22 +03:00
local player_name = placer : get_player_name ( )
local creative_enabled = minetest.is_creative_enabled ( player_name )
2017-06-05 20:11:01 +03:00
-- place slab using under node orientation
2024-07-31 01:11:22 +03:00
local dir = vector.subtract ( above , under )
local p2 = unode.param2
2017-06-05 20:11:01 +03:00
2024-07-31 01:11:22 +03:00
if minetest.is_protected ( under , player_name ) and not
minetest.check_player_privs ( placer , " protection_bypass " ) then
minetest.record_protection_violation ( under , player_name )
return
end
2017-06-05 20:11:01 +03:00
-- combine two slabs if possible
-- Requirements: Same slab material, must be placed on top of lower slab, or on bottom of upper slab
2024-07-31 01:11:22 +03:00
if ( wield_item == unode.name or ( udefs and wield_item == udefs._mcl_other_slab_half ) ) and
not ( ( dir.y >= 0 and minetest.get_item_group ( unode.name , " slab_top " ) == 1 ) or
( dir.y <= 0 and minetest.get_item_group ( unode.name , " slab_top " ) == 0 ) ) then
minetest.set_node ( under , { name = double_slab , param2 = p2 } )
if not creative_enabled then
itemstack : take_item ( )
2017-06-05 20:11:01 +03:00
end
2024-07-31 01:11:22 +03:00
return itemstack
elseif ( wield_item == anode.name or ( adefs and wield_item == adefs._mcl_other_slab_half ) ) then
minetest.set_node ( above , { name = double_slab , param2 = p2 } )
2017-06-05 20:11:01 +03:00
if not creative_enabled then
itemstack : take_item ( )
end
return itemstack
-- No combination possible: Place slab normally
else
return place_slab_normal ( itemstack , placer , pointed_thing )
end
end ,
_mcl_hardness = hardness ,
2021-04-17 10:26:37 +03:00
_mcl_blast_resistance = blast_resistance ,
2017-06-05 20:11:01 +03:00
_mcl_other_slab_half = upper_slab ,
2019-12-09 21:09:44 +03:00
on_rotate = function ( pos , node , user , mode , param2 )
-- Flip slab
if mode == screwdriver.ROTATE_AXIS then
node.name = upper_slab
minetest.set_node ( pos , node )
return true
end
return false
end ,
2017-06-05 20:11:01 +03:00
}
minetest.register_node ( " : " .. lower_slab , slabdef )
-- Register the upper slab.
-- Using facedir is not an option, as this would rotate the textures as well and would make
-- e.g. upper sandstone slabs look completely wrong.
local topdef = table.copy ( slabdef )
topdef.groups . slab = 1
topdef.groups . slab_top = 1
topdef.groups . not_in_creative_inventory = 1
topdef.groups . not_in_craft_guide = 1
2019-03-08 02:46:35 +03:00
topdef.description = S ( " Upper @1 " , description )
2017-06-05 20:11:01 +03:00
topdef._doc_items_create_entry = false
topdef._doc_items_longdesc = nil
topdef._doc_items_usagehelp = nil
topdef.drop = lower_slab
topdef._mcl_other_slab_half = lower_slab
2021-05-29 17:12:33 +03:00
function topdef . on_rotate ( pos , node , user , mode , param2 )
2019-12-09 21:09:44 +03:00
-- Flip slab
if mode == screwdriver.ROTATE_AXIS then
node.name = lower_slab
minetest.set_node ( pos , node )
return true
end
return false
end
2017-06-05 20:11:01 +03:00
topdef.node_box = {
type = " fixed " ,
fixed = { - 0.5 , 0 , - 0.5 , 0.5 , 0.5 , 0.5 } ,
}
topdef.selection_box = {
type = " fixed " ,
fixed = { - 0.5 , 0 , - 0.5 , 0.5 , 0.5 , 0.5 } ,
}
minetest.register_node ( " : " .. upper_slab , topdef )
-- Double slab node
local dgroups = table.copy ( groups )
dgroups.not_in_creative_inventory = 1
dgroups.not_in_craft_guide = 1
dgroups.slab = nil
dgroups.double_slab = 1
minetest.register_node ( " : " .. double_slab , {
description = double_description ,
2019-03-08 02:46:35 +03:00
_doc_items_longdesc = S ( " Double slabs are full blocks which are created by placing two slabs of the same kind on each other. " ) ,
2017-06-05 20:11:01 +03:00
tiles = images ,
is_ground_content = false ,
groups = dgroups ,
sounds = sounds ,
drop = lower_slab .. " 2 " ,
_mcl_hardness = hardness ,
2021-04-17 10:26:37 +03:00
_mcl_blast_resistance = blast_resistance ,
2017-06-05 20:11:01 +03:00
} )
if recipeitem then
minetest.register_craft ( {
output = lower_slab .. " 6 " ,
recipe = {
{ recipeitem , recipeitem , recipeitem } ,
} ,
} )
2023-11-13 23:16:05 +03:00
mcl_stonecutter.register_recipe ( recipeitem , lower_slab , 2 )
2017-06-05 20:11:01 +03:00
end
-- Help alias for the upper slab
if minetest.get_modpath ( " doc " ) then
doc.add_entry_alias ( " nodes " , lower_slab , " nodes " , upper_slab )
end
end
-- Stair/slab registration function.
-- Nodes will be called mcl_stairs:{stair,slab}_<subname>
function mcl_stairs . register_stair_and_slab ( subname , recipeitem ,
2020-09-08 14:06:52 +03:00
groups , images , desc_stair , desc_slab , sounds , blast_resistance , hardness ,
2018-05-29 15:28:26 +03:00
double_description , corner_stair_texture_override )
2020-09-08 14:06:52 +03:00
mcl_stairs.register_stair ( subname , recipeitem , groups , images , desc_stair , sounds , blast_resistance , hardness , corner_stair_texture_override )
mcl_stairs.register_slab ( subname , recipeitem , groups , images , desc_slab , sounds , blast_resistance , hardness , double_description )
2017-06-05 20:11:01 +03:00
end
-- Very simple registration function
-- Makes stair and slab out of a source node
2018-05-29 15:28:26 +03:00
function mcl_stairs . register_stair_and_slab_simple ( subname , sourcenode , desc_stair , desc_slab , desc_double_slab , corner_stair_texture_override )
2017-06-05 20:11:01 +03:00
local def = minetest.registered_nodes [ sourcenode ]
local groups = { }
-- Only allow a strict set of groups to be added to stairs and slabs for more predictable results
local allowed_groups = { " dig_immediate " , " handy " , " pickaxey " , " axey " , " shovely " , " shearsy " , " shearsy_wool " , " swordy " , " swordy_wool " }
for a = 1 , # allowed_groups do
if def.groups [ allowed_groups [ a ] ] then
groups [ allowed_groups [ a ] ] = def.groups [ allowed_groups [ a ] ]
end
end
2020-09-08 14:06:52 +03:00
mcl_stairs.register_stair_and_slab ( subname , sourcenode , groups , def.tiles , desc_stair , desc_slab , def.sounds , def._mcl_blast_resistance , def._mcl_hardness , desc_double_slab , corner_stair_texture_override )
2017-06-05 20:11:01 +03:00
end