From 8cfd1185e312e749a5ee0e56fd571646f5c8c96a Mon Sep 17 00:00:00 2001
From: isaiah658 <isaiah658@outlook.com>
Date: Mon, 4 Jun 2018 07:03:07 -0500
Subject: Fix issue where storage drawer saves item name for items with max
 stack of 1 (#17)

Fix issue where storage drawer saves item name for items with max stack of 1 causing you to not be able to put in other items until breaking the drawer or reloading the game.
---
 lua/visual.lua | 1 +
 1 file changed, 1 insertion(+)

diff --git a/lua/visual.lua b/lua/visual.lua
index ca7db07..99de5b0 100755
--- a/lua/visual.lua
+++ b/lua/visual.lua
@@ -245,6 +245,7 @@ core.register_entity("drawers:visual", {
 
 		-- Don't add items stackable only to 1
 		if self.itemStackMax == 1 then
+			self.itemName = ""
 			return itemstack
 		end
 
-- 
cgit v1.2.3


From 60518668d5d1a8e27f34336849bf50e71af14c38 Mon Sep 17 00:00:00 2001
From: isaiah658 <isaiah658@outlook.com>
Date: Sat, 16 Jun 2018 02:03:48 -0500
Subject: Add drawer controller (#19)

Drawer controllers can sort items into drawer in a certain range. As normal drawers they have support
for pipeworks and MineClone 2. Crafting ingredients are one drawer, two diamonds, four steel/iron
ingots and a copper and tin ingot or two gold ingots.
---
 LICENSE.txt                               |   2 +
 README.md                                 |   2 +-
 init.lua                                  |   1 +
 lua/controller.lua                        | 420 ++++++++++++++++++++++++++++++
 textures/drawer_controller_side.png       | Bin 0 -> 577 bytes
 textures/drawer_controller_top_bottom.png | Bin 0 -> 490 bytes
 6 files changed, 424 insertions(+), 1 deletion(-)
 create mode 100644 lua/controller.lua
 create mode 100644 textures/drawer_controller_side.png
 create mode 100644 textures/drawer_controller_top_bottom.png

diff --git a/LICENSE.txt b/LICENSE.txt
index c01553f..3ff1ed4 100644
--- a/LICENSE.txt
+++ b/LICENSE.txt
@@ -108,6 +108,8 @@ Copyright (C) 2014 Justin Aquadro (MIT):
   textures/drawers_wood_front_2.png
   textures/drawers_wood_front_4.png
   textures/drawers_wood.png
+  textures/drawer_controller_side.png
+  textures/drawer_controller_top_bottom.png
 
 Everything not listed in here:
 Copyright (C) 2017 LNJ <git@lnj.li> (MIT)
diff --git a/README.md b/README.md
index 9be05a0..4528cee 100644
--- a/README.md
+++ b/README.md
@@ -12,7 +12,7 @@ MineClone 2 mods are only optional dependencies for crafting recipes.
 
 ## To-Do
 - [x] Add usable 1x1 drawer
-- [ ] Add a drawer controller for auto-sorting items into a drawer-network
+- [x] Add a drawer controller for auto-sorting items into a drawer-network
 - [ ] Add half-sized drawers
 - [x] Add 2x2 and 1x2 drawers
 - [ ] Add compacting drawers for auto-crafting blocks/ingots/fragments
diff --git a/init.lua b/init.lua
index fd66846..a7cc3d9 100755
--- a/init.lua
+++ b/init.lua
@@ -67,6 +67,7 @@ drawers.gui_slots = "listcolors[#00000069;#5A5A5A;#141318;#30434C;#FFF]"
 dofile(MP .. "/lua/helpers.lua")
 dofile(MP .. "/lua/visual.lua")
 dofile(MP .. "/lua/api.lua")
+dofile(MP .. "/lua/controller.lua")
 
 
 --
diff --git a/lua/controller.lua b/lua/controller.lua
new file mode 100644
index 0000000..315ae5c
--- /dev/null
+++ b/lua/controller.lua
@@ -0,0 +1,420 @@
+--[[
+Minetest Mod Storage Drawers - A Mod adding storage drawers
+
+Copyright (C) 2018 isaiah658
+Copyright (C) 2017 LNJ <git@lnj.li>
+
+MIT License
+
+Permission is hereby granted, free of charge, to any person obtaining a copy
+of this software and associated documentation files (the "Software"), to deal
+in the Software without restriction, including without limitation the rights
+to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+copies of the Software, and to permit persons to whom the Software is
+furnished to do so, subject to the following conditions:
+
+The above copyright notice and this permission notice shall be included in all
+copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
+SOFTWARE.
+]]--
+
+--[[ The gist of how the drawers mod stores data is that there are entities 
+and the drawer node itself. The entities are needed to allow having multiple 
+drawers in one node. The entities and node each store metadata about the item 
+counts and such. It is necessary to change both at once otherwise in some cases 
+the entity values are used and in other cases the node metadata is used.
+
+The gist of how the controller works is this. The drawer controller scans the 
+adjacent tiles (length and height is configurable) and puts the item names and 
+other info such as coordinates and the visualid of the entity in a table. That 
+table is saved in the controllers metadata. The table is used to help prevent 
+needing to scan all the drawers to deposit an item in certain situations. The 
+table is only updated on an as needed basis, not by a specific time/interval. 
+Controllers that have no items will not continue scanning drawers. ]]--
+
+local function controller_can_dig(pos, player)
+	local meta = core.get_meta(pos);
+	local inv = meta:get_inventory()
+	return inv:is_empty("src")
+end
+
+local function controller_allow_metadata_inventory_put(pos, listname, index, stack, player)
+	if core.is_protected(pos, player:get_player_name()) then
+		return 0
+	end
+	if listname == "src" then
+		return stack:get_count()
+	end
+end
+
+local function controller_allow_metadata_inventory_move(pos, from_list, from_index, to_list, to_index, count, player)
+	local meta = core.get_meta(pos)
+	local inv = meta:get_inventory()
+	local stack = inv:get_stack(from_list, from_index)
+	return controller_allow_metadata_inventory_put(pos, to_list, to_index, stack, player)
+end
+
+local function controller_allow_metadata_inventory_take(pos, listname, index, stack, player)
+	if core.is_protected(pos, player:get_player_name()) then
+		return 0
+	end
+	return stack:get_count()
+end
+
+local function controller_formspec(pos, meta_current_state)
+	local formspec = 
+		"size[8,8.5]"..
+		default.gui_bg..
+		default.gui_bg_img..
+		default.gui_slots..
+		"label[0,0;Current State: " .. meta_current_state .. "]" ..
+		"list[current_name;src;3.5,1.75;1,1;]"..
+		"list[current_player;main;0,4.25;8,1;]"..
+		"list[current_player;main;0,5.5;8,3;8]"..
+		"listring[current_player;main]"..
+		"listring[current_name;src]"..
+		"listring[current_player;main]"..
+		default.get_hotbar_bg(0, 4.25)
+	return formspec
+end
+
+local function index_drawers(pos)
+	--[[ The pos parameter is the controllers position
+	
+	We store the item name as a string key and the value is a table with position x,
+	position y, position z, and visualid. Those are all strings as well with the 
+	values assigned to them that way we don't need to worry about the ordering of 
+	the table. The count and max count are not stored as those values have a high 
+	potential of being outdated quickly. It's better to grab the values from the 
+	drawer when needed so you know you are working with accurate numbers.
+	
+	Indexing starts on the row (meaning same y coordinate) of the controller on the 
+	adjacent sides and moves each column on the same y level. A row is searched until 
+	it either hits the max length setting size or until a node that isn't a drawer is 
+	indexed and then moves up on (y coordinate increases by 1) and starts the process 
+	until the y coord is reaches the max height or until a node that isn't a drawer 
+	is indexed and at the point it stops indexing all together. This makes it so all 
+	drawers need to be next to each other on the rows without spacing or other blocks 
+	in between. ]]--
+	
+	local drawers_table_index = {}
+	local x_or_z_axis = 1
+	
+	-- Variables for managing the max length and max height that is searched in each adjacent direction from the controller
+	-- These could potentially be exposed to the user through the formspec, allowed to bigger with upgrades, etc
+	local max_search_length = 8
+	local max_search_height = 8
+	
+	-- Index the x axis and z axis in both the positive and negative directions
+	for x_z = 1,4 do
+		for y = 1,max_search_height do
+			for x_or_z = 1,max_search_length do
+				-- x_z if for controlling which axis and direction is being searched
+				-- x_or_z is for the column in each row that is searched
+				-- x_or_z_axis is used for breaking out of the loop if the first block searched in a row is not a drawer
+				x_or_z_axis = x_or_z
+				local drawer_pos
+				-- If x_z is 1, we check the positive x axis
+				if x_z == 1 then
+					drawer_pos = {x = pos.x + x_or_z, y = pos.y + y - 1, z = pos.z}
+				-- If x_z is 2, we check the negative x axis
+				elseif x_z == 2 then
+					drawer_pos = {x = pos.x - x_or_z, y = pos.y + y - 1, z = pos.z}
+				-- If x_z is 3, we check the positive z axis
+				elseif x_z == 3 then
+					drawer_pos = {x = pos.x, y = pos.y + y - 1, z = pos.z + x_or_z}
+				-- If x_z is 4, we check the negative z axis
+				elseif x_z == 4 then
+					drawer_pos = {x = pos.x, y = pos.y + y - 1, z = pos.z - x_or_z}
+				end
+				local drawer_meta = core.get_meta(drawer_pos)
+				local drawer_node = core.get_node(drawer_pos)
+				
+				-- There might be a better way to know if the node is a drawer other than matching a string in the node name
+				-- Can't trust metadata only in case another mod has a block with the same metadata strings
+				if string.match(drawer_node.name, 'drawers:') and drawer_node.name ~= "drawers:controller" and drawer_meta ~= nil then
+					for i = 0,4 do
+						-- This is needed for the special case where drawers that store one item don't have an id appended to them
+						local visualid = i
+						if i == 0 then
+							visualid = ""
+						end
+						local drawer_meta_name = drawer_meta:get_string("name" .. visualid)
+						local drawer_meta_entity_infotext = drawer_meta:get_string("entity_infotext" .. visualid)
+						-- Only one empty drawer needs to be indexed because everything is indexed again when an item isn't found in the index
+						if drawer_meta_name == "" and not drawers_table_index["empty"] and drawer_meta_entity_infotext ~= "" then
+							drawers_table_index["empty"] = {drawer_pos_x = drawer_pos.x, drawer_pos_y = drawer_pos.y, drawer_pos_z = drawer_pos.z, visualid = visualid}
+						elseif drawer_meta_name ~= "" then
+							-- If we already indexed this item previously, check which drawer has the most space and have that one be the one indexed
+							if drawers_table_index[drawer_meta_name] then
+								local indexed_drawer_meta = core.get_meta({x = drawers_table_index[drawer_meta_name]["drawer_pos_x"], y = drawers_table_index[drawer_meta_name]["drawer_pos_y"], z = drawers_table_index[drawer_meta_name]["drawer_pos_z"]})
+								local indexed_drawer_meta_count = indexed_drawer_meta:get_int("count" .. drawers_table_index[drawer_meta_name]["visualid"])
+								local indexed_drawer_meta_max_count = indexed_drawer_meta:get_int("max_count" .. drawers_table_index[drawer_meta_name]["visualid"])
+								local drawer_meta_count = drawer_meta:get_int("count" .. visualid)
+								local drawer_meta_max_count = drawer_meta:get_int("max_count" .. visualid)
+								-- If the already indexed drawer has less space, we override the table index for that item with the new drawer
+								if indexed_drawer_meta_max_count - indexed_drawer_meta_count < drawer_meta_max_count - drawer_meta_count then
+									drawers_table_index[drawer_meta_name] = {drawer_pos_x = drawer_pos.x, drawer_pos_y = drawer_pos.y, drawer_pos_z = drawer_pos.z, visualid = visualid}
+								end
+							else
+								drawers_table_index[drawer_meta_name] = {drawer_pos_x = drawer_pos.x, drawer_pos_y = drawer_pos.y, drawer_pos_z = drawer_pos.z, visualid = visualid}
+							end
+							-- If the drawer contained something and was a drawer type that only holds one item, stop the loop as there is no need to search through other drawer types
+							if i == 0 then
+								break
+							end
+						end
+					end
+				-- If the node isn't a drawer or doesn't have metadata, we break the loop to stop searching the row
+				else
+					break
+				end
+			end
+			-- If we break out of the above loop while x or z is 1, it means the first block searched in a row did not contain a drawer.
+			-- All searching for an axis is stopped when a row starts with a non-drawer.
+			if x_or_z_axis == 1 then
+				break
+			end
+		end
+	end
+	
+	return drawers_table_index
+end
+
+local function controller_node_timer(pos, elapsed)
+	-- Inizialize metadata
+	local meta = core.get_meta(pos)
+	local meta_current_state = meta:get_string("current_state")
+	local meta_times_ran_while_jammed = meta:get_float("times_ran_while_jammed")
+	local meta_jammed_item_name = meta:get_string("jammed_item_name")
+	local inv = meta:get_inventory()
+	local src = inv:get_stack("src", 1)
+	local src_name = src:get_name()
+	
+	--[[ There are four scenarios for the item slot in the controller. 
+	1: No item is in the controller. 
+	2: Item is not stackable. 
+	3. Item is allowed and there is either an existing drawer for that item with room or an empty drawer. 
+	4: Item is allowed, but there is no room.
+	
+	There are three different possibilities for "current_state". 
+	1: "running" which means means it's operating normally. 
+	2: "stopped" meaning the controller makes no attempt to put in the item possibly due to being unallowed for various reasons. 
+	3: "jammed" meaning the item is allowed in to drawers, but there was no space to deposit it last time it ran. ]]--
+	
+	--[[ If current state is jammed, the item that jammed it is the same item in the 
+	src inv slot, and the amount of times ran while jammed is 8 or higher, we 
+	set the current state to stopped. Will possibly want to make an option in the 
+	formspec to ignore this an continue running if the user plans on using the 
+	system in a way that may cause frequent jams making it a hassle to manually 
+	clear it each time ]]--
+	if meta_current_state == "jammed" and meta_jammed_item_name == src_name and meta_times_ran_while_jammed >= 8 then
+		meta:set_string("current_state", "stopped")
+		meta:set_string("formspec", controller_formspec(pos, "Stopped"))
+		return true
+	end
+	
+	-- If current state is stopped, and the item that jammed it is the same item in the src inv slot, we don't do anything
+	if meta_current_state == "stopped" and meta_jammed_item_name == src_name then
+		return true
+	end
+	
+	-- If current state is stopped, and the item that jammed it is not the same item in the src inv slot, we set the current state to running and clear the jam counter
+	if meta_current_state == "stopped" and meta_jammed_item_name ~= src_name then
+		meta:set_string("current_state", "running")
+		meta:set_string("formspec", controller_formspec(pos, "Running"))
+		meta:set_float("times_ran_while_jammed", 0)
+	end
+	
+	-- If no item is in the controller, nothing is searched and current_state is set to running and no jams
+	if inv:is_empty("src") then
+		meta:set_string("current_state", "running")
+		meta:set_string("formspec", controller_formspec(pos, "Running"))
+		meta:set_float("times_ran_while_jammed", 0)
+		return true
+	end
+	
+	-- If a non stackable item is in the controller, such as a written book, set the current_state to stopped because they are not allowed in drawers
+	if src:get_stack_max() == 1 then
+		meta:set_string("current_state", "stopped")
+		meta:set_string("formspec", controller_formspec(pos, "Stopped"))
+		meta:set_string("jammed_item_name", src_name)
+		meta:set_float("times_ran_while_jammed", 1)
+		return true
+	end
+	
+	-- If the index has not been created, the item isn't in the index, the item in the drawer is no longer the same item in the index, or the item is in the index but it's full, run the index_drawers function
+	local drawers_table_index = core.deserialize(meta:get_string("drawers_table_index"))
+	-- If the index has not been created
+	if not drawers_table_index then
+		drawers_table_index = index_drawers(pos)
+		meta:set_string("drawers_table_index", core.serialize(drawers_table_index))
+	-- If the item isn't in the index
+	elseif not drawers_table_index[src_name] then
+		drawers_table_index = index_drawers(pos)
+		meta:set_string("drawers_table_index", core.serialize(drawers_table_index))
+	-- If the item is in the index but either the name that was indexed is not the same as what is currently in the drawer or the drawer is full
+	elseif drawers_table_index[src_name] then
+		local visualid = drawers_table_index[src_name]["visualid"]
+		local indexed_drawer_meta = core.get_meta({x = drawers_table_index[src_name]["drawer_pos_x"], y = drawers_table_index[src_name]["drawer_pos_y"], z = drawers_table_index[src_name]["drawer_pos_z"]})
+		local indexed_drawer_meta_name = indexed_drawer_meta:get_string("name" .. visualid)
+		local indexed_drawer_meta_count = indexed_drawer_meta:get_int("count" .. visualid)
+		local indexed_drawer_meta_max_count = indexed_drawer_meta:get_int("max_count" .. visualid)
+		if indexed_drawer_meta_name ~= src_name or indexed_drawer_meta_count >= indexed_drawer_meta_max_count then
+			drawers_table_index = index_drawers(pos)
+			meta:set_string("drawers_table_index", core.serialize(drawers_table_index))
+		end
+	end
+	
+	-- This might not be needed, but my concern is if the above indexing takes enough time, there could be a "race condition" where the item in the src inventory is no longer the same item when we checked before or the quantity of the items changed so I'm having it grab the item stack again just in case
+	-- If a race condition does occur, items could be lost or duplicated
+	src = inv:get_stack("src", 1)
+	src_name = src:get_name()
+	local src_count = src:get_count()
+	local src_stack_max = src:get_stack_max()
+	
+	-- At this point, the item either was in the index or everything was reindexed so we check again
+	-- If there is a drawer with the item and it isn't full, we will put the items we can in to it
+	if drawers_table_index[src_name] then
+		local indexed_drawer_pos = {x = drawers_table_index[src_name]["drawer_pos_x"], y = drawers_table_index[src_name]["drawer_pos_y"], z = drawers_table_index[src_name]["drawer_pos_z"]}
+		local visualid = drawers_table_index[src_name]["visualid"]
+		local indexed_drawer_meta = core.get_meta(indexed_drawer_pos)
+		local indexed_drawer_meta_name = indexed_drawer_meta:get_string("name" .. visualid)
+		local indexed_drawer_meta_count = indexed_drawer_meta:get_int("count" .. visualid)
+		local indexed_drawer_meta_max_count = indexed_drawer_meta:get_int("max_count" .. visualid)
+		-- If the the item in the drawer is the same as the one we are trying to store, the drawer is not full, and the drawer entity is loaded, we will put the items in the drawer
+		if indexed_drawer_meta_name == src_name and indexed_drawer_meta_count < indexed_drawer_meta_max_count and drawers.drawer_visuals[core.serialize(indexed_drawer_pos)] then
+			local leftover = drawers.drawer_insert_object(indexed_drawer_pos, nil, src, nil)
+			inv:set_stack("src", 1, leftover)
+			-- Set the controller metadata
+			meta:set_string("current_state", "running")
+			meta:set_string("formspec", controller_formspec(pos, "Running"))
+			meta:set_float("times_ran_while_jammed", 0)
+		else
+			meta:set_string("current_state", "jammed")
+			meta:set_string("formspec", controller_formspec(pos, "Jammed"))
+			meta:set_string("jammed_item_name", src_name)
+			meta:set_float("times_ran_while_jammed", meta_times_ran_while_jammed + 1)
+		end
+	elseif drawers_table_index["empty"] then
+		local indexed_drawer_pos = {x = drawers_table_index["empty"]["drawer_pos_x"], y = drawers_table_index["empty"]["drawer_pos_y"], z = drawers_table_index["empty"]["drawer_pos_z"]}
+		local visualid = drawers_table_index["empty"]["visualid"]
+		local indexed_drawer_meta = core.get_meta(indexed_drawer_pos)
+		local indexed_drawer_meta_name = indexed_drawer_meta:get_string("name" .. visualid)
+		-- If the drawer is still empty and the drawer entity is loaded, we will put the items in the drawer
+		if indexed_drawer_meta_name == "" and drawers.drawer_visuals[core.serialize(indexed_drawer_pos)] then
+			local leftover = drawers.drawer_insert_object(indexed_drawer_pos, nil, src, nil)
+			inv:set_stack("src", 1, leftover)
+			-- Add the item to the drawers table index and set the empty one to nil
+			drawers_table_index["empty"]  = nil
+			drawers_table_index[src_name] = {drawer_pos_x = indexed_drawer_pos.x, drawer_pos_y = indexed_drawer_pos.y, drawer_pos_z = indexed_drawer_pos.z, visualid = visualid}
+			-- Set the controller metadata
+			meta:set_string("current_state", "running")
+			meta:set_string("formspec", controller_formspec(pos, "Running"))
+			meta:set_float("times_ran_while_jammed", 0)
+			meta:set_string("drawers_table_index", core.serialize(drawers_table_index))
+		else
+			meta:set_string("current_state", "jammed")
+			meta:set_string("formspec", controller_formspec(pos, "Jammed"))
+			meta:set_string("jammed_item_name", src_name)
+			meta:set_float("times_ran_while_jammed", meta_times_ran_while_jammed + 1)
+		end
+	else
+		meta:set_string("current_state", "jammed")
+		meta:set_string("formspec", controller_formspec(pos, "Jammed"))
+		meta:set_string("jammed_item_name", src_name)
+		meta:set_float("times_ran_while_jammed", meta_times_ran_while_jammed + 1)
+	end
+	
+	return true
+end
+
+-- Set the controller definition using a table to allow for pipeworks and potentially other mod support
+local controller_def = {}
+controller_def.description = 'Drawer Controller'
+controller_def.tiles = {"drawer_controller_top_bottom.png", "drawer_controller_top_bottom.png", "drawer_controller_side.png", "drawer_controller_side.png", "drawer_controller_side.png", "drawer_controller_side.png"}
+controller_def.can_dig = controller_can_dig
+controller_def.groups = {cracky = 3, level = 2}
+controller_def.on_construct = function(pos)
+	local meta = core.get_meta(pos)
+	local inv = meta:get_inventory()
+	inv:set_size('src', 1)
+	meta:set_string("current_state", "running")
+	meta:set_float("times_ran_while_jammed", 0)
+	meta:set_string("jammed_item_name", "")
+	meta:set_string("drawers_table_index", "")
+	meta:set_string("formspec", controller_formspec(pos, "Running"))
+	local timer = core.get_node_timer(pos)
+	timer:start(7)
+end
+controller_def.on_blast = function(pos)
+	local drops = {}
+	default.get_inventory_drops(pos, "src", drops)
+	drops[#drops+1] = "drawers:controller"
+	core.remove_node(pos)
+	return drops
+end
+controller_def.on_timer = controller_node_timer
+controller_def.allow_metadata_inventory_put = controller_allow_metadata_inventory_put
+controller_def.allow_metadata_inventory_move = controller_allow_metadata_inventory_move
+controller_def.allow_metadata_inventory_take = controller_allow_metadata_inventory_take
+
+-- Mostly copied from the drawers in the drawer mod to add pipeworks support
+if core.get_modpath("pipeworks") and pipeworks then
+	controller_def.groups.tubedevice = 1
+	controller_def.groups.tubedevice_receiver = 1
+	controller_def.tube = controller_def.tube or {}
+	controller_def.tube.insert_object = function(pos, node, stack, tubedir)
+		local meta = core.get_meta(pos)
+		local inv = meta:get_inventory()
+		return inv:add_item("src", stack)
+	end
+	controller_def.tube.can_insert = function(pos, node, stack, tubedir)
+		local meta = core.get_meta(pos)
+		local inv = meta:get_inventory()
+		return inv:room_for_item("src", stack)
+	end
+	controller_def.tube.connect_sides = {left = 1, right = 1, back = 1, front = 1,
+		top = 1, bottom = 1}
+	controller_def.after_place_node = pipeworks.after_place
+	controller_def.after_dig_node = pipeworks.after_dig
+end
+
+core.register_node('drawers:controller', controller_def)
+
+-- Because the rest of the drawers mod doesn't have a hard depend on default, I changed the recipe to have an alternative
+if core.get_modpath("default") and default then
+	core.register_craft({
+		output = 'drawers:controller',
+		recipe = {
+			{'default:steel_ingot', 'default:diamond', 'default:steel_ingot'},
+			{'default:tin_ingot', 'group:drawer', 'default:copper_ingot'},
+			{'default:steel_ingot', 'default:diamond', 'default:steel_ingot'},
+		}
+	})
+elseif core.get_modpath("mcl_core") and mcl_core then
+	core.register_craft({
+		output = 'drawers:controller',
+		recipe = {
+			{'mcl_core:iron_ingot', 'mcl_core:diamond', 'mcl_core:iron_ingot'},
+			{'mcl_core:gold_ingot', 'group:drawer', 'mcl_core:gold_ingot'},
+			{'mcl_core:iron_ingot', 'mcl_core:diamond', 'mcl_core:iron_ingot'},
+		}
+	})
+else
+	core.register_craft({
+		output = 'drawers:controller',
+		recipe = {
+			{'group:stone', 'group:stone', 'group:stone'},
+			{'group:stone', 'group:drawer', 'group:stone'},
+			{'group:stone', 'group:stone', 'group:stone'},
+		}
+	})
+end
diff --git a/textures/drawer_controller_side.png b/textures/drawer_controller_side.png
new file mode 100644
index 0000000..5f9f7ee
Binary files /dev/null and b/textures/drawer_controller_side.png differ
diff --git a/textures/drawer_controller_top_bottom.png b/textures/drawer_controller_top_bottom.png
new file mode 100644
index 0000000..0afb826
Binary files /dev/null and b/textures/drawer_controller_top_bottom.png differ
-- 
cgit v1.2.3


From f50412bed03bcc568b3819533877136097e28eb4 Mon Sep 17 00:00:00 2001
From: ditokp <ditokpl@gmail.com>
Date: Tue, 15 May 2018 10:51:29 +0000
Subject: Added translation using Weblate (Indonesian)

---
 locale/id.po | 105 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
 1 file changed, 105 insertions(+)
 create mode 100644 locale/id.po

diff --git a/locale/id.po b/locale/id.po
new file mode 100644
index 0000000..21b1626
--- /dev/null
+++ b/locale/id.po
@@ -0,0 +1,105 @@
+# SOME DESCRIPTIVE TITLE.
+# Copyright (C) YEAR THE PACKAGE'S COPYRIGHT HOLDER
+# This file is distributed under the same license as the PACKAGE package.
+# FIRST AUTHOR <EMAIL@ADDRESS>, YEAR.
+#
+msgid ""
+msgstr ""
+"Project-Id-Version: PACKAGE VERSION\n"
+"Report-Msgid-Bugs-To: \n"
+"POT-Creation-Date: 2017-06-03 17:04+0200\n"
+"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
+"Last-Translator: Automatically generated\n"
+"Language-Team: none\n"
+"Language: id\n"
+"MIME-Version: 1.0\n"
+"Content-Type: text/plain; charset=UTF-8\n"
+"Content-Transfer-Encoding: 8bit\n"
+
+#: lua/visual.lua
+msgid "Empty"
+msgstr ""
+
+#: lua/api.lua init.lua
+msgid "Wooden"
+msgstr ""
+
+#: lua/api.lua
+msgid "@1 Drawer"
+msgstr ""
+
+#: lua/api.lua
+msgid "@1 Drawers (1x2)"
+msgstr ""
+
+#: lua/api.lua
+msgid "@1 Drawers (2x2)"
+msgstr ""
+
+#: lua/helpers.lua
+msgid "@1 (@2% full)"
+msgstr ""
+
+#: lua/helpers.lua
+msgid "@1 @2 (@3% full)"
+msgstr ""
+
+#: init.lua
+msgid "Acacia Wood"
+msgstr ""
+
+#: init.lua
+msgid "Aspen Wood"
+msgstr ""
+
+#: init.lua
+msgid "Junglewood"
+msgstr ""
+
+#: init.lua
+msgid "Pine Wood"
+msgstr ""
+
+#: init.lua
+msgid "Oak Wood"
+msgstr ""
+
+#: init.lua
+msgid "Birch Wood"
+msgstr ""
+
+#: init.lua
+msgid "Dark Oak Wood"
+msgstr ""
+
+#: init.lua
+msgid "Spruce Wood"
+msgstr ""
+
+#: init.lua
+msgid "Steel Drawer Upgrade (x2)"
+msgstr ""
+
+#: init.lua
+msgid "Gold Drawer Upgrade (x3)"
+msgstr ""
+
+#: init.lua
+msgid "Obsidian Drawer Upgrade (x5)"
+msgstr ""
+
+#: init.lua
+msgid "Diamond Drawer Upgrade (x8)"
+msgstr ""
+
+#: init.lua
+msgid "Iron Drawer Upgrade (x2)"
+msgstr ""
+
+#: init.lua
+msgid "Emerald Drawer Upgrade (x13)"
+msgstr ""
+
+#: init.lua
+msgid "Drawer Upgrade Template"
+msgstr ""
-- 
cgit v1.2.3


From 28587a3a1025cd3fd2fdd942cd298d5bd5897a03 Mon Sep 17 00:00:00 2001
From: ditokp <ditokpl@gmail.com>
Date: Tue, 15 May 2018 11:06:43 +0000
Subject: Translated using Weblate (Indonesian)

Currently translated at 100.0% (22 of 22 strings)

Translation: Minetest/Mod: Storage Drawers
Translate-URL: https://hosted.weblate.org/projects/minetest/mod-storage-drawers/id/
---
 locale/id.po | 53 ++++++++++++++++++++++++++++-------------------------
 1 file changed, 28 insertions(+), 25 deletions(-)

diff --git a/locale/id.po b/locale/id.po
index 21b1626..79b5856 100644
--- a/locale/id.po
+++ b/locale/id.po
@@ -8,98 +8,101 @@ msgstr ""
 "Project-Id-Version: PACKAGE VERSION\n"
 "Report-Msgid-Bugs-To: \n"
 "POT-Creation-Date: 2017-06-03 17:04+0200\n"
-"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
-"Last-Translator: Automatically generated\n"
-"Language-Team: none\n"
+"PO-Revision-Date: 2018-05-15 11:12+0000\n"
+"Last-Translator: ditokp <ditokpl@gmail.com>\n"
+"Language-Team: Indonesian <https://hosted.weblate.org/projects/minetest/"
+"mod-storage-drawers/id/>\n"
 "Language: id\n"
 "MIME-Version: 1.0\n"
 "Content-Type: text/plain; charset=UTF-8\n"
 "Content-Transfer-Encoding: 8bit\n"
+"Plural-Forms: nplurals=1; plural=0;\n"
+"X-Generator: Weblate 3.0-dev\n"
 
 #: lua/visual.lua
 msgid "Empty"
-msgstr ""
+msgstr "Kosong"
 
 #: lua/api.lua init.lua
 msgid "Wooden"
-msgstr ""
+msgstr "Kayu"
 
 #: lua/api.lua
 msgid "@1 Drawer"
-msgstr ""
+msgstr "@1 Laci"
 
 #: lua/api.lua
 msgid "@1 Drawers (1x2)"
-msgstr ""
+msgstr "@1 Laci (1x2)"
 
 #: lua/api.lua
 msgid "@1 Drawers (2x2)"
-msgstr ""
+msgstr "@1 Laci (2x2)"
 
 #: lua/helpers.lua
 msgid "@1 (@2% full)"
-msgstr ""
+msgstr "@1 (@2% penuh)"
 
 #: lua/helpers.lua
 msgid "@1 @2 (@3% full)"
-msgstr ""
+msgstr "@1 @2 (@3% penuh)"
 
 #: init.lua
 msgid "Acacia Wood"
-msgstr ""
+msgstr "Kayu Akasia"
 
 #: init.lua
 msgid "Aspen Wood"
-msgstr ""
+msgstr "Kayu Aspen"
 
 #: init.lua
 msgid "Junglewood"
-msgstr ""
+msgstr "Kayu Rimba"
 
 #: init.lua
 msgid "Pine Wood"
-msgstr ""
+msgstr "Kayu Pinus"
 
 #: init.lua
 msgid "Oak Wood"
-msgstr ""
+msgstr "Kayu Ek"
 
 #: init.lua
 msgid "Birch Wood"
-msgstr ""
+msgstr "Kayu Birch"
 
 #: init.lua
 msgid "Dark Oak Wood"
-msgstr ""
+msgstr "Kayu Ek Gelap"
 
 #: init.lua
 msgid "Spruce Wood"
-msgstr ""
+msgstr "Kayu Cemara"
 
 #: init.lua
 msgid "Steel Drawer Upgrade (x2)"
-msgstr ""
+msgstr "Tingkatkan menjadi Laci Baja (x2)"
 
 #: init.lua
 msgid "Gold Drawer Upgrade (x3)"
-msgstr ""
+msgstr "Tingkatkan menjadi Laci Emas (x3)"
 
 #: init.lua
 msgid "Obsidian Drawer Upgrade (x5)"
-msgstr ""
+msgstr "Tingkatkan menjadi Laci Obsidian (x5)"
 
 #: init.lua
 msgid "Diamond Drawer Upgrade (x8)"
-msgstr ""
+msgstr "Tingkatkan menjadi Laci Berlian (x8)"
 
 #: init.lua
 msgid "Iron Drawer Upgrade (x2)"
-msgstr ""
+msgstr "Tingkatkan menjadi Laci Besi (x2)"
 
 #: init.lua
 msgid "Emerald Drawer Upgrade (x13)"
-msgstr ""
+msgstr "Tingkatkan menjadi Laci Zamrud (x13)"
 
 #: init.lua
 msgid "Drawer Upgrade Template"
-msgstr ""
+msgstr "Template Peningkatan Laci"
-- 
cgit v1.2.3


From a35fb4ca866f2b668177e6b2586a58f4a1fde1e3 Mon Sep 17 00:00:00 2001
From: Muhammad Rifqi Priyo Susanto <muhammadrifqipriyosusanto@gmail.com>
Date: Sat, 19 May 2018 16:35:05 +0000
Subject: Translated using Weblate (Indonesian)

Currently translated at 100.0% (22 of 22 strings)

Translation: Minetest/Mod: Storage Drawers
Translate-URL: https://hosted.weblate.org/projects/minetest/mod-storage-drawers/id/
---
 locale/id.po | 7 ++++---
 1 file changed, 4 insertions(+), 3 deletions(-)

diff --git a/locale/id.po b/locale/id.po
index 79b5856..94ea4f9 100644
--- a/locale/id.po
+++ b/locale/id.po
@@ -8,8 +8,9 @@ msgstr ""
 "Project-Id-Version: PACKAGE VERSION\n"
 "Report-Msgid-Bugs-To: \n"
 "POT-Creation-Date: 2017-06-03 17:04+0200\n"
-"PO-Revision-Date: 2018-05-15 11:12+0000\n"
-"Last-Translator: ditokp <ditokpl@gmail.com>\n"
+"PO-Revision-Date: 2018-05-20 16:38+0000\n"
+"Last-Translator: Muhammad Rifqi Priyo Susanto "
+"<muhammadrifqipriyosusanto@gmail.com>\n"
 "Language-Team: Indonesian <https://hosted.weblate.org/projects/minetest/"
 "mod-storage-drawers/id/>\n"
 "Language: id\n"
@@ -105,4 +106,4 @@ msgstr "Tingkatkan menjadi Laci Zamrud (x13)"
 
 #: init.lua
 msgid "Drawer Upgrade Template"
-msgstr "Template Peningkatan Laci"
+msgstr "Templat Peningkatan Laci"
-- 
cgit v1.2.3


From 19a15d15d04d894dce549f7b194082fd525e28fe Mon Sep 17 00:00:00 2001
From: LNJ <git@lnj.li>
Date: Sat, 16 Jun 2018 09:17:24 +0200
Subject: Update translation sources

---
 locale/es.po        | 51 ++++++++++++++++++++++++++++++++++++++++-----------
 locale/pl.po        | 51 ++++++++++++++++++++++++++++++++++++++++-----------
 locale/template.pot |  2 +-
 3 files changed, 81 insertions(+), 23 deletions(-)

diff --git a/locale/es.po b/locale/es.po
index 3498991..d38f84e 100644
--- a/locale/es.po
+++ b/locale/es.po
@@ -7,7 +7,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: PACKAGE VERSION\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-04-14 16:13+0200\n"
+"POT-Creation-Date: 2018-06-16 09:14+0200\n"
 "PO-Revision-Date: 2017-06-01 21:02+0000\n"
 "Last-Translator: Miguel Isaac <mineyoshist@protonmail.ch>\n"
 "Language-Team: Spanish <https://hosted.weblate.org/projects/minetest/mod-"
@@ -19,13 +19,9 @@ msgstr ""
 "Plural-Forms: nplurals=2; plural=n != 1;\n"
 "X-Generator: Weblate 2.15-dev\n"
 
-#: lua/helpers.lua
-msgid "@1 (@2% full)"
-msgstr "@1 (@2% lleno)"
-
-#: lua/helpers.lua
-msgid "@1 @2 (@3% full)"
-msgstr "@1 @2 (@3% lleno)"
+#: lua/visual.lua
+msgid "Empty"
+msgstr "Vacio"
 
 #: lua/api.lua init.lua
 msgid "Wooden"
@@ -43,9 +39,13 @@ msgstr "@1 Gavetas (1x2)"
 msgid "@1 Drawers (2x2)"
 msgstr "@1 Gavetas (2x2)"
 
-#: lua/visual.lua
-msgid "Empty"
-msgstr "Vacio"
+#: lua/helpers.lua
+msgid "@1 (@2% full)"
+msgstr "@1 (@2% lleno)"
+
+#: lua/helpers.lua
+msgid "@1 @2 (@3% full)"
+msgstr "@1 @2 (@3% lleno)"
 
 #: init.lua
 msgid "Acacia Wood"
@@ -78,3 +78,32 @@ msgstr "Madera de Roble Oscuro"
 #: init.lua
 msgid "Spruce Wood"
 msgstr "Madera de Pícea"
+
+#: init.lua
+msgid "Steel Drawer Upgrade (x2)"
+msgstr ""
+
+#: init.lua
+msgid "Gold Drawer Upgrade (x3)"
+msgstr ""
+
+#: init.lua
+msgid "Obsidian Drawer Upgrade (x5)"
+msgstr ""
+
+#: init.lua
+msgid "Diamond Drawer Upgrade (x8)"
+msgstr ""
+
+#: init.lua
+#, fuzzy
+msgid "Iron Drawer Upgrade (x2)"
+msgstr "@1 Gavetas (1x2)"
+
+#: init.lua
+msgid "Emerald Drawer Upgrade (x13)"
+msgstr ""
+
+#: init.lua
+msgid "Drawer Upgrade Template"
+msgstr ""
diff --git a/locale/pl.po b/locale/pl.po
index a241c73..0087706 100644
--- a/locale/pl.po
+++ b/locale/pl.po
@@ -7,7 +7,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: PACKAGE VERSION\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-04-14 16:13+0200\n"
+"POT-Creation-Date: 2018-06-16 09:14+0200\n"
 "PO-Revision-Date: 2017-06-04 23:03+0000\n"
 "Last-Translator: LNJ <git@lnj.li>\n"
 "Language-Team: Polish <https://hosted.weblate.org/projects/minetest/mod-"
@@ -20,13 +20,9 @@ msgstr ""
 "|| n%100>=20) ? 1 : 2;\n"
 "X-Generator: Weblate 2.15-dev\n"
 
-#: lua/helpers.lua
-msgid "@1 (@2% full)"
-msgstr "@1 (@2% napełnione)"
-
-#: lua/helpers.lua
-msgid "@1 @2 (@3% full)"
-msgstr "@1 @2 (@3% napełnione)"
+#: lua/visual.lua
+msgid "Empty"
+msgstr "Pusta"
 
 #: lua/api.lua init.lua
 msgid "Wooden"
@@ -44,9 +40,13 @@ msgstr "@1 Szuflady (1x2)"
 msgid "@1 Drawers (2x2)"
 msgstr "@1 Szuflady (2x2)"
 
-#: lua/visual.lua
-msgid "Empty"
-msgstr "Pusta"
+#: lua/helpers.lua
+msgid "@1 (@2% full)"
+msgstr "@1 (@2% napełnione)"
+
+#: lua/helpers.lua
+msgid "@1 @2 (@3% full)"
+msgstr "@1 @2 (@3% napełnione)"
 
 #: init.lua
 msgid "Acacia Wood"
@@ -79,3 +79,32 @@ msgstr "Drewno ciemne dębowe"
 #: init.lua
 msgid "Spruce Wood"
 msgstr "Drwno świerkowe"
+
+#: init.lua
+msgid "Steel Drawer Upgrade (x2)"
+msgstr ""
+
+#: init.lua
+msgid "Gold Drawer Upgrade (x3)"
+msgstr ""
+
+#: init.lua
+msgid "Obsidian Drawer Upgrade (x5)"
+msgstr ""
+
+#: init.lua
+msgid "Diamond Drawer Upgrade (x8)"
+msgstr ""
+
+#: init.lua
+#, fuzzy
+msgid "Iron Drawer Upgrade (x2)"
+msgstr "@1 Szuflady (1x2)"
+
+#: init.lua
+msgid "Emerald Drawer Upgrade (x13)"
+msgstr ""
+
+#: init.lua
+msgid "Drawer Upgrade Template"
+msgstr ""
diff --git a/locale/template.pot b/locale/template.pot
index 05bb0ef..e265169 100644
--- a/locale/template.pot
+++ b/locale/template.pot
@@ -8,7 +8,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: PACKAGE VERSION\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-06-03 17:04+0200\n"
+"POT-Creation-Date: 2018-06-16 09:14+0200\n"
 "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
 "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
 "Language-Team: LANGUAGE <LL@li.org>\n"
-- 
cgit v1.2.3


From e8ee40a141807857503d1368d9154a6f60b8ab63 Mon Sep 17 00:00:00 2001
From: isaiah658 <isaiah658@outlook.com>
Date: Mon, 18 Jun 2018 11:40:19 -0500
Subject: Add drawer controller translations (#20)

Added "Drawer Controller", "Current State: ", "Running", "Jammed", and "Stopped" to the
translation files to be translated.

While testing out the drawer controller translations, I noticed that in certain cases the entity
infotext wasn't being translated. This fixes that.
---
 locale/be.po        | 38 +++++++++++++++++++++++++++++---------
 locale/cs.po        | 34 +++++++++++++++++++++++++++-------
 locale/de.po        | 34 +++++++++++++++++++++++++++-------
 locale/es.po        | 34 +++++++++++++++++++++++++++-------
 locale/fr.po        | 34 +++++++++++++++++++++++++++-------
 locale/id.po        | 38 +++++++++++++++++++++++++++++---------
 locale/it.po        | 34 +++++++++++++++++++++++++++-------
 locale/ms.po        | 38 +++++++++++++++++++++++++++++---------
 locale/nl.po        | 34 +++++++++++++++++++++++++++-------
 locale/pl.po        | 34 +++++++++++++++++++++++++++-------
 locale/ru.po        | 38 +++++++++++++++++++++++++++++---------
 locale/template.pot | 30 +++++++++++++++++++++++++-----
 locale/tr.po        | 34 +++++++++++++++++++++++++++-------
 lua/api.lua         |  2 +-
 lua/controller.lua  | 28 ++++++++++++++++------------
 15 files changed, 374 insertions(+), 110 deletions(-)

diff --git a/locale/be.po b/locale/be.po
index c7ff613..8702ab5 100644
--- a/locale/be.po
+++ b/locale/be.po
@@ -7,7 +7,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: PACKAGE VERSION\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-06-03 17:04+0200\n"
+"POT-Creation-Date: 2018-06-17 11:36-0500\n"
 "PO-Revision-Date: 2017-06-03 21:34+0000\n"
 "Last-Translator: Viktar Vauchkevich <victorenator@gmail.com>\n"
 "Language-Team: Belarusian <https://hosted.weblate.org/projects/minetest/mod-"
@@ -16,10 +16,18 @@ msgstr ""
 "MIME-Version: 1.0\n"
 "Content-Type: text/plain; charset=UTF-8\n"
 "Content-Transfer-Encoding: 8bit\n"
-"Plural-Forms: nplurals=3; plural=n%10==1 && n%100!=11 ? 0 : n%10>=2 && n%10<="
-"4 && (n%100<10 || n%100>=20) ? 1 : 2;\n"
+"Plural-Forms: nplurals=3; plural=n%10==1 && n%100!=11 ? 0 : n%10>=2 && n"
+"%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2;\n"
 "X-Generator: Weblate 2.15-dev\n"
 
+#: lua/helpers.lua
+msgid "@1 (@2% full)"
+msgstr "@1 (@2% поўны)"
+
+#: lua/helpers.lua
+msgid "@1 @2 (@3% full)"
+msgstr "@1 @2 (@3% поўны)"
+
 #: lua/visual.lua
 msgid "Empty"
 msgstr "Пустая"
@@ -40,13 +48,25 @@ msgstr "@1 шуфляды (1x2)"
 msgid "@1 Drawers (2x2)"
 msgstr "@1 шуфляды (2x2)"
 
-#: lua/helpers.lua
-msgid "@1 (@2% full)"
-msgstr "@1 (@2% поўны)"
+#: lua/controller.lua
+msgid "Current State: "
+msgstr ""
 
-#: lua/helpers.lua
-msgid "@1 @2 (@3% full)"
-msgstr "@1 @2 (@3% поўны)"
+#: lua/controller.lua
+msgid "Stopped"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Running"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Jammed"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Drawer Controller"
+msgstr ""
 
 #: init.lua
 msgid "Acacia Wood"
diff --git a/locale/cs.po b/locale/cs.po
index fd12442..b1aaee5 100644
--- a/locale/cs.po
+++ b/locale/cs.po
@@ -7,7 +7,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: PACKAGE VERSION\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-06-03 17:04+0200\n"
+"POT-Creation-Date: 2018-06-17 11:36-0500\n"
 "PO-Revision-Date: 2017-06-21 18:49+0000\n"
 "Last-Translator: Jakub Vaněk <vanek.jakub4@seznam.cz>\n"
 "Language-Team: Czech <https://hosted.weblate.org/projects/minetest/mod-"
@@ -19,6 +19,14 @@ msgstr ""
 "Plural-Forms: nplurals=3; plural=(n==1) ? 0 : (n>=2 && n<=4) ? 1 : 2;\n"
 "X-Generator: Weblate 2.15-dev\n"
 
+#: lua/helpers.lua
+msgid "@1 (@2% full)"
+msgstr "@1 (z @2 % plný)"
+
+#: lua/helpers.lua
+msgid "@1 @2 (@3% full)"
+msgstr "@1 @2 (z @3 % plný)"
+
 #: lua/visual.lua
 msgid "Empty"
 msgstr "Prázdný"
@@ -39,13 +47,25 @@ msgstr "Šuplíky @1 (1x2)"
 msgid "@1 Drawers (2x2)"
 msgstr "Šuplíky @1 (2x2)"
 
-#: lua/helpers.lua
-msgid "@1 (@2% full)"
-msgstr "@1 (z @2 % plný)"
+#: lua/controller.lua
+msgid "Current State: "
+msgstr ""
 
-#: lua/helpers.lua
-msgid "@1 @2 (@3% full)"
-msgstr "@1 @2 (z @3 % plný)"
+#: lua/controller.lua
+msgid "Stopped"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Running"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Jammed"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Drawer Controller"
+msgstr ""
 
 #: init.lua
 msgid "Acacia Wood"
diff --git a/locale/de.po b/locale/de.po
index 23b73e4..6f710b2 100644
--- a/locale/de.po
+++ b/locale/de.po
@@ -7,7 +7,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: PACKAGE VERSION\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-06-03 17:04+0200\n"
+"POT-Creation-Date: 2018-06-17 11:36-0500\n"
 "PO-Revision-Date: 2017-10-23 23:46+0000\n"
 "Last-Translator: Wuzzy <almikes@aol.com>\n"
 "Language-Team: German <https://hosted.weblate.org/projects/minetest/mod-"
@@ -19,6 +19,14 @@ msgstr ""
 "Plural-Forms: nplurals=2; plural=n != 1;\n"
 "X-Generator: Weblate 2.17\n"
 
+#: lua/helpers.lua
+msgid "@1 (@2% full)"
+msgstr "@1 (@2% voll)"
+
+#: lua/helpers.lua
+msgid "@1 @2 (@3% full)"
+msgstr "@1 @2 (@3% voll)"
+
 #: lua/visual.lua
 msgid "Empty"
 msgstr "Leer"
@@ -39,13 +47,25 @@ msgstr "@1schubfächer (1×2)"
 msgid "@1 Drawers (2x2)"
 msgstr "@1schubfächer (2×2)"
 
-#: lua/helpers.lua
-msgid "@1 (@2% full)"
-msgstr "@1 (@2% voll)"
+#: lua/controller.lua
+msgid "Current State: "
+msgstr ""
 
-#: lua/helpers.lua
-msgid "@1 @2 (@3% full)"
-msgstr "@1 @2 (@3% voll)"
+#: lua/controller.lua
+msgid "Stopped"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Running"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Jammed"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Drawer Controller"
+msgstr ""
 
 #: init.lua
 msgid "Acacia Wood"
diff --git a/locale/es.po b/locale/es.po
index d38f84e..032f8d4 100644
--- a/locale/es.po
+++ b/locale/es.po
@@ -7,7 +7,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: PACKAGE VERSION\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2018-06-16 09:14+0200\n"
+"POT-Creation-Date: 2018-06-17 11:36-0500\n"
 "PO-Revision-Date: 2017-06-01 21:02+0000\n"
 "Last-Translator: Miguel Isaac <mineyoshist@protonmail.ch>\n"
 "Language-Team: Spanish <https://hosted.weblate.org/projects/minetest/mod-"
@@ -19,6 +19,14 @@ msgstr ""
 "Plural-Forms: nplurals=2; plural=n != 1;\n"
 "X-Generator: Weblate 2.15-dev\n"
 
+#: lua/helpers.lua
+msgid "@1 (@2% full)"
+msgstr "@1 (@2% lleno)"
+
+#: lua/helpers.lua
+msgid "@1 @2 (@3% full)"
+msgstr "@1 @2 (@3% lleno)"
+
 #: lua/visual.lua
 msgid "Empty"
 msgstr "Vacio"
@@ -39,13 +47,25 @@ msgstr "@1 Gavetas (1x2)"
 msgid "@1 Drawers (2x2)"
 msgstr "@1 Gavetas (2x2)"
 
-#: lua/helpers.lua
-msgid "@1 (@2% full)"
-msgstr "@1 (@2% lleno)"
+#: lua/controller.lua
+msgid "Current State: "
+msgstr ""
 
-#: lua/helpers.lua
-msgid "@1 @2 (@3% full)"
-msgstr "@1 @2 (@3% lleno)"
+#: lua/controller.lua
+msgid "Stopped"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Running"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Jammed"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Drawer Controller"
+msgstr ""
 
 #: init.lua
 msgid "Acacia Wood"
diff --git a/locale/fr.po b/locale/fr.po
index 986f534..991b766 100644
--- a/locale/fr.po
+++ b/locale/fr.po
@@ -7,7 +7,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: PACKAGE VERSION\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-06-03 17:04+0200\n"
+"POT-Creation-Date: 2018-06-17 11:36-0500\n"
 "PO-Revision-Date: 2017-10-15 17:45+0000\n"
 "Last-Translator: xin <xin@riseup.net>\n"
 "Language-Team: French <https://hosted.weblate.org/projects/minetest/mod-"
@@ -19,6 +19,14 @@ msgstr ""
 "Plural-Forms: nplurals=2; plural=n > 1;\n"
 "X-Generator: Weblate 2.17\n"
 
+#: lua/helpers.lua
+msgid "@1 (@2% full)"
+msgstr "@1 (@2% plein)"
+
+#: lua/helpers.lua
+msgid "@1 @2 (@3% full)"
+msgstr "@1 @2 (@3% plein)"
+
 #: lua/visual.lua
 msgid "Empty"
 msgstr "Vide"
@@ -39,13 +47,25 @@ msgstr "Tiroirs (1x2) @1"
 msgid "@1 Drawers (2x2)"
 msgstr "Tiroirs (2x2) @1"
 
-#: lua/helpers.lua
-msgid "@1 (@2% full)"
-msgstr "@1 (@2% plein)"
+#: lua/controller.lua
+msgid "Current State: "
+msgstr ""
 
-#: lua/helpers.lua
-msgid "@1 @2 (@3% full)"
-msgstr "@1 @2 (@3% plein)"
+#: lua/controller.lua
+msgid "Stopped"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Running"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Jammed"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Drawer Controller"
+msgstr ""
 
 #: init.lua
 msgid "Acacia Wood"
diff --git a/locale/id.po b/locale/id.po
index 94ea4f9..dbf680c 100644
--- a/locale/id.po
+++ b/locale/id.po
@@ -7,12 +7,12 @@ msgid ""
 msgstr ""
 "Project-Id-Version: PACKAGE VERSION\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-06-03 17:04+0200\n"
+"POT-Creation-Date: 2018-06-17 11:36-0500\n"
 "PO-Revision-Date: 2018-05-20 16:38+0000\n"
 "Last-Translator: Muhammad Rifqi Priyo Susanto "
 "<muhammadrifqipriyosusanto@gmail.com>\n"
-"Language-Team: Indonesian <https://hosted.weblate.org/projects/minetest/"
-"mod-storage-drawers/id/>\n"
+"Language-Team: Indonesian <https://hosted.weblate.org/projects/minetest/mod-"
+"storage-drawers/id/>\n"
 "Language: id\n"
 "MIME-Version: 1.0\n"
 "Content-Type: text/plain; charset=UTF-8\n"
@@ -20,6 +20,14 @@ msgstr ""
 "Plural-Forms: nplurals=1; plural=0;\n"
 "X-Generator: Weblate 3.0-dev\n"
 
+#: lua/helpers.lua
+msgid "@1 (@2% full)"
+msgstr "@1 (@2% penuh)"
+
+#: lua/helpers.lua
+msgid "@1 @2 (@3% full)"
+msgstr "@1 @2 (@3% penuh)"
+
 #: lua/visual.lua
 msgid "Empty"
 msgstr "Kosong"
@@ -40,13 +48,25 @@ msgstr "@1 Laci (1x2)"
 msgid "@1 Drawers (2x2)"
 msgstr "@1 Laci (2x2)"
 
-#: lua/helpers.lua
-msgid "@1 (@2% full)"
-msgstr "@1 (@2% penuh)"
+#: lua/controller.lua
+msgid "Current State: "
+msgstr ""
 
-#: lua/helpers.lua
-msgid "@1 @2 (@3% full)"
-msgstr "@1 @2 (@3% penuh)"
+#: lua/controller.lua
+msgid "Stopped"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Running"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Jammed"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Drawer Controller"
+msgstr ""
 
 #: init.lua
 msgid "Acacia Wood"
diff --git a/locale/it.po b/locale/it.po
index e459a87..c71f1f2 100644
--- a/locale/it.po
+++ b/locale/it.po
@@ -7,7 +7,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: PACKAGE VERSION\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-06-03 17:04+0200\n"
+"POT-Creation-Date: 2018-06-17 11:36-0500\n"
 "PO-Revision-Date: 2017-09-28 16:09+0000\n"
 "Last-Translator: Hamlet <h4mlet@riseup.net>\n"
 "Language-Team: Italian <https://hosted.weblate.org/projects/minetest/mod-"
@@ -19,6 +19,14 @@ msgstr ""
 "Plural-Forms: nplurals=2; plural=n != 1;\n"
 "X-Generator: Weblate 2.17-dev\n"
 
+#: lua/helpers.lua
+msgid "@1 (@2% full)"
+msgstr "@1 (@2% pieno)"
+
+#: lua/helpers.lua
+msgid "@1 @2 (@3% full)"
+msgstr "@1 @2 (@3% pieno)"
+
 #: lua/visual.lua
 msgid "Empty"
 msgstr "Vuoto"
@@ -39,13 +47,25 @@ msgstr "@1 Cassetti (1x2)"
 msgid "@1 Drawers (2x2)"
 msgstr "@1 Cassetti (2x2)"
 
-#: lua/helpers.lua
-msgid "@1 (@2% full)"
-msgstr "@1 (@2% pieno)"
+#: lua/controller.lua
+msgid "Current State: "
+msgstr ""
 
-#: lua/helpers.lua
-msgid "@1 @2 (@3% full)"
-msgstr "@1 @2 (@3% pieno)"
+#: lua/controller.lua
+msgid "Stopped"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Running"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Jammed"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Drawer Controller"
+msgstr ""
 
 #: init.lua
 msgid "Acacia Wood"
diff --git a/locale/ms.po b/locale/ms.po
index 1f6fb99..df7e0fc 100644
--- a/locale/ms.po
+++ b/locale/ms.po
@@ -7,10 +7,10 @@ msgid ""
 msgstr ""
 "Project-Id-Version: PACKAGE VERSION\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-06-03 17:04+0200\n"
+"POT-Creation-Date: 2018-06-17 11:36-0500\n"
 "PO-Revision-Date: 2017-07-01 04:31+0000\n"
-"Last-Translator: Muhammad Nur Hidayat Yasuyoshi <muhdnurhidayat96@yahoo.com>"
-"\n"
+"Last-Translator: Muhammad Nur Hidayat Yasuyoshi <muhdnurhidayat96@yahoo."
+"com>\n"
 "Language-Team: Malay <https://hosted.weblate.org/projects/minetest/mod-"
 "storage-drawers/ms/>\n"
 "Language: ms\n"
@@ -20,6 +20,14 @@ msgstr ""
 "Plural-Forms: nplurals=1; plural=0;\n"
 "X-Generator: Weblate 2.16-dev\n"
 
+#: lua/helpers.lua
+msgid "@1 (@2% full)"
+msgstr "@1 (@2% penuh)"
+
+#: lua/helpers.lua
+msgid "@1 @2 (@3% full)"
+msgstr "@1 @2 (@3% penuh)"
+
 #: lua/visual.lua
 msgid "Empty"
 msgstr "Kosong"
@@ -40,13 +48,25 @@ msgstr "Laci @1 (1x2)"
 msgid "@1 Drawers (2x2)"
 msgstr "Laci @1 (2x2)"
 
-#: lua/helpers.lua
-msgid "@1 (@2% full)"
-msgstr "@1 (@2% penuh)"
+#: lua/controller.lua
+msgid "Current State: "
+msgstr ""
 
-#: lua/helpers.lua
-msgid "@1 @2 (@3% full)"
-msgstr "@1 @2 (@3% penuh)"
+#: lua/controller.lua
+msgid "Stopped"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Running"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Jammed"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Drawer Controller"
+msgstr ""
 
 #: init.lua
 msgid "Acacia Wood"
diff --git a/locale/nl.po b/locale/nl.po
index fc93b67..d27801c 100644
--- a/locale/nl.po
+++ b/locale/nl.po
@@ -7,7 +7,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: PACKAGE VERSION\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-06-03 17:04+0200\n"
+"POT-Creation-Date: 2018-06-17 11:36-0500\n"
 "PO-Revision-Date: 2017-09-19 19:45+0000\n"
 "Last-Translator: Oscar <kingoscargames@gmail.com>\n"
 "Language-Team: Dutch <https://hosted.weblate.org/projects/minetest/mod-"
@@ -19,6 +19,14 @@ msgstr ""
 "Plural-Forms: nplurals=2; plural=n != 1;\n"
 "X-Generator: Weblate 2.17-dev\n"
 
+#: lua/helpers.lua
+msgid "@1 (@2% full)"
+msgstr "@1 (@2% vol)"
+
+#: lua/helpers.lua
+msgid "@1 @2 (@3% full)"
+msgstr "@1 @2 (@3% vol)"
+
 #: lua/visual.lua
 msgid "Empty"
 msgstr "Leeg"
@@ -39,13 +47,25 @@ msgstr "@1 Lades (1x2)"
 msgid "@1 Drawers (2x2)"
 msgstr "@1 Lades (2x2)"
 
-#: lua/helpers.lua
-msgid "@1 (@2% full)"
-msgstr "@1 (@2% vol)"
+#: lua/controller.lua
+msgid "Current State: "
+msgstr ""
 
-#: lua/helpers.lua
-msgid "@1 @2 (@3% full)"
-msgstr "@1 @2 (@3% vol)"
+#: lua/controller.lua
+msgid "Stopped"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Running"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Jammed"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Drawer Controller"
+msgstr ""
 
 #: init.lua
 msgid "Acacia Wood"
diff --git a/locale/pl.po b/locale/pl.po
index 0087706..f508c7f 100644
--- a/locale/pl.po
+++ b/locale/pl.po
@@ -7,7 +7,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: PACKAGE VERSION\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2018-06-16 09:14+0200\n"
+"POT-Creation-Date: 2018-06-17 11:36-0500\n"
 "PO-Revision-Date: 2017-06-04 23:03+0000\n"
 "Last-Translator: LNJ <git@lnj.li>\n"
 "Language-Team: Polish <https://hosted.weblate.org/projects/minetest/mod-"
@@ -20,6 +20,14 @@ msgstr ""
 "|| n%100>=20) ? 1 : 2;\n"
 "X-Generator: Weblate 2.15-dev\n"
 
+#: lua/helpers.lua
+msgid "@1 (@2% full)"
+msgstr "@1 (@2% napełnione)"
+
+#: lua/helpers.lua
+msgid "@1 @2 (@3% full)"
+msgstr "@1 @2 (@3% napełnione)"
+
 #: lua/visual.lua
 msgid "Empty"
 msgstr "Pusta"
@@ -40,13 +48,25 @@ msgstr "@1 Szuflady (1x2)"
 msgid "@1 Drawers (2x2)"
 msgstr "@1 Szuflady (2x2)"
 
-#: lua/helpers.lua
-msgid "@1 (@2% full)"
-msgstr "@1 (@2% napełnione)"
+#: lua/controller.lua
+msgid "Current State: "
+msgstr ""
 
-#: lua/helpers.lua
-msgid "@1 @2 (@3% full)"
-msgstr "@1 @2 (@3% napełnione)"
+#: lua/controller.lua
+msgid "Stopped"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Running"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Jammed"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Drawer Controller"
+msgstr ""
 
 #: init.lua
 msgid "Acacia Wood"
diff --git a/locale/ru.po b/locale/ru.po
index e367590..7af4583 100644
--- a/locale/ru.po
+++ b/locale/ru.po
@@ -7,7 +7,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: PACKAGE VERSION\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-06-03 17:04+0200\n"
+"POT-Creation-Date: 2018-06-17 11:36-0500\n"
 "PO-Revision-Date: 2017-09-04 21:55+0000\n"
 "Last-Translator: pitchblack <pitchblack@mail.ru>\n"
 "Language-Team: Russian <https://hosted.weblate.org/projects/minetest/mod-"
@@ -16,10 +16,18 @@ msgstr ""
 "MIME-Version: 1.0\n"
 "Content-Type: text/plain; charset=UTF-8\n"
 "Content-Transfer-Encoding: 8bit\n"
-"Plural-Forms: nplurals=3; plural=n%10==1 && n%100!=11 ? 0 : n%10>=2 && n%10<="
-"4 && (n%100<10 || n%100>=20) ? 1 : 2;\n"
+"Plural-Forms: nplurals=3; plural=n%10==1 && n%100!=11 ? 0 : n%10>=2 && n"
+"%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2;\n"
 "X-Generator: Weblate 2.17-dev\n"
 
+#: lua/helpers.lua
+msgid "@1 (@2% full)"
+msgstr "@1 (@2% полон)"
+
+#: lua/helpers.lua
+msgid "@1 @2 (@3% full)"
+msgstr "@1 @2 (@3% полон)"
+
 #: lua/visual.lua
 msgid "Empty"
 msgstr "Пусто"
@@ -40,13 +48,25 @@ msgstr "@1 Ящики (1x2)"
 msgid "@1 Drawers (2x2)"
 msgstr "@1 Ящики (2x2)"
 
-#: lua/helpers.lua
-msgid "@1 (@2% full)"
-msgstr "@1 (@2% полон)"
+#: lua/controller.lua
+msgid "Current State: "
+msgstr ""
 
-#: lua/helpers.lua
-msgid "@1 @2 (@3% full)"
-msgstr "@1 @2 (@3% полон)"
+#: lua/controller.lua
+msgid "Stopped"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Running"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Jammed"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Drawer Controller"
+msgstr ""
 
 #: init.lua
 msgid "Acacia Wood"
diff --git a/locale/template.pot b/locale/template.pot
index e265169..ada429b 100644
--- a/locale/template.pot
+++ b/locale/template.pot
@@ -8,7 +8,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: PACKAGE VERSION\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2018-06-16 09:14+0200\n"
+"POT-Creation-Date: 2018-06-17 11:36-0500\n"
 "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
 "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
 "Language-Team: LANGUAGE <LL@li.org>\n"
@@ -17,6 +17,14 @@ msgstr ""
 "Content-Type: text/plain; charset=CHARSET\n"
 "Content-Transfer-Encoding: 8bit\n"
 
+#: lua/helpers.lua
+msgid "@1 (@2% full)"
+msgstr ""
+
+#: lua/helpers.lua
+msgid "@1 @2 (@3% full)"
+msgstr ""
+
 #: lua/visual.lua
 msgid "Empty"
 msgstr ""
@@ -37,12 +45,24 @@ msgstr ""
 msgid "@1 Drawers (2x2)"
 msgstr ""
 
-#: lua/helpers.lua
-msgid "@1 (@2% full)"
+#: lua/controller.lua
+msgid "Current State: "
 msgstr ""
 
-#: lua/helpers.lua
-msgid "@1 @2 (@3% full)"
+#: lua/controller.lua
+msgid "Stopped"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Running"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Jammed"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Drawer Controller"
 msgstr ""
 
 #: init.lua
diff --git a/locale/tr.po b/locale/tr.po
index 658dc22..7e14532 100644
--- a/locale/tr.po
+++ b/locale/tr.po
@@ -7,7 +7,7 @@ msgid ""
 msgstr ""
 "Project-Id-Version: PACKAGE VERSION\n"
 "Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2017-06-03 17:04+0200\n"
+"POT-Creation-Date: 2018-06-17 11:36-0500\n"
 "PO-Revision-Date: 2017-06-04 14:23+0000\n"
 "Last-Translator: monolifed <monolifed@gmail.com>\n"
 "Language-Team: Turkish <https://hosted.weblate.org/projects/minetest/mod-"
@@ -19,6 +19,14 @@ msgstr ""
 "Plural-Forms: nplurals=2; plural=n > 1;\n"
 "X-Generator: Weblate 2.15-dev\n"
 
+#: lua/helpers.lua
+msgid "@1 (@2% full)"
+msgstr "@1 (%@2 dolu)"
+
+#: lua/helpers.lua
+msgid "@1 @2 (@3% full)"
+msgstr "@1 @2 (%@3 dolu)"
+
 #: lua/visual.lua
 msgid "Empty"
 msgstr "Boş"
@@ -39,13 +47,25 @@ msgstr "@1 Çekmece (1x2)"
 msgid "@1 Drawers (2x2)"
 msgstr "@1 Çekmece (2x2)"
 
-#: lua/helpers.lua
-msgid "@1 (@2% full)"
-msgstr "@1 (%@2 dolu)"
+#: lua/controller.lua
+msgid "Current State: "
+msgstr ""
 
-#: lua/helpers.lua
-msgid "@1 @2 (@3% full)"
-msgstr "@1 @2 (%@3 dolu)"
+#: lua/controller.lua
+msgid "Stopped"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Running"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Jammed"
+msgstr ""
+
+#: lua/controller.lua
+msgid "Drawer Controller"
+msgstr ""
 
 #: init.lua
 msgid "Acacia Wood"
diff --git a/lua/api.lua b/lua/api.lua
index d12a17c..6c87f94 100755
--- a/lua/api.lua
+++ b/lua/api.lua
@@ -67,7 +67,7 @@ function drawers.drawer_on_construct(pos)
 		meta:set_int("count"..vid, 0)
 		meta:set_int("max_count"..vid, base_stack_max * stack_max_factor)
 		meta:set_int("base_stack_max"..vid, base_stack_max)
-		meta:set_string("entity_infotext"..vid, drawers.gen_info_text("Empty", 0,
+		meta:set_string("entity_infotext"..vid, drawers.gen_info_text(S("Empty"), 0,
 			stack_max_factor, base_stack_max))
 		meta:set_int("stack_max_factor"..vid, stack_max_factor)
 
diff --git a/lua/controller.lua b/lua/controller.lua
index 315ae5c..6582c78 100644
--- a/lua/controller.lua
+++ b/lua/controller.lua
@@ -39,6 +39,10 @@ needing to scan all the drawers to deposit an item in certain situations. The
 table is only updated on an as needed basis, not by a specific time/interval. 
 Controllers that have no items will not continue scanning drawers. ]]--
 
+-- Load support for intllib.
+local MP = core.get_modpath(core.get_current_modname())
+local S, NS = dofile(MP.."/intllib.lua")
+
 local function controller_can_dig(pos, player)
 	local meta = core.get_meta(pos);
 	local inv = meta:get_inventory()
@@ -74,7 +78,7 @@ local function controller_formspec(pos, meta_current_state)
 		default.gui_bg..
 		default.gui_bg_img..
 		default.gui_slots..
-		"label[0,0;Current State: " .. meta_current_state .. "]" ..
+		"label[0,0;" .. S("Current State: ") .. meta_current_state .. "]" ..
 		"list[current_name;src;3.5,1.75;1,1;]"..
 		"list[current_player;main;0,4.25;8,1;]"..
 		"list[current_player;main;0,5.5;8,3;8]"..
@@ -217,7 +221,7 @@ local function controller_node_timer(pos, elapsed)
 	clear it each time ]]--
 	if meta_current_state == "jammed" and meta_jammed_item_name == src_name and meta_times_ran_while_jammed >= 8 then
 		meta:set_string("current_state", "stopped")
-		meta:set_string("formspec", controller_formspec(pos, "Stopped"))
+		meta:set_string("formspec", controller_formspec(pos, S("Stopped")))
 		return true
 	end
 	
@@ -229,14 +233,14 @@ local function controller_node_timer(pos, elapsed)
 	-- If current state is stopped, and the item that jammed it is not the same item in the src inv slot, we set the current state to running and clear the jam counter
 	if meta_current_state == "stopped" and meta_jammed_item_name ~= src_name then
 		meta:set_string("current_state", "running")
-		meta:set_string("formspec", controller_formspec(pos, "Running"))
+		meta:set_string("formspec", controller_formspec(pos, S("Running")))
 		meta:set_float("times_ran_while_jammed", 0)
 	end
 	
 	-- If no item is in the controller, nothing is searched and current_state is set to running and no jams
 	if inv:is_empty("src") then
 		meta:set_string("current_state", "running")
-		meta:set_string("formspec", controller_formspec(pos, "Running"))
+		meta:set_string("formspec", controller_formspec(pos, S("Running")))
 		meta:set_float("times_ran_while_jammed", 0)
 		return true
 	end
@@ -244,7 +248,7 @@ local function controller_node_timer(pos, elapsed)
 	-- If a non stackable item is in the controller, such as a written book, set the current_state to stopped because they are not allowed in drawers
 	if src:get_stack_max() == 1 then
 		meta:set_string("current_state", "stopped")
-		meta:set_string("formspec", controller_formspec(pos, "Stopped"))
+		meta:set_string("formspec", controller_formspec(pos, S("Stopped")))
 		meta:set_string("jammed_item_name", src_name)
 		meta:set_float("times_ran_while_jammed", 1)
 		return true
@@ -295,11 +299,11 @@ local function controller_node_timer(pos, elapsed)
 			inv:set_stack("src", 1, leftover)
 			-- Set the controller metadata
 			meta:set_string("current_state", "running")
-			meta:set_string("formspec", controller_formspec(pos, "Running"))
+			meta:set_string("formspec", controller_formspec(pos, S("Running")))
 			meta:set_float("times_ran_while_jammed", 0)
 		else
 			meta:set_string("current_state", "jammed")
-			meta:set_string("formspec", controller_formspec(pos, "Jammed"))
+			meta:set_string("formspec", controller_formspec(pos, S("Jammed")))
 			meta:set_string("jammed_item_name", src_name)
 			meta:set_float("times_ran_while_jammed", meta_times_ran_while_jammed + 1)
 		end
@@ -317,18 +321,18 @@ local function controller_node_timer(pos, elapsed)
 			drawers_table_index[src_name] = {drawer_pos_x = indexed_drawer_pos.x, drawer_pos_y = indexed_drawer_pos.y, drawer_pos_z = indexed_drawer_pos.z, visualid = visualid}
 			-- Set the controller metadata
 			meta:set_string("current_state", "running")
-			meta:set_string("formspec", controller_formspec(pos, "Running"))
+			meta:set_string("formspec", controller_formspec(pos, S("Running")))
 			meta:set_float("times_ran_while_jammed", 0)
 			meta:set_string("drawers_table_index", core.serialize(drawers_table_index))
 		else
 			meta:set_string("current_state", "jammed")
-			meta:set_string("formspec", controller_formspec(pos, "Jammed"))
+			meta:set_string("formspec", controller_formspec(pos, S("Jammed")))
 			meta:set_string("jammed_item_name", src_name)
 			meta:set_float("times_ran_while_jammed", meta_times_ran_while_jammed + 1)
 		end
 	else
 		meta:set_string("current_state", "jammed")
-		meta:set_string("formspec", controller_formspec(pos, "Jammed"))
+		meta:set_string("formspec", controller_formspec(pos, S("Jammed")))
 		meta:set_string("jammed_item_name", src_name)
 		meta:set_float("times_ran_while_jammed", meta_times_ran_while_jammed + 1)
 	end
@@ -338,7 +342,7 @@ end
 
 -- Set the controller definition using a table to allow for pipeworks and potentially other mod support
 local controller_def = {}
-controller_def.description = 'Drawer Controller'
+controller_def.description = S("Drawer Controller")
 controller_def.tiles = {"drawer_controller_top_bottom.png", "drawer_controller_top_bottom.png", "drawer_controller_side.png", "drawer_controller_side.png", "drawer_controller_side.png", "drawer_controller_side.png"}
 controller_def.can_dig = controller_can_dig
 controller_def.groups = {cracky = 3, level = 2}
@@ -350,7 +354,7 @@ controller_def.on_construct = function(pos)
 	meta:set_float("times_ran_while_jammed", 0)
 	meta:set_string("jammed_item_name", "")
 	meta:set_string("drawers_table_index", "")
-	meta:set_string("formspec", controller_formspec(pos, "Running"))
+	meta:set_string("formspec", controller_formspec(pos, S("Running")))
 	local timer = core.get_node_timer(pos)
 	timer:start(7)
 end
-- 
cgit v1.2.3


From 279d0ab9e8775b543034682f82b8a71cf150c9c9 Mon Sep 17 00:00:00 2001
From: LNJ <git@lnj.li>
Date: Mon, 18 Jun 2018 16:41:30 +0000
Subject: Translated using Weblate (German)

Currently translated at 100.0% (27 of 27 strings)

Translation: Minetest/Mod: Storage Drawers
Translate-URL: https://hosted.weblate.org/projects/minetest/mod-storage-drawers/de/
---
 locale/de.po | 20 ++++++++++----------
 1 file changed, 10 insertions(+), 10 deletions(-)

diff --git a/locale/de.po b/locale/de.po
index 6f710b2..c11e686 100644
--- a/locale/de.po
+++ b/locale/de.po
@@ -8,16 +8,16 @@ msgstr ""
 "Project-Id-Version: PACKAGE VERSION\n"
 "Report-Msgid-Bugs-To: \n"
 "POT-Creation-Date: 2018-06-17 11:36-0500\n"
-"PO-Revision-Date: 2017-10-23 23:46+0000\n"
-"Last-Translator: Wuzzy <almikes@aol.com>\n"
-"Language-Team: German <https://hosted.weblate.org/projects/minetest/mod-"
-"storage-drawers/de/>\n"
+"PO-Revision-Date: 2018-06-18 16:42+0000\n"
+"Last-Translator: LNJ <git@lnj.li>\n"
+"Language-Team: German <https://hosted.weblate.org/projects/minetest/"
+"mod-storage-drawers/de/>\n"
 "Language: de\n"
 "MIME-Version: 1.0\n"
 "Content-Type: text/plain; charset=UTF-8\n"
 "Content-Transfer-Encoding: 8bit\n"
 "Plural-Forms: nplurals=2; plural=n != 1;\n"
-"X-Generator: Weblate 2.17\n"
+"X-Generator: Weblate 3.0.1\n"
 
 #: lua/helpers.lua
 msgid "@1 (@2% full)"
@@ -49,23 +49,23 @@ msgstr "@1schubfächer (2×2)"
 
 #: lua/controller.lua
 msgid "Current State: "
-msgstr ""
+msgstr "Momentaner Zustand: "
 
 #: lua/controller.lua
 msgid "Stopped"
-msgstr ""
+msgstr "Angehalten"
 
 #: lua/controller.lua
 msgid "Running"
-msgstr ""
+msgstr "Läuft"
 
 #: lua/controller.lua
 msgid "Jammed"
-msgstr ""
+msgstr "Gestaut"
 
 #: lua/controller.lua
 msgid "Drawer Controller"
-msgstr ""
+msgstr "Schubfach Steuerung"
 
 #: init.lua
 msgid "Acacia Wood"
-- 
cgit v1.2.3


From d8e1d83fd6a6db04fdc96bc77e5018d2c3f8345e Mon Sep 17 00:00:00 2001
From: Linus Jahn <lnj@kaidan.im>
Date: Mon, 10 Sep 2018 19:57:23 +0200
Subject: MCL2: Fix drawer upgrade inventory background

---
 init.lua | 6 +++++-
 1 file changed, 5 insertions(+), 1 deletion(-)

diff --git a/init.lua b/init.lua
index a7cc3d9..2e079eb 100755
--- a/init.lua
+++ b/init.lua
@@ -57,8 +57,12 @@ drawers.enable_2x2 = not core.settings:get_bool("drawers_disable_2x2")
 --
 
 drawers.gui_bg = "bgcolor[#080808BB;true]"
-drawers.gui_bg_img = "background[5,5;1,1;gui_formbg.png;true]"
 drawers.gui_slots = "listcolors[#00000069;#5A5A5A;#141318;#30434C;#FFF]"
+if (core.get_modpath("mcl_core")) and mcl_core then -- MCL2
+	drawers.gui_bg_img = "background[5,5;1,1;crafting_creative_bg.png;true]"
+else
+	drawers.gui_bg_img = "background[5,5;1,1;gui_formbg.png;true]"
+end
 
 --
 -- Load API
-- 
cgit v1.2.3


From 8702b6ac927342043697f3930fa14b96131ab3b5 Mon Sep 17 00:00:00 2001
From: Linus Jahn <lnj@kaidan.im>
Date: Mon, 10 Sep 2018 21:23:29 +0200
Subject: Make drawer controller update interval configurable

---
 lua/controller.lua | 4 +++-
 settingtypes.txt   | 1 +
 2 files changed, 4 insertions(+), 1 deletion(-)
 create mode 100644 settingtypes.txt

diff --git a/lua/controller.lua b/lua/controller.lua
index 6582c78..e36ff71 100644
--- a/lua/controller.lua
+++ b/lua/controller.lua
@@ -43,6 +43,8 @@ Controllers that have no items will not continue scanning drawers. ]]--
 local MP = core.get_modpath(core.get_current_modname())
 local S, NS = dofile(MP.."/intllib.lua")
 
+local controller_interval = tonumber(core.setting_get("drawers_controller_interval"))
+
 local function controller_can_dig(pos, player)
 	local meta = core.get_meta(pos);
 	local inv = meta:get_inventory()
@@ -356,7 +358,7 @@ controller_def.on_construct = function(pos)
 	meta:set_string("drawers_table_index", "")
 	meta:set_string("formspec", controller_formspec(pos, S("Running")))
 	local timer = core.get_node_timer(pos)
-	timer:start(7)
+	timer:start(controller_interval)
 end
 controller_def.on_blast = function(pos)
 	local drops = {}
diff --git a/settingtypes.txt b/settingtypes.txt
new file mode 100644
index 0000000..8175c15
--- /dev/null
+++ b/settingtypes.txt
@@ -0,0 +1 @@
+drawers_controller_interval (Drawer Controller update interval in seconds) float 7.0 0.1 10.0
-- 
cgit v1.2.3


From bc48fc0c077a453c20d758318eee6ad6d06783b9 Mon Sep 17 00:00:00 2001
From: Linus Jahn <lnj@kaidan.im>
Date: Tue, 11 Sep 2018 17:52:07 +0200
Subject: Add Wooden Trim for easy drawer network connections

This node is only used for connecting drawers, so that they are all in one
drawer network and can be used with a drawer controller.
---
 LICENSE.txt               |   1 +
 init.lua                  |  18 ++++++++++++++++++
 textures/drawers_trim.png | Bin 0 -> 180 bytes
 3 files changed, 19 insertions(+)
 create mode 100644 textures/drawers_trim.png

diff --git a/LICENSE.txt b/LICENSE.txt
index 3ff1ed4..60a1cdd 100644
--- a/LICENSE.txt
+++ b/LICENSE.txt
@@ -96,6 +96,7 @@ Copyright (C) 2014 Justin Aquadro (MIT):
   textures/drawers_spruce_wood_front_2.png
   textures/drawers_spruce_wood_front_4.png
   textures/drawers_spruce_wood.png
+  textures/drawers_trim.png
   textures/drawers_upgrade_diamond.png
   textures/drawers_upgrade_emerald.png
   textures/drawers_upgrade_gold.png
diff --git a/init.lua b/init.lua
index 2e079eb..ebdf5fb 100755
--- a/init.lua
+++ b/init.lua
@@ -314,6 +314,24 @@ elseif core.get_modpath("mcl_core") and mcl_core then
 	})
 end
 
+--
+-- Register drawer trim
+--
+
+core.register_node("drawers:trim", {
+	description = S("Wooden Trim"),
+	tiles = {"drawers_trim.png"},
+	groups = {drawer = 1, choppy = 3, oddly_breakable_by_hand = 2}
+})
+
+core.register_craft({
+	output = "drawers:trim 6",
+	recipe = {
+		{"group:stick", "group:wood", "group:stick"},
+		{"group:wood",  "group:wood",  "group:wood"},
+		{"group:stick", "group:wood", "group:stick"}
+	}
+})
 
 --
 -- Register drawer upgrade template
diff --git a/textures/drawers_trim.png b/textures/drawers_trim.png
new file mode 100644
index 0000000..ab398c4
Binary files /dev/null and b/textures/drawers_trim.png differ
-- 
cgit v1.2.3