2019-03-07 21:35:02 +01:00
local S = minetest.get_translator ( " mcl_beds " )
2017-02-12 23:43:30 +01:00
2021-01-16 19:50:29 +01:00
local function destruct_bed ( pos , oldnode )
local node = oldnode or minetest.get_node ( pos )
if not node then return end
2019-02-20 19:22:24 +01:00
local dir = minetest.facedir_to_dir ( node.param2 )
2021-01-16 19:50:29 +01:00
local pos2 , node2
if string.sub ( node.name , - 4 ) == " _top " then
pos2 = vector.subtract ( pos , dir )
node2 = minetest.get_node ( pos2 )
if node2 and string.sub ( node2.name , - 7 ) == " _bottom " then
minetest.remove_node ( pos2 )
end
minetest.check_for_falling ( pos )
elseif string.sub ( node.name , - 7 ) == " _bottom " then
pos2 = vector.add ( pos , dir )
node2 = minetest.get_node ( pos2 )
if node2 and string.sub ( node2.name , - 4 ) == " _top " then
minetest.remove_node ( pos2 )
end
2017-02-12 23:43:30 +01:00
end
end
2019-02-20 20:52:06 +01:00
local function kick_player_after_destruct ( destruct_pos )
for name , player_bed_pos in pairs ( mcl_beds.bed_pos ) do
if vector.distance ( destruct_pos , player_bed_pos ) < 0.1 then
2019-02-20 18:01:21 +01:00
local player = minetest.get_player_by_name ( name )
if player and player : is_player ( ) then
mcl_beds.kick_player ( player )
break
end
end
end
end
2019-03-07 21:35:02 +01:00
local beddesc = S ( " Beds allow you to sleep at night and make the time pass faster. " )
local beduse = S ( " To use a bed, stand close to it and right-click the bed to sleep in it. Sleeping only works when the sun sets, at night or during a thunderstorm. The bed must also be clear of any danger. " )
2019-03-24 09:07:42 +01:00
if minetest.settings : get_bool ( " enable_bed_respawn " ) == false then
2019-03-14 06:20:05 +01:00
beddesc = beddesc .. " \n " .. S ( " You have heard of other worlds in which a bed would set the start point for your next life. But this world is not one of them. " )
2017-03-10 01:21:18 +01:00
else
2019-03-07 21:35:02 +01:00
beddesc = beddesc .. " \n " .. S ( " By using a bed, you set the starting point for your next life. If you die, you will start your next life at this bed, unless it is obstructed or destroyed. " )
2017-03-10 01:21:18 +01:00
end
2019-03-24 09:07:42 +01:00
if minetest.settings : get_bool ( " enable_bed_night_skip " ) == false then
2019-03-14 06:20:05 +01:00
beddesc = beddesc .. " \n " .. S ( " In this world, going to bed won't skip the night, but it will skip thunderstorms. " )
2017-03-10 01:21:18 +01:00
else
2019-03-07 21:35:02 +01:00
beddesc = beddesc .. " \n " .. S ( " Sleeping allows you to skip the night. The night is skipped when all players in this world went to sleep. The night is skipped after sleeping for a few seconds. Thunderstorms can be skipped in the same manner. " )
2017-03-10 01:21:18 +01:00
end
2017-07-26 19:07:57 +02:00
local default_sounds
if minetest.get_modpath ( " mcl_sounds " ) then
2019-06-14 23:03:12 +02:00
default_sounds = mcl_sounds.node_sound_wood_defaults ( {
footstep = { gain = 0.5 , name = " mcl_sounds_cloth " } ,
} )
2017-07-26 19:07:57 +02:00
end
2017-05-07 20:21:37 +02:00
function mcl_beds . register_bed ( name , def )
2017-05-09 19:02:24 +02:00
local node_box_bottom , selection_box_bottom , collision_box_bottom
if def.nodebox and def.nodebox . bottom then
node_box_bottom = { type = " fixed " , fixed = def.nodebox . bottom }
end
if def.selectionbox and def.selectionbox . bottom then
selection_box_bottom = { type = " fixed " , fixed = def.selectionbox . bottom }
end
if def.collisionbox and def.collisionbox . bottom then
collision_box_bottom = { type = " fixed " , fixed = def.collisionbox . bottom }
end
2017-02-12 23:43:30 +01:00
minetest.register_node ( name .. " _bottom " , {
description = def.description ,
2020-02-19 04:54:17 +01:00
_tt_help = S ( " Allows you to sleep " ) ,
2017-03-10 01:21:18 +01:00
_doc_items_longdesc = def._doc_items_longdesc or beddesc ,
_doc_items_usagehelp = def._doc_items_usagehelp or beduse ,
2019-03-24 09:06:35 +01:00
_doc_items_create_entry = def._doc_items_create_entry ,
_doc_items_entry_name = def._doc_items_entry_name ,
2017-02-12 23:43:30 +01:00
inventory_image = def.inventory_image ,
wield_image = def.wield_image ,
drawtype = " nodebox " ,
tiles = def.tiles . bottom ,
paramtype = " light " ,
paramtype2 = " facedir " ,
is_ground_content = false ,
stack_max = 1 ,
2020-04-18 23:24:42 +02:00
groups = { handy = 1 , flammable = 3 , bed = 1 , dig_by_piston = 1 , bouncy = 66 , fall_damage_add_percent =- 50 , deco_block = 1 , flammable =- 1 } ,
2017-02-27 00:51:50 +01:00
_mcl_hardness = 0.2 ,
2017-03-20 18:38:58 +01:00
_mcl_blast_resistance = 1 ,
2017-07-26 19:07:57 +02:00
sounds = def.sounds or default_sounds ,
2017-05-09 19:02:24 +02:00
node_box = node_box_bottom ,
selection_box = selection_box_bottom ,
collision_box = collision_box_bottom ,
2017-02-12 23:43:30 +01:00
on_place = function ( itemstack , placer , pointed_thing )
local under = pointed_thing.under
2017-03-02 16:20:19 +01:00
-- Use pointed node's on_rightclick function first, if present
local node = minetest.get_node ( 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-02-12 23:43:30 +01:00
local pos
2017-06-29 13:02:53 +02:00
local undername = minetest.get_node ( under ) . name
if minetest.registered_items [ undername ] and minetest.registered_items [ undername ] . buildable_to then
2017-02-12 23:43:30 +01:00
pos = under
else
pos = pointed_thing.above
end
if minetest.is_protected ( pos , placer : get_player_name ( ) ) and
not minetest.check_player_privs ( placer , " protection_bypass " ) then
minetest.record_protection_violation ( pos , placer : get_player_name ( ) )
return itemstack
end
local node_def = minetest.registered_nodes [ minetest.get_node ( pos ) . name ]
if not node_def or not node_def.buildable_to then
return itemstack
end
local dir = minetest.dir_to_facedir ( placer : get_look_dir ( ) )
local botpos = vector.add ( pos , minetest.facedir_to_dir ( dir ) )
if minetest.is_protected ( botpos , placer : get_player_name ( ) ) and
not minetest.check_player_privs ( placer , " protection_bypass " ) then
minetest.record_protection_violation ( botpos , placer : get_player_name ( ) )
return itemstack
end
local botdef = minetest.registered_nodes [ minetest.get_node ( botpos ) . name ]
if not botdef or not botdef.buildable_to then
return itemstack
end
minetest.set_node ( pos , { name = name .. " _bottom " , param2 = dir } )
minetest.set_node ( botpos , { name = name .. " _top " , param2 = dir } )
2020-07-10 16:08:40 +02:00
if not minetest.is_creative_enabled ( placer : get_player_name ( ) ) then
2017-02-12 23:43:30 +01:00
itemstack : take_item ( )
end
return itemstack
end ,
2021-01-16 19:50:29 +01:00
after_destruct = destruct_bed ,
2017-02-12 23:43:30 +01:00
2021-01-19 22:46:52 +01:00
on_destruct = kick_player_after_destruct ,
2017-02-12 23:43:30 +01:00
on_rightclick = function ( pos , node , clicker , itemstack , pointed_thing )
2019-02-20 19:22:24 +01:00
mcl_beds.on_rightclick ( pos , clicker , false )
2017-02-12 23:43:30 +01:00
return itemstack
end ,
on_rotate = function ( pos , node , user , mode , new_param2 )
local dir = minetest.facedir_to_dir ( node.param2 )
local p = vector.add ( pos , dir )
local node2 = minetest.get_node_or_nil ( p )
if not node2 or not minetest.get_item_group ( node2.name , " bed " ) == 2 or
not node.param2 == node2.param2 then
return false
end
if minetest.is_protected ( p , user : get_player_name ( ) ) then
minetest.record_protection_violation ( p , user : get_player_name ( ) )
return false
end
if mode ~= screwdriver.ROTATE_FACE then
return false
end
local newp = vector.add ( pos , minetest.facedir_to_dir ( new_param2 ) )
local node3 = minetest.get_node_or_nil ( newp )
local node_def = node3 and minetest.registered_nodes [ node3.name ]
if not node_def or not node_def.buildable_to then
return false
end
if minetest.is_protected ( newp , user : get_player_name ( ) ) then
minetest.record_protection_violation ( newp , user : get_player_name ( ) )
return false
end
node.param2 = new_param2
-- do not remove_node here - it will trigger destroy_bed()
minetest.set_node ( p , { name = " air " } )
minetest.set_node ( pos , node )
minetest.set_node ( newp , { name = name .. " _top " , param2 = new_param2 } )
return true
end ,
} )
2017-05-09 19:02:24 +02:00
local node_box_top , selection_box_top , collision_box_top
if def.nodebox and def.nodebox . top then
node_box_top = { type = " fixed " , fixed = def.nodebox . top }
end
if def.selectionbox and def.selectionbox . top then
selection_box_top = { type = " fixed " , fixed = def.selectionbox . top }
end
if d ef.collisionbox and def.collisionbox . top then
collision_box_top = { type = " fixed " , fixed = def.collisionbox . top }
end
2017-02-12 23:43:30 +01:00
minetest.register_node ( name .. " _top " , {
drawtype = " nodebox " ,
tiles = def.tiles . top ,
paramtype = " light " ,
paramtype2 = " facedir " ,
is_ground_content = false ,
2017-05-09 19:12:35 +02:00
-- FIXME: Should be bouncy=66, but this would be a higher bounciness than slime blocks!
2019-02-06 11:15:53 +01:00
groups = { handy = 1 , flammable = 3 , bed = 2 , dig_by_piston = 1 , bouncy = 33 , fall_damage_add_percent =- 50 , not_in_creative_inventory = 1 } ,
2017-02-27 00:51:50 +01:00
_mcl_hardness = 0.2 ,
2017-03-20 18:38:58 +01:00
_mcl_blast_resistance = 1 ,
2017-07-26 19:07:57 +02:00
sounds = def.sounds or default_sounds ,
2021-01-16 19:50:29 +01:00
drop = " " ,
2017-05-09 19:02:24 +02:00
node_box = node_box_top ,
selection_box = selection_box_top ,
collision_box = collision_box_top ,
2017-05-09 18:52:40 +02:00
on_rightclick = function ( pos , node , clicker , itemstack , pointed_thing )
2019-02-20 19:22:24 +01:00
mcl_beds.on_rightclick ( pos , clicker , true )
2017-05-09 18:52:40 +02:00
return itemstack
end ,
on_rotate = false ,
2021-01-16 19:50:29 +01:00
after_destruct = destruct_bed ,
2017-02-12 23:43:30 +01:00
} )
minetest.register_alias ( name , name .. " _bottom " )
2017-07-26 19:07:57 +02:00
if def.recipe then
minetest.register_craft ( {
output = name ,
recipe = def.recipe
} )
end
2017-05-09 18:52:40 +02:00
doc.add_entry_alias ( " nodes " , name .. " _bottom " , " nodes " , name .. " _top " )
2017-02-12 23:43:30 +01:00
end
2017-11-24 07:04:03 +01:00