2018-02-17 02:31:53 +01:00
|
|
|
|
local pipeworks_enabled = minetest.get_modpath("pipeworks") ~= nil
|
|
|
|
|
|
2017-10-12 00:55:45 -07:00
|
|
|
|
-- Sends a message onto the Digilines network.
|
|
|
|
|
-- pos: the position of the Digilines chest node.
|
|
|
|
|
-- action: the action string indicating what happened.
|
|
|
|
|
-- stack: the ItemStack that the action acted on (optional).
|
2017-10-20 00:24:32 -07:00
|
|
|
|
-- from_slot: the slot number that is taken from (optional).
|
|
|
|
|
-- to_slot: the slot number that is put into (optional).
|
2017-10-12 00:55:45 -07:00
|
|
|
|
-- side: which side of the chest the action occurred (optional).
|
2017-10-20 00:24:32 -07:00
|
|
|
|
local function send_message(pos, action, stack, from_slot, to_slot, side)
|
2017-10-12 00:55:45 -07:00
|
|
|
|
local channel = minetest.get_meta(pos):get_string("channel")
|
|
|
|
|
local msg = {
|
|
|
|
|
action = action,
|
|
|
|
|
stack = stack and stack:to_table(),
|
2017-10-20 00:24:32 -07:00
|
|
|
|
from_slot = from_slot,
|
|
|
|
|
to_slot = to_slot,
|
2017-10-12 00:55:45 -07:00
|
|
|
|
-- Duplicate the vector in case the caller expects it not to change.
|
|
|
|
|
side = side and vector.new(side)
|
|
|
|
|
}
|
|
|
|
|
digilines.receptor_send(pos, digilines.rules.default, channel, msg)
|
2014-01-26 08:48:46 +01:00
|
|
|
|
end
|
|
|
|
|
|
2017-10-12 00:55:45 -07:00
|
|
|
|
-- Checks if the inventory has become empty and, if so, sends an empty message.
|
|
|
|
|
local function check_empty(pos)
|
|
|
|
|
if minetest.get_meta(pos):get_inventory():is_empty("main") then
|
|
|
|
|
send_message(pos, "empty")
|
2014-01-26 08:48:46 +01:00
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
2017-10-12 00:55:45 -07:00
|
|
|
|
-- Checks if the inventory has become full for a particular type of item and,
|
|
|
|
|
-- if so, sends a full message.
|
|
|
|
|
local function check_full(pos, stack)
|
|
|
|
|
local one_item_stack = ItemStack(stack)
|
|
|
|
|
one_item_stack:set_count(1)
|
|
|
|
|
if not minetest.get_meta(pos):get_inventory():room_for_item("main", one_item_stack) then
|
|
|
|
|
send_message(pos, "full", one_item_stack)
|
2014-11-19 19:27:42 +01:00
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
2018-02-17 02:31:53 +01:00
|
|
|
|
local tubeconn = pipeworks_enabled and "^pipeworks_tube_connection_wooden.png" or ""
|
|
|
|
|
local tubescan = pipeworks_enabled and function(pos) pipeworks.scan_for_tube_objects(pos) end or nil
|
2014-11-19 19:27:42 +01:00
|
|
|
|
|
2017-10-12 00:55:45 -07:00
|
|
|
|
-- A place to remember things from allow_metadata_inventory_put to
|
|
|
|
|
-- on_metadata_inventory_put. This is a hack due to issue
|
|
|
|
|
-- minetest/minetest#6534 that should be removed once that’s fixed.
|
|
|
|
|
local last_inventory_put_index
|
|
|
|
|
local last_inventory_put_stack
|
|
|
|
|
|
|
|
|
|
-- A place to remember things from allow_metadata_inventory_take to
|
|
|
|
|
-- tube.remove_items. This is a hack due to issue minetest-mods/pipeworks#205
|
|
|
|
|
-- that should be removed once that’s fixed.
|
|
|
|
|
local last_inventory_take_index
|
|
|
|
|
|
2016-05-27 16:08:53 -07:00
|
|
|
|
minetest.register_alias("digilines_inventory:chest", "digilines:chest")
|
|
|
|
|
minetest.register_node("digilines:chest", {
|
2014-01-26 08:48:46 +01:00
|
|
|
|
description = "Digiline Chest",
|
2014-11-19 19:27:42 +01:00
|
|
|
|
tiles = {
|
|
|
|
|
"default_chest_top.png"..tubeconn,
|
|
|
|
|
"default_chest_top.png"..tubeconn,
|
|
|
|
|
"default_chest_side.png"..tubeconn,
|
|
|
|
|
"default_chest_side.png"..tubeconn,
|
|
|
|
|
"default_chest_side.png"..tubeconn,
|
|
|
|
|
"default_chest_front.png",
|
2014-01-26 08:48:46 +01:00
|
|
|
|
},
|
2014-11-19 19:27:42 +01:00
|
|
|
|
paramtype2 = "facedir",
|
|
|
|
|
legacy_facedir_simple = true,
|
|
|
|
|
groups = {choppy=2, oddly_breakable_by_hand=2, tubedevice=1, tubedevice_receiver=1},
|
|
|
|
|
sounds = default.node_sound_wood_defaults(),
|
2014-01-26 08:48:46 +01:00
|
|
|
|
on_construct = function(pos)
|
|
|
|
|
local meta = minetest.get_meta(pos)
|
2014-11-19 19:27:42 +01:00
|
|
|
|
meta:set_string("infotext", "Digiline Chest")
|
|
|
|
|
meta:set_string("formspec", "size[8,10]"..
|
|
|
|
|
((default and default.gui_bg) or "")..
|
|
|
|
|
((default and default.gui_bg_img) or "")..
|
|
|
|
|
((default and default.gui_slots) or "")..
|
|
|
|
|
"label[0,0;Digiline Chest]"..
|
|
|
|
|
"list[current_name;main;0,1;8,4;]"..
|
|
|
|
|
"field[2,5.5;5,1;channel;Channel;${channel}]"..
|
|
|
|
|
((default and default.get_hotbar_bg) and default.get_hotbar_bg(0,6) or "")..
|
2018-02-17 02:31:53 +01:00
|
|
|
|
"list[current_player;main;0,6;8,4;]"..
|
|
|
|
|
"listring[]")
|
2014-11-19 19:27:42 +01:00
|
|
|
|
local inv = meta:get_inventory()
|
|
|
|
|
inv:set_size("main", 8*4)
|
|
|
|
|
end,
|
|
|
|
|
after_place_node = tubescan,
|
|
|
|
|
after_dig_node = tubescan,
|
2017-03-01 07:31:48 -03:00
|
|
|
|
can_dig = function(pos)
|
2014-11-19 19:27:42 +01:00
|
|
|
|
return minetest.get_meta(pos):get_inventory():is_empty("main")
|
2014-01-26 08:48:46 +01:00
|
|
|
|
end,
|
2017-03-01 07:31:48 -03:00
|
|
|
|
on_receive_fields = function(pos, _, fields, sender)
|
2016-05-23 13:13:24 -05:00
|
|
|
|
local name = sender:get_player_name()
|
|
|
|
|
if minetest.is_protected(pos, name) and not minetest.check_player_privs(name, {protection_bypass=true}) then
|
|
|
|
|
minetest.record_protection_violation(pos, name)
|
|
|
|
|
return
|
|
|
|
|
end
|
2014-06-05 00:55:29 -05:00
|
|
|
|
if fields.channel ~= nil then
|
|
|
|
|
minetest.get_meta(pos):set_string("channel",fields.channel)
|
|
|
|
|
end
|
2014-01-26 08:48:46 +01:00
|
|
|
|
end,
|
2021-01-22 03:47:55 +11:00
|
|
|
|
digilines = {
|
2014-11-19 19:27:42 +01:00
|
|
|
|
receptor = {},
|
|
|
|
|
effector = {
|
2017-03-01 07:31:48 -03:00
|
|
|
|
action = function() end
|
2014-11-19 19:27:42 +01:00
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
tube = {
|
|
|
|
|
connect_sides = {left=1, right=1, back=1, front=1, bottom=1, top=1},
|
2014-01-26 08:48:46 +01:00
|
|
|
|
connects = function(i,param2)
|
|
|
|
|
return not pipeworks.connects.facingFront(i,param2)
|
|
|
|
|
end,
|
2014-11-19 19:27:42 +01:00
|
|
|
|
input_inventory = "main",
|
2017-10-12 00:55:45 -07:00
|
|
|
|
can_insert = function(pos, _, stack, direction)
|
|
|
|
|
local ret = minetest.get_meta(pos):get_inventory():room_for_item("main", stack)
|
|
|
|
|
if not ret then
|
|
|
|
|
-- The stack cannot be accepted. It will never be passed to
|
|
|
|
|
-- insert_object, but it should be reported as a toverflow.
|
|
|
|
|
-- Here, direction = direction item is moving, which is into
|
|
|
|
|
-- side.
|
|
|
|
|
local side = vector.multiply(direction, -1)
|
2017-10-20 00:24:32 -07:00
|
|
|
|
send_message(pos, "toverflow", stack, nil, nil, side)
|
2017-10-12 00:55:45 -07:00
|
|
|
|
end
|
|
|
|
|
return ret
|
2014-11-19 19:27:42 +01:00
|
|
|
|
end,
|
2017-10-20 00:24:32 -07:00
|
|
|
|
insert_object = function(pos, _, original_stack, direction)
|
2017-10-12 00:55:45 -07:00
|
|
|
|
-- Here, direction = direction item is moving, which is into side.
|
|
|
|
|
local side = vector.multiply(direction, -1)
|
2017-10-20 00:24:32 -07:00
|
|
|
|
local inv = minetest.get_meta(pos):get_inventory()
|
|
|
|
|
local inv_contents = inv:get_list("main")
|
|
|
|
|
local any_put = false
|
|
|
|
|
local stack = original_stack
|
|
|
|
|
local stack_name = stack:get_name()
|
2017-10-12 00:55:45 -07:00
|
|
|
|
local stack_count = stack:get_count()
|
2017-10-20 00:24:32 -07:00
|
|
|
|
-- Walk the inventory, adding items to existing stacks of the same
|
|
|
|
|
-- type.
|
|
|
|
|
for i = 1, #inv_contents do
|
|
|
|
|
local existing_stack = inv_contents[i]
|
|
|
|
|
if not existing_stack:is_empty() and existing_stack:get_name() == stack_name then
|
|
|
|
|
local leftover = existing_stack:add_item(stack)
|
|
|
|
|
local leftover_count = leftover:get_count()
|
|
|
|
|
if leftover_count ~= stack_count then
|
|
|
|
|
-- We put some items into the slot. Update the slot in
|
|
|
|
|
-- the inventory, tell Digilines listeners about it,
|
|
|
|
|
-- and keep looking for the a place to put the
|
|
|
|
|
-- leftovers if any.
|
|
|
|
|
any_put = true
|
|
|
|
|
inv:set_stack("main", i, existing_stack)
|
|
|
|
|
local stack_that_was_put
|
|
|
|
|
if leftover_count == 0 then
|
|
|
|
|
stack_that_was_put = stack
|
|
|
|
|
else
|
|
|
|
|
stack_that_was_put = ItemStack(stack)
|
|
|
|
|
stack_that_was_put:set_count(stack_count - leftover_count)
|
|
|
|
|
end
|
|
|
|
|
send_message(pos, "tput", stack_that_was_put, nil, i, side)
|
|
|
|
|
stack = leftover
|
|
|
|
|
stack_count = leftover_count
|
|
|
|
|
if stack_count == 0 then
|
|
|
|
|
break
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
2017-10-12 00:55:45 -07:00
|
|
|
|
end
|
2017-10-20 00:24:32 -07:00
|
|
|
|
if stack_count ~= 0 then
|
|
|
|
|
-- Walk the inventory, adding items to empty slots.
|
|
|
|
|
for i = 1, #inv_contents do
|
|
|
|
|
local existing_stack = inv_contents[i]
|
|
|
|
|
if existing_stack:is_empty() then
|
|
|
|
|
local leftover = existing_stack:add_item(stack)
|
|
|
|
|
local leftover_count = leftover:get_count()
|
|
|
|
|
if leftover_count ~= stack_count then
|
|
|
|
|
-- We put some items into the slot. Update the slot in
|
|
|
|
|
-- the inventory, tell Digilines listeners about it,
|
|
|
|
|
-- and keep looking for the a place to put the
|
|
|
|
|
-- leftovers if any.
|
|
|
|
|
any_put = true
|
|
|
|
|
inv:set_stack("main", i, existing_stack)
|
|
|
|
|
local stack_that_was_put
|
|
|
|
|
if leftover_count == 0 then
|
|
|
|
|
stack_that_was_put = stack
|
|
|
|
|
else
|
|
|
|
|
stack_that_was_put = ItemStack(stack)
|
|
|
|
|
stack_that_was_put:set_count(stack_count - leftover_count)
|
|
|
|
|
end
|
|
|
|
|
send_message(pos, "tput", stack_that_was_put, nil, i, side)
|
|
|
|
|
stack = leftover
|
|
|
|
|
stack_count = leftover_count
|
|
|
|
|
if stack_count == 0 then
|
|
|
|
|
break
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
if any_put then
|
|
|
|
|
check_full(pos, original_stack)
|
|
|
|
|
end
|
|
|
|
|
if stack_count ~= 0 then
|
2017-10-12 00:55:45 -07:00
|
|
|
|
-- Some items could not be added and bounced back. Report them.
|
2017-10-20 00:24:32 -07:00
|
|
|
|
send_message(pos, "toverflow", stack, nil, nil, side)
|
2014-01-26 08:48:46 +01:00
|
|
|
|
end
|
2017-10-20 00:24:32 -07:00
|
|
|
|
return stack
|
2014-01-26 08:48:46 +01:00
|
|
|
|
end,
|
2017-10-12 00:55:45 -07:00
|
|
|
|
remove_items = function(pos, _, stack, dir, count)
|
|
|
|
|
-- Here, stack is the ItemStack in our own inventory that is being
|
|
|
|
|
-- pulled from, NOT the stack that is actually pulled out.
|
|
|
|
|
-- Combining it with count gives the stack that is pulled out.
|
|
|
|
|
-- Also, note that Pipeworks doesn’t pass the index to this
|
|
|
|
|
-- function, so we use the one recorded in
|
|
|
|
|
-- allow_metadata_inventory_take; because we don’t implement
|
|
|
|
|
-- tube.can_remove, Pipeworks will call
|
|
|
|
|
-- allow_metadata_inventory_take instead and will pass it the
|
|
|
|
|
-- index.
|
|
|
|
|
local taken = stack:take_item(count)
|
|
|
|
|
minetest.get_meta(pos):get_inventory():set_stack("main", last_inventory_take_index, stack)
|
2017-10-20 00:24:32 -07:00
|
|
|
|
send_message(pos, "ttake", taken, last_inventory_take_index, nil, dir)
|
2017-10-12 00:55:45 -07:00
|
|
|
|
check_empty(pos)
|
|
|
|
|
return taken
|
|
|
|
|
end,
|
2014-11-19 19:27:42 +01:00
|
|
|
|
},
|
2017-10-12 00:55:45 -07:00
|
|
|
|
allow_metadata_inventory_put = function(pos, _, index, stack)
|
|
|
|
|
-- Remember what was in the target slot before the put; see
|
|
|
|
|
-- on_metadata_inventory_put for why we care.
|
|
|
|
|
last_inventory_put_index = index
|
|
|
|
|
last_inventory_put_stack = minetest.get_meta(pos):get_inventory():get_stack("main", index)
|
2014-01-26 08:48:46 +01:00
|
|
|
|
return stack:get_count()
|
|
|
|
|
end,
|
2017-10-12 00:55:45 -07:00
|
|
|
|
allow_metadata_inventory_take = function(_, _, index, stack)
|
|
|
|
|
-- Remember the index value; see tube.remove_items for why we care.
|
|
|
|
|
last_inventory_take_index = index
|
|
|
|
|
return stack:get_count()
|
|
|
|
|
end,
|
2021-01-16 02:59:19 -06:00
|
|
|
|
on_metadata_inventory_move = function(pos, _, from_index, _, to_index, count, player)
|
2017-10-20 00:24:32 -07:00
|
|
|
|
-- See what would happen if we were to move the items back from in the
|
|
|
|
|
-- opposite direction. In the event of a normal move, this must
|
|
|
|
|
-- succeed, because a normal move subtracts some items from the from
|
|
|
|
|
-- stack and adds them to the to stack; the two stacks naturally must
|
|
|
|
|
-- be compatible and so the reverse operation must succeed. However, if
|
|
|
|
|
-- the user *swaps* the two stacks instead, then due to issue
|
|
|
|
|
-- minetest/minetest#6534, this function is only called once; however,
|
|
|
|
|
-- when it is called, the stack that used to be in the to stack has
|
|
|
|
|
-- already been moved to the from stack, so we can detect the situation
|
|
|
|
|
-- by the fact that the reverse move will fail due to the from stack
|
|
|
|
|
-- being incompatible with its former contents.
|
|
|
|
|
local inv = minetest.get_meta(pos):get_inventory()
|
|
|
|
|
local from_stack = inv:get_stack("main", from_index)
|
|
|
|
|
local to_stack = inv:get_stack("main", to_index)
|
|
|
|
|
local reverse_move_stack = ItemStack(to_stack)
|
|
|
|
|
reverse_move_stack:set_count(count)
|
|
|
|
|
local swapped = from_stack:add_item(reverse_move_stack):get_count() == count
|
|
|
|
|
if swapped then
|
|
|
|
|
local channel = minetest.get_meta(pos):get_string("channel")
|
|
|
|
|
to_stack:set_count(count)
|
|
|
|
|
local msg = {
|
|
|
|
|
action = "uswap",
|
|
|
|
|
-- The slot and stack do not match because this function is
|
|
|
|
|
-- called after the action has taken place, but the Digilines
|
|
|
|
|
-- message is from the perspective of a viewer who hasn’t
|
|
|
|
|
-- observed the movement yet.
|
|
|
|
|
x_stack = to_stack:to_table(),
|
|
|
|
|
x_slot = from_index,
|
|
|
|
|
y_stack = from_stack:to_table(),
|
|
|
|
|
y_slot = to_index,
|
|
|
|
|
}
|
|
|
|
|
digilines.receptor_send(pos, digilines.rules.default, channel, msg)
|
|
|
|
|
else
|
|
|
|
|
to_stack:set_count(count)
|
|
|
|
|
send_message(pos, "umove", to_stack, from_index, to_index)
|
|
|
|
|
end
|
2014-11-19 19:27:42 +01:00
|
|
|
|
minetest.log("action", player:get_player_name().." moves stuff in chest at "..minetest.pos_to_string(pos))
|
|
|
|
|
end,
|
2017-10-12 00:55:45 -07:00
|
|
|
|
on_metadata_inventory_put = function(pos, _, index, stack, player)
|
|
|
|
|
-- Get what was in the target slot before the put; it has disappeared
|
|
|
|
|
-- by now (been replaced by the result of the put action) but we saved
|
|
|
|
|
-- it in allow_metadata_inventory_put. This should always work
|
|
|
|
|
-- (allow_metadata_inventory_put should AFAICT always be called
|
|
|
|
|
-- immediately before on_metadata_inventory_put), but in case of
|
|
|
|
|
-- something weird happening, just fall back to using an empty
|
|
|
|
|
-- ItemStack rather than crashing.
|
|
|
|
|
local old_stack
|
|
|
|
|
if last_inventory_put_index == index then
|
|
|
|
|
old_stack = last_inventory_put_stack
|
|
|
|
|
last_inventory_put_index = nil
|
|
|
|
|
last_inventory_put_stack = nil
|
2014-01-26 08:48:46 +01:00
|
|
|
|
else
|
2017-10-12 00:55:45 -07:00
|
|
|
|
old_stack = ItemStack(nil)
|
|
|
|
|
end
|
|
|
|
|
-- If the player tries to place a stack into an inventory, there’s
|
|
|
|
|
-- already a stack there, and the existing stack is either of a
|
|
|
|
|
-- different item type or full, then obviously the stacks can’t be
|
|
|
|
|
-- merged; instead the stacks are swapped. This information is not
|
|
|
|
|
-- reported to mods (Minetest core neither tells us that a particular
|
|
|
|
|
-- action was a swap, nor tells us a take followed by a put). In core,
|
|
|
|
|
-- the condition for swapping is that you try to add the new stack to
|
|
|
|
|
-- the existing stack and the leftovers are as big as the original
|
|
|
|
|
-- stack to put. Replicate that logic here using the old stack saved in
|
|
|
|
|
-- allow_metadata_inventory_put. If a swap happened, report it to the
|
|
|
|
|
-- Digilines network as a utake followed by a uput.
|
|
|
|
|
local leftovers = old_stack:add_item(stack)
|
|
|
|
|
if leftovers:get_count() == stack:get_count() then
|
2017-10-20 00:24:32 -07:00
|
|
|
|
send_message(pos, "utake", old_stack, index)
|
2014-01-26 08:48:46 +01:00
|
|
|
|
end
|
2017-10-20 00:24:32 -07:00
|
|
|
|
send_message(pos, "uput", stack, nil, index)
|
2017-10-12 00:55:45 -07:00
|
|
|
|
check_full(pos, stack)
|
2014-11-19 19:27:42 +01:00
|
|
|
|
minetest.log("action", player:get_player_name().." puts stuff into chest at "..minetest.pos_to_string(pos))
|
2014-01-26 08:48:46 +01:00
|
|
|
|
end,
|
2017-10-12 00:55:45 -07:00
|
|
|
|
on_metadata_inventory_take = function(pos, _, index, stack, player)
|
2017-10-20 00:24:32 -07:00
|
|
|
|
send_message(pos, "utake", stack, index)
|
2017-10-12 00:55:45 -07:00
|
|
|
|
check_empty(pos)
|
2014-11-19 19:27:42 +01:00
|
|
|
|
minetest.log("action", player:get_player_name().." takes stuff from chest at "..minetest.pos_to_string(pos))
|
2014-01-26 08:48:46 +01:00
|
|
|
|
end
|
|
|
|
|
})
|
2015-02-05 15:24:22 -05:00
|
|
|
|
|
|
|
|
|
minetest.register_craft({
|
|
|
|
|
type = "shapeless",
|
2016-05-27 16:08:53 -07:00
|
|
|
|
output = "digilines:chest",
|
2015-02-05 15:24:22 -05:00
|
|
|
|
recipe = {"default:chest", "digilines:wire_std_00000000"}
|
|
|
|
|
})
|