2017-02-14 03:23:06 +01:00
--[[ This mod registers 3 nodes:
- One node for the horizontal - facing dispensers ( mcl_dispensers : dispenser )
- One node for the upwards - facing dispensers ( mcl_dispenser : dispenser_up )
- One node for the downwards - facing dispensers ( mcl_dispenser : dispenser_down )
3 node definitions are needed because of the way the textures are defined .
All node definitions share a lot of code , so this is the reason why there
are so many weird tables below .
] ]
2021-05-29 16:12:33 +02:00
local S = minetest.get_translator ( minetest.get_current_modname ( ) )
2022-09-10 11:39:10 +02:00
local C = minetest.colorize
local F = minetest.formspec_escape
2017-02-14 03:23:06 +01:00
2025-01-05 15:43:04 +01:00
-- TODO: actually should have a slight lag as in MC?
local COOLDOWN = 0.19
2022-09-10 11:39:10 +02:00
local dispenser_formspec = table.concat ( {
" formspec_version[4] " ,
" size[11.75,10.425] " ,
" label[4.125,0.375; " .. F ( C ( mcl_formspec.label_color , S ( " Dispenser " ) ) ) .. " ] " ,
mcl_formspec.get_itemslot_bg_v4 ( 4.125 , 0.75 , 3 , 3 ) ,
" list[context;main;4.125,0.75;3,3;] " ,
" label[0.375,4.7; " .. F ( C ( mcl_formspec.label_color , S ( " Inventory " ) ) ) .. " ] " ,
mcl_formspec.get_itemslot_bg_v4 ( 0.375 , 5.1 , 9 , 3 ) ,
" list[current_player;main;0.375,5.1;9,3;9] " ,
mcl_formspec.get_itemslot_bg_v4 ( 0.375 , 9.05 , 9 , 1 ) ,
" list[current_player;main;0.375,9.05;9,1;] " ,
" listring[context;main] " ,
" listring[current_player;main] " ,
} )
---For after_place_node
---@param pos Vector
2021-05-29 16:12:33 +02:00
local function setup_dispenser ( pos )
2017-02-14 03:23:06 +01:00
-- Set formspec and inventory
local meta = minetest.get_meta ( pos )
2022-09-10 11:39:10 +02:00
meta : set_string ( " formspec " , dispenser_formspec )
2017-02-14 03:23:06 +01:00
local inv = meta : get_inventory ( )
inv : set_size ( " main " , 9 )
end
2021-05-29 16:12:33 +02:00
local function orientate_dispenser ( pos , placer )
2018-01-11 01:28:36 +01:00
-- Not placed by player
if not placer then return end
-- Pitch in degrees
local pitch = placer : get_look_vertical ( ) * ( 180 / math.pi )
local node = minetest.get_node ( pos )
if pitch > 55 then
2022-09-10 11:39:10 +02:00
minetest.swap_node ( pos , { name = " mcl_dispensers:dispenser_up " , param2 = node.param2 } )
2018-01-11 01:28:36 +01:00
elseif pitch < - 55 then
2022-09-10 11:39:10 +02:00
minetest.swap_node ( pos , { name = " mcl_dispensers:dispenser_down " , param2 = node.param2 } )
2018-01-11 01:28:36 +01:00
end
end
2017-12-05 14:09:39 +01:00
local on_rotate
if minetest.get_modpath ( " screwdriver " ) then
on_rotate = screwdriver.rotate_simple
end
2017-02-14 03:23:06 +01:00
-- Shared core definition table
local dispenserdef = {
is_ground_content = false ,
sounds = mcl_sounds.node_sound_stone_defaults ( ) ,
2019-02-08 21:59:01 +01:00
allow_metadata_inventory_move = function ( pos , from_list , from_index , to_list , to_index , count , player )
local name = player : get_player_name ( )
if minetest.is_protected ( pos , name ) then
minetest.record_protection_violation ( pos , name )
return 0
else
return count
end
end ,
allow_metadata_inventory_take = function ( pos , listname , index , stack , player )
local name = player : get_player_name ( )
if minetest.is_protected ( pos , name ) then
minetest.record_protection_violation ( pos , name )
return 0
else
return stack : get_count ( )
end
end ,
allow_metadata_inventory_put = function ( pos , listname , index , stack , player )
local name = player : get_player_name ( )
if minetest.is_protected ( pos , name ) then
minetest.record_protection_violation ( pos , name )
return 0
else
return stack : get_count ( )
end
end ,
2017-02-14 03:23:06 +01:00
after_dig_node = function ( pos , oldnode , oldmetadata , digger )
local meta = minetest.get_meta ( pos )
2021-11-29 19:12:49 +01:00
local meta2 = meta : to_table ( )
2017-02-14 03:23:06 +01:00
meta : from_table ( oldmetadata )
local inv = meta : get_inventory ( )
2022-09-10 11:39:10 +02:00
for i = 1 , inv : get_size ( " main " ) do
2017-02-14 03:23:06 +01:00
local stack = inv : get_stack ( " main " , i )
if not stack : is_empty ( ) then
2023-08-19 18:27:00 +02:00
minetest.add_item ( vector.offset ( pos , math.random ( 0 , 10 ) / 10 - 0.5 , 0 , math.random ( 0 , 10 ) / 10 - 0.5 ) , stack )
2017-02-14 03:23:06 +01:00
end
end
2021-11-29 19:12:49 +01:00
meta : from_table ( meta2 )
2017-02-14 03:23:06 +01:00
end ,
2020-04-17 21:40:13 +02:00
_mcl_blast_resistance = 3.5 ,
2017-02-27 01:52:24 +01:00
_mcl_hardness = 3.5 ,
2021-05-23 11:50:16 +02:00
mesecons = {
effector = {
-- Dispense random item when triggered
2021-05-29 16:12:33 +02:00
action_on = function ( pos , node )
2021-05-23 11:50:16 +02:00
local meta = minetest.get_meta ( pos )
2025-01-05 15:43:04 +01:00
local gametime = core.get_gametime ( )
if gametime < meta : get_float ( " cooldown " ) then return end
meta : set_float ( " cooldown " , gametime + COOLDOWN )
2021-05-23 11:50:16 +02:00
local inv = meta : get_inventory ( )
local droppos , dropdir
if node.name == " mcl_dispensers:dispenser " then
dropdir = vector.multiply ( minetest.facedir_to_dir ( node.param2 ) , - 1 )
droppos = vector.add ( pos , dropdir )
elseif node.name == " mcl_dispensers:dispenser_up " then
2023-08-19 18:27:00 +02:00
dropdir = vector.new ( 0 , 1 , 0 )
droppos = vector.offset ( pos , 0 , 1 , 0 )
2021-05-23 11:50:16 +02:00
elseif node.name == " mcl_dispensers:dispenser_down " then
2023-08-19 18:27:00 +02:00
dropdir = vector.new ( 0 , - 1 , 0 )
droppos = vector.offset ( pos , 0 , - 1 , 0 )
2017-02-14 03:23:06 +01:00
end
2021-05-23 11:50:16 +02:00
local dropnode = minetest.get_node ( droppos )
local dropnodedef = minetest.registered_nodes [ dropnode.name ]
2024-12-29 08:04:59 +01:00
if not dropnodedef then
dropnodedef = minetest.registered_nodes [ " mapgen_stone " ]
end
2021-05-23 11:50:16 +02:00
local stacks = { }
2022-09-10 11:39:10 +02:00
for i = 1 , inv : get_size ( " main " ) do
2021-05-23 11:50:16 +02:00
local stack = inv : get_stack ( " main " , i )
if not stack : is_empty ( ) then
2022-09-10 11:39:10 +02:00
table.insert ( stacks , { stack = stack , stackpos = i } )
2021-05-23 11:50:16 +02:00
end
end
if # stacks >= 1 then
local r = math.random ( 1 , # stacks )
local stack = stacks [ r ] . stack
local dropitem = ItemStack ( stack )
dropitem : set_count ( 1 )
local stack_id = stacks [ r ] . stackpos
2024-12-29 08:04:59 +01:00
local stackdef = core.registered_items [ stack : get_name ( ) ]
2021-07-08 15:31:27 +02:00
if not stackdef then
return
end
2022-05-26 07:29:28 +02:00
2021-05-23 11:50:16 +02:00
local iname = stack : get_name ( )
2021-07-08 15:31:27 +02:00
local igroups = stackdef.groups
2021-05-23 11:50:16 +02:00
--[===[ Dispense item ]===]
-- Hardcoded dispensions --
-- Armor, mob heads and pumpkins
if igroups.armor then
2023-08-19 18:27:00 +02:00
local droppos_below = vector.offset ( droppos , 0 , - 1 , 0 )
2021-05-23 11:50:16 +02:00
2022-09-10 11:39:10 +02:00
for _ , objs in ipairs ( { minetest.get_objects_inside_radius ( droppos , 1 ) ,
minetest.get_objects_inside_radius ( droppos_below , 1 ) } ) do
2021-05-23 11:50:16 +02:00
for _ , obj in ipairs ( objs ) do
stack = mcl_armor.equip ( stack , obj )
if stack : is_empty ( ) then
break
end
end
2021-04-22 13:51:36 +02:00
if stack : is_empty ( ) then
2017-02-14 19:45:02 +01:00
break
end
end
2017-06-11 22:53:12 +02:00
2022-09-10 11:39:10 +02:00
-- Place head or pumpkin as node, if equipping it as armor has failed
2021-05-23 11:50:16 +02:00
if not stack : is_empty ( ) then
if igroups.head or iname == " mcl_farming:pumpkin_face " then
if dropnodedef.buildable_to then
2022-09-10 11:39:10 +02:00
minetest.set_node ( droppos , { name = iname , param2 = node.param2 } )
2021-05-23 11:50:16 +02:00
stack : take_item ( )
end
2017-06-11 22:53:12 +02:00
end
end
2017-02-15 00:31:00 +01:00
inv : set_stack ( " main " , stack_id , stack )
2021-09-06 15:30:08 +02:00
2022-09-10 11:39:10 +02:00
-- Use shears on sheeps
2021-09-06 15:30:08 +02:00
elseif igroups.shears then
for _ , obj in pairs ( minetest.get_objects_inside_radius ( droppos , 1 ) ) do
local entity = obj : get_luaentity ( )
if entity and not entity.child and not entity.gotten then
local entname = entity.name
local pos = obj : get_pos ( )
local used , texture = false
if entname == " mobs_mc:sheep " then
2023-06-19 18:49:20 +02:00
if entity.drops [ 2 ] then
minetest.add_item ( pos , entity.drops [ 2 ] . name .. " " .. math.random ( 1 , 3 ) )
end
2021-09-06 15:30:08 +02:00
if not entity.color then
entity.color = " unicolor_white "
end
entity.base_texture = { " blank.png " , " mobs_mc_sheep.png " }
texture = entity.base_texture
entity.drops = {
2022-05-25 23:25:15 +02:00
{ name = " mcl_mobitems:mutton " , chance = 1 , min = 1 , max = 2 } ,
2021-09-06 15:30:08 +02:00
}
used = true
elseif entname == " mobs_mc:snowman " then
texture = {
" mobs_mc_snowman.png " ,
" blank.png " , " blank.png " ,
" blank.png " , " blank.png " ,
" blank.png " , " blank.png " ,
}
used = true
elseif entname == " mobs_mc:mooshroom " then
local droppos = vector.offset ( pos , 0 , 1.4 , 0 )
if entity.base_texture [ 1 ] == " mobs_mc_mooshroom_brown.png " then
2022-05-25 23:25:15 +02:00
minetest.add_item ( droppos , " mcl_mushrooms:mushroom_brown 5 " )
2021-09-06 15:30:08 +02:00
else
2022-05-25 23:25:15 +02:00
minetest.add_item ( droppos , " mcl_mushrooms:mushroom_red 5 " )
2021-09-06 15:30:08 +02:00
end
2021-09-09 12:07:45 +02:00
obj = mcl_util.replace_mob ( obj , " mobs_mc:cow " )
entity = obj : get_luaentity ( )
2021-09-06 15:30:08 +02:00
used = true
end
if used then
obj : set_properties ( { textures = texture } )
entity.gotten = true
minetest.sound_play ( " mcl_tools_shears_cut " , { pos = pos } , true )
stack : add_wear ( 65535 / stackdef._mcl_diggroups . shearsy.uses )
2024-05-22 20:55:07 +02:00
tt.reload_itemstack_description ( stack ) -- update tooltip
2021-09-06 15:30:08 +02:00
inv : set_stack ( " main " , stack_id , stack )
2021-09-08 16:22:53 +02:00
break
2021-09-06 15:30:08 +02:00
end
end
end
2022-09-10 11:39:10 +02:00
-- Spawn Egg
2021-05-23 11:50:16 +02:00
elseif igroups.spawn_egg then
-- Spawn mob
if not dropnodedef.walkable then
--pointed_thing = { above = droppos, under = { x=droppos.x, y=droppos.y-1, z=droppos.z } }
minetest.add_entity ( droppos , stack : get_name ( ) )
2017-02-15 00:31:00 +01:00
2021-05-23 11:50:16 +02:00
stack : take_item ( )
2018-02-01 22:45:19 +01:00
inv : set_stack ( " main " , stack_id , stack )
2021-05-23 11:50:16 +02:00
end
2022-09-10 11:39:10 +02:00
-- Generalized dispension
2021-05-23 11:50:16 +02:00
elseif ( not dropnodedef.walkable or stackdef._dispense_into_walkable ) then
--[[ _on_dispense(stack, pos, droppos, dropnode, dropdir)
* stack : Itemstack which is dispense
* pos : Position of dispenser
* droppos : Position to which to dispense item
* dropnode : Node of droppos
* dropdir : Drop direction
_dispense_into_walkable : If true , can dispense into walkable nodes
] ]
if stackdef._on_dispense then
-- Item-specific dispension (if defined)
local od_ret = stackdef._on_dispense ( dropitem , pos , droppos , dropnode , dropdir )
if od_ret then
local newcount = stack : get_count ( ) - 1
stack : set_count ( newcount )
inv : set_stack ( " main " , stack_id , stack )
if newcount == 0 then
inv : set_stack ( " main " , stack_id , od_ret )
elseif inv : room_for_item ( " main " , od_ret ) then
inv : add_item ( " main " , od_ret )
else
2022-07-22 14:38:36 +02:00
local pos_variation = 100
droppos = {
x = droppos.x + math.random ( - pos_variation , pos_variation ) / 1000 ,
y = droppos.y + math.random ( - pos_variation , pos_variation ) / 1000 ,
z = droppos.z + math.random ( - pos_variation , pos_variation ) / 1000 ,
}
local item_entity = minetest.add_item ( droppos , dropitem )
local drop_vel = vector.subtract ( droppos , pos )
local speed = 3
2022-09-10 11:39:10 +02:00
item_entity : set_velocity ( vector.multiply ( drop_vel , speed ) )
2021-05-23 11:50:16 +02:00
end
2018-02-01 22:45:19 +01:00
else
2021-05-23 11:50:16 +02:00
stack : take_item ( )
inv : set_stack ( " main " , stack_id , stack )
2018-02-01 22:45:19 +01:00
end
else
2021-05-23 11:50:16 +02:00
-- Drop item otherwise
2022-07-22 14:38:36 +02:00
local pos_variation = 100
droppos = {
x = droppos.x + math.random ( - pos_variation , pos_variation ) / 1000 ,
y = droppos.y + math.random ( - pos_variation , pos_variation ) / 1000 ,
z = droppos.z + math.random ( - pos_variation , pos_variation ) / 1000 ,
}
local item_entity = minetest.add_item ( droppos , dropitem )
local drop_vel = vector.subtract ( droppos , pos )
local speed = 3
2022-09-10 11:39:10 +02:00
item_entity : set_velocity ( vector.multiply ( drop_vel , speed ) )
2018-02-01 22:45:19 +01:00
stack : take_item ( )
inv : set_stack ( " main " , stack_id , stack )
end
end
2021-05-23 11:50:16 +02:00
end
end ,
rules = mesecon.rules . alldirs ,
} ,
} ,
2017-12-05 14:09:39 +01:00
on_rotate = on_rotate ,
2017-02-14 03:23:06 +01:00
}
-- Horizontal dispenser
local horizontal_def = table.copy ( dispenserdef )
2019-03-08 01:07:41 +01:00
horizontal_def.description = S ( " Dispenser " )
2022-09-10 11:39:10 +02:00
horizontal_def._tt_help = S ( " 9 inventory slots " ) .. " \n " .. S ( " Launches item when powered by redstone power " )
2019-03-08 01:07:41 +01:00
horizontal_def._doc_items_longdesc = S ( " A dispenser is a block which acts as a redstone component which, when powered with redstone power, dispenses an item. It has a container with 9 inventory slots. " )
2022-09-10 11:39:10 +02:00
horizontal_def._doc_items_usagehelp = S ( " Place the dispenser in one of 6 possible directions. The “hole” is where items will fly out of the dispenser. Use the dispenser to access its inventory. Insert the items you wish to dispense. Supply the dispenser with redstone energy once to dispense a random item. " )
.. " \n \n " ..
S ( " The dispenser will do different things, depending on the dispensed item: " ) .. " \n \n " ..
S ( " • Arrows: Are launched " ) .. " \n " ..
S ( " • Eggs and snowballs: Are thrown " ) .. " \n " ..
S ( " • Fire charges: Are fired in a straight line " ) .. " \n " ..
S ( " • Armor: Will be equipped to players and armor stands " ) .. " \n " ..
S ( " • Boats: Are placed on water or are dropped " ) .. " \n " ..
S ( " • Minecart: Are placed on rails or are dropped " ) .. " \n " ..
S ( " • Bone meal: Is applied on the block it is facing " ) .. " \n " ..
S ( " • Empty buckets: Are used to collect a liquid source " ) .. " \n " ..
S ( " • Filled buckets: Are used to place a liquid source " ) .. " \n " ..
S ( " • Heads, pumpkins: Equipped to players and armor stands, or placed as a block " ) .. " \n " ..
S ( " • Shulker boxes: Are placed as a block " ) .. " \n " ..
S ( " • TNT: Is placed and ignited " ) .. " \n " ..
S ( " • Flint and steel: Is used to ignite a fire in air and to ignite TNT " ) .. " \n " ..
S ( " • Spawn eggs: Will summon the mob they contain " ) .. " \n " ..
S ( " • Other items: Are simply dropped " )
2017-03-11 02:50:12 +01:00
2021-05-29 16:12:33 +02:00
function horizontal_def . after_place_node ( pos , placer , itemstack , pointed_thing )
2017-02-14 03:23:06 +01:00
setup_dispenser ( pos )
2018-01-11 01:28:36 +01:00
orientate_dispenser ( pos , placer )
2017-02-14 03:23:06 +01:00
end
2021-05-29 16:12:33 +02:00
2017-02-14 03:23:06 +01:00
horizontal_def.tiles = {
" default_furnace_top.png " , " default_furnace_bottom.png " ,
" default_furnace_side.png " , " default_furnace_side.png " ,
" default_furnace_side.png " , " mcl_dispensers_dispenser_front_horizontal.png "
}
horizontal_def.paramtype2 = " facedir "
2022-09-10 11:39:10 +02:00
horizontal_def.groups = { pickaxey = 1 , container = 2 , material_stone = 1 }
2017-02-14 03:23:06 +01:00
minetest.register_node ( " mcl_dispensers:dispenser " , horizontal_def )
-- Down dispenser
local down_def = table.copy ( dispenserdef )
2019-03-08 01:07:41 +01:00
down_def.description = S ( " Downwards-Facing Dispenser " )
2017-02-14 03:23:06 +01:00
down_def.after_place_node = setup_dispenser
down_def.tiles = {
" default_furnace_top.png " , " mcl_dispensers_dispenser_front_vertical.png " ,
" default_furnace_side.png " , " default_furnace_side.png " ,
" default_furnace_side.png " , " default_furnace_side.png "
}
2022-09-10 11:39:10 +02:00
down_def.groups = { pickaxey = 1 , container = 2 , not_in_creative_inventory = 1 , material_stone = 1 }
2017-03-02 19:53:53 +01:00
down_def._doc_items_create_entry = false
2017-02-14 03:23:06 +01:00
down_def.drop = " mcl_dispensers:dispenser "
minetest.register_node ( " mcl_dispensers:dispenser_down " , down_def )
-- Up dispenser
-- The up dispenser is almost identical to the down dispenser , it only differs in textures
2017-08-02 01:34:12 +02:00
local up_def = table.copy ( down_def )
2019-03-08 01:07:41 +01:00
up_def.description = S ( " Upwards-Facing Dispenser " )
2017-02-14 03:23:06 +01:00
up_def.tiles = {
" mcl_dispensers_dispenser_front_vertical.png " , " default_furnace_bottom.png " ,
" default_furnace_side.png " , " default_furnace_side.png " ,
" default_furnace_side.png " , " default_furnace_side.png "
}
minetest.register_node ( " mcl_dispensers:dispenser_up " , up_def )
minetest.register_craft ( {
2021-05-29 16:12:33 +02:00
output = " mcl_dispensers:dispenser " ,
2017-02-14 03:23:06 +01:00
recipe = {
2022-09-10 11:39:10 +02:00
{ " mcl_core:cobble " , " mcl_core:cobble " , " mcl_core:cobble " , } ,
{ " mcl_core:cobble " , " mcl_bows:bow " , " mcl_core:cobble " , } ,
{ " mcl_core:cobble " , " mesecons:redstone " , " mcl_core:cobble " , } ,
2017-02-14 03:23:06 +01:00
}
} )
2017-02-14 04:39:37 +01:00
2017-03-21 04:27:50 +01:00
-- Add entry aliases for the Help
if minetest.get_modpath ( " doc " ) then
doc.add_entry_alias ( " nodes " , " mcl_dispensers:dispenser " , " nodes " , " mcl_dispensers:dispenser_down " )
doc.add_entry_alias ( " nodes " , " mcl_dispensers:dispenser " , " nodes " , " mcl_dispensers:dispenser_up " )
end
2019-03-21 16:08:29 +01:00
2020-03-29 14:35:01 +02:00
-- Legacy
2019-03-21 16:08:29 +01:00
minetest.register_lbm ( {
2020-03-29 14:35:01 +02:00
label = " Update dispenser formspecs (0.60.0) " ,
name = " mcl_dispensers:update_formspecs_0_60_0 " ,
2019-03-21 16:08:29 +01:00
nodenames = { " mcl_dispensers:dispenser " , " mcl_dispensers:dispenser_down " , " mcl_dispensers:dispenser_up " } ,
action = function ( pos , node )
setup_dispenser ( pos )
2022-09-10 11:39:10 +02:00
minetest.log ( " action " , " [mcl_dispenser] Node formspec updated at " .. minetest.pos_to_string ( pos ) )
2019-03-21 16:08:29 +01:00
end ,
} )