Allow to configure item ratios in distributor filters
It allows to control the ratio in which certain items will use the different outputs. This is done by adding an item multiple times to the same filter. Because of this, the hardcoded permutations are replaced with a Fisher-Yates-Shuffle. This is theoretically a little bit slower; but according to my measurements, about one million of copy-shuffle operations can be executed per second (tested with a table length of four); so it doesn't seem to be a big problem. Additionally, a bit of outdated information was removed from the manuals.
This commit is contained in:
parent
4dac6b3a66
commit
d544224dd6
@ -35,19 +35,6 @@ local SlotColors = {"red", "green", "blue", "yellow"}
|
|||||||
local Num2Ascii = {"B", "L", "F", "R"}
|
local Num2Ascii = {"B", "L", "F", "R"}
|
||||||
local FilterCache = {} -- local cache for filter settings
|
local FilterCache = {} -- local cache for filter settings
|
||||||
|
|
||||||
-- Permutation table to improve distribution between ports (number of ports: 1-4)
|
|
||||||
-- Usage: permIdx[num_ports][math.random(1, #permIdx[num_ports])][idx]
|
|
||||||
local permIdx = {
|
|
||||||
{ { 1 } },
|
|
||||||
{ { 1, 2 }, { 2, 1 } },
|
|
||||||
{ { 1, 2, 3 }, { 1, 3, 2 }, { 2, 1, 3 }, { 2, 3, 1 }, { 3, 1, 2 }, { 3, 2, 1 } },
|
|
||||||
{ { 1, 2, 3, 4 }, { 1, 2, 4, 3 }, { 1, 3, 2, 4 }, { 1, 3, 4, 2 }, { 1, 4, 2, 3 },
|
|
||||||
{ 1, 4, 3, 2 }, { 2, 1, 3, 4 }, { 2, 1, 4, 3 }, { 2, 3, 1, 4 }, { 2, 3, 4, 1 },
|
|
||||||
{ 2, 4, 1, 3 }, { 2, 4, 3, 1 }, { 3, 1, 2, 4 }, { 3, 1, 4, 2 }, { 3, 2, 1, 4 },
|
|
||||||
{ 3, 2, 4, 1 }, { 3, 4, 1, 2 }, { 3, 4, 2, 1 }, { 4, 1, 2, 3 }, { 4, 1, 3, 2 },
|
|
||||||
{ 4, 2, 1, 3 }, { 4, 2, 3, 1 }, { 4, 3, 1, 2 }, { 4, 3, 2, 1 }, }
|
|
||||||
}
|
|
||||||
|
|
||||||
local function filter_settings(pos)
|
local function filter_settings(pos)
|
||||||
local meta = M(pos)
|
local meta = M(pos)
|
||||||
local param2 = techage.get_node_lvm(pos).param2
|
local param2 = techage.get_node_lvm(pos).param2
|
||||||
@ -69,12 +56,14 @@ local function filter_settings(pos)
|
|||||||
if not ItemFilter[name] then
|
if not ItemFilter[name] then
|
||||||
ItemFilter[name] = {}
|
ItemFilter[name] = {}
|
||||||
end
|
end
|
||||||
|
for _ = 1,stack:get_count() do
|
||||||
table.insert(ItemFilter[name], out_dir)
|
table.insert(ItemFilter[name], out_dir)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
end
|
||||||
|
|
||||||
FilterCache[minetest.hash_node_position(pos)] = {
|
FilterCache[minetest.hash_node_position(pos)] = {
|
||||||
ItemFilter = ItemFilter,
|
ItemFilter = ItemFilter,
|
||||||
@ -181,8 +170,8 @@ local function allow_metadata_inventory_put(pos, listname, index, stack, player)
|
|||||||
if listname == "src" then
|
if listname == "src" then
|
||||||
CRD(pos).State:start_if_standby(pos)
|
CRD(pos).State:start_if_standby(pos)
|
||||||
return stack:get_count()
|
return stack:get_count()
|
||||||
elseif list[index]:get_count() == 0 then
|
else
|
||||||
stack:set_count(1)
|
stack:add_item(list[index])
|
||||||
inv:set_stack(listname, index, stack)
|
inv:set_stack(listname, index, stack)
|
||||||
return 0
|
return 0
|
||||||
end
|
end
|
||||||
@ -197,7 +186,9 @@ local function allow_metadata_inventory_take(pos, listname, index, stack, player
|
|||||||
return stack:get_count()
|
return stack:get_count()
|
||||||
else
|
else
|
||||||
local inv = M(pos):get_inventory()
|
local inv = M(pos):get_inventory()
|
||||||
inv:set_stack(listname, index, nil)
|
local list = inv:get_list(listname)
|
||||||
|
list[index]:take_item(stack:get_count())
|
||||||
|
inv:set_stack(listname, index, list[index])
|
||||||
return 0
|
return 0
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@ -209,17 +200,15 @@ local function allow_metadata_inventory_move(pos, from_list, from_index, to_list
|
|||||||
local inv = minetest.get_meta(pos):get_inventory()
|
local inv = minetest.get_meta(pos):get_inventory()
|
||||||
local stack = inv:get_stack(from_list, from_index)
|
local stack = inv:get_stack(from_list, from_index)
|
||||||
|
|
||||||
if from_list == "src" and to_list ~= "src" and not inv:contains_item(to_list, {name = stack:get_name()}) then
|
if from_list == "src" and to_list ~= "src" then
|
||||||
stack:set_count(1)
|
stack:add_item(to_list[to_index])
|
||||||
inv:set_stack(to_list, to_index, stack)
|
inv:set_stack(to_list, to_index, stack)
|
||||||
return 0
|
return 0
|
||||||
elseif from_list ~= "src" and to_list == "src" then
|
elseif from_list ~= "src" and to_list == "src" then
|
||||||
inv:set_stack(from_list, from_index, nil)
|
inv:set_stack(from_list, from_index, nil)
|
||||||
return 0
|
return 0
|
||||||
elseif not inv:contains_item(to_list, {name = stack:get_name()}) then
|
|
||||||
return 1
|
|
||||||
else
|
else
|
||||||
return 0
|
return stack:get_count()
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -240,17 +229,24 @@ local function tubelib2_on_update2(pos, outdir, tlib2, node)
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
local function push_item(pos, filter, itemstack, num_items, nvm)
|
local function shuffle(list)
|
||||||
|
for i = #list, 2, -1 do
|
||||||
|
local j = math.random(i)
|
||||||
|
list[i], list[j] = list[j], list[i]
|
||||||
|
end
|
||||||
|
return list
|
||||||
|
end
|
||||||
|
|
||||||
|
local function push_item(pos, base_filter, itemstack, num_items, nvm)
|
||||||
|
local filter = shuffle(table.copy(base_filter))
|
||||||
local idx = 1
|
local idx = 1
|
||||||
local num_pushed = 0
|
local num_pushed = 0
|
||||||
local num_ports = #filter
|
local num_ports = #filter
|
||||||
num_ports = techage.in_range(num_ports, 1, 4)
|
|
||||||
local randidx = permIdx[num_ports][math.random(1, #permIdx[num_ports])]
|
|
||||||
local amount = math.floor(math.max((num_items + 1) / num_ports, 1))
|
local amount = math.floor(math.max((num_items + 1) / num_ports, 1))
|
||||||
local num_of_trials = 0
|
local num_of_trials = 0
|
||||||
while num_pushed < num_items and num_of_trials <= 8 do
|
while num_pushed < num_items and num_of_trials <= 8 do
|
||||||
num_of_trials = num_of_trials + 1
|
num_of_trials = num_of_trials + 1
|
||||||
local push_dir = filter[randidx[idx]]
|
local push_dir = filter[idx]
|
||||||
local num_to_push = math.min(amount, num_items - num_pushed)
|
local num_to_push = math.min(amount, num_items - num_pushed)
|
||||||
if techage.push_items(pos, push_dir, itemstack:peek_item(num_to_push)) then
|
if techage.push_items(pos, push_dir, itemstack:peek_item(num_to_push)) then
|
||||||
num_pushed = num_pushed + num_to_push
|
num_pushed = num_pushed + num_to_push
|
||||||
@ -282,11 +278,12 @@ local function distributing(pos, inv, crd, nvm)
|
|||||||
local num_items = stack:get_count()
|
local num_items = stack:get_count()
|
||||||
local num_to_push = math.min((nvm.num_items or crd.num_items) - sum_num_pushed, num_items)
|
local num_to_push = math.min((nvm.num_items or crd.num_items) - sum_num_pushed, num_items)
|
||||||
local stack_to_push = stack:peek_item(num_to_push)
|
local stack_to_push = stack:peek_item(num_to_push)
|
||||||
|
local filter = item_filter[item_name]
|
||||||
num_pushed = 0
|
num_pushed = 0
|
||||||
|
|
||||||
if item_filter[item_name] then
|
if filter and #filter > 0 then
|
||||||
-- Push items based on filter
|
-- Push items based on filter
|
||||||
num_pushed = push_item(pos, item_filter[item_name], stack_to_push, num_to_push, nvm)
|
num_pushed = push_item(pos, filter, stack_to_push, num_to_push, nvm)
|
||||||
elseif blocking_mode and #open_ports > 0 then
|
elseif blocking_mode and #open_ports > 0 then
|
||||||
-- Push items based on open ports
|
-- Push items based on open ports
|
||||||
num_pushed = push_item(pos, open_ports, stack_to_push, num_to_push, nvm)
|
num_pushed = push_item(pos, open_ports, stack_to_push, num_to_push, nvm)
|
||||||
|
@ -453,6 +453,10 @@ techage.manual_DE.aText = {
|
|||||||
"\n"..
|
"\n"..
|
||||||
"Der Verarbeitungsleistung eines TA2 Verteilers beträgt 4 Items alle 2 s\\, wobei der Verteiler dabei versucht\\, die 4 Items auf die offenen Ausgänge zu verteilen.\n"..
|
"Der Verarbeitungsleistung eines TA2 Verteilers beträgt 4 Items alle 2 s\\, wobei der Verteiler dabei versucht\\, die 4 Items auf die offenen Ausgänge zu verteilen.\n"..
|
||||||
"\n"..
|
"\n"..
|
||||||
|
"Wird dasselbe Item in einem Filter mehrfach hinterlegt\\, so beeinflusst dies das langfristige Verteilungsverhältnis entsprechend.\n"..
|
||||||
|
"\n"..
|
||||||
|
"Bitte beachte\\, dass die Verteilung ein probabilistischer Vorgang ist\\, d.h. die Verhältnisse werden nicht exakt\\, sondern nur langfristig eingehalten.\n"..
|
||||||
|
"\n"..
|
||||||
"\n"..
|
"\n"..
|
||||||
"\n",
|
"\n",
|
||||||
"Die Kieswaschanlage ist eine komplexere Maschine mit dem Ziel\\, Usmium Nuggets aus gesiebtem Kies auszuwaschen. Für den Aufbau wird ein TA2 Kiesspüler mit Achsenantrieb\\, ein Trichter\\, eine Kiste\\, sowie fließendes Wasser benötigt. \n"..
|
"Die Kieswaschanlage ist eine komplexere Maschine mit dem Ziel\\, Usmium Nuggets aus gesiebtem Kies auszuwaschen. Für den Aufbau wird ein TA2 Kiesspüler mit Achsenantrieb\\, ein Trichter\\, eine Kiste\\, sowie fließendes Wasser benötigt. \n"..
|
||||||
@ -1056,15 +1060,7 @@ techage.manual_DE.aText = {
|
|||||||
"\n"..
|
"\n"..
|
||||||
"\n"..
|
"\n"..
|
||||||
"\n",
|
"\n",
|
||||||
"Die Funktion des TA3 Verteilers entspricht der von TA2 mit einer weiteren Betriebart.\n"..
|
"Die Funktion des TA3 Verteilers entspricht der von TA2.\n"..
|
||||||
"\n"..
|
|
||||||
"*1:1 Bestückungsfunktion*\n"..
|
|
||||||
"\n"..
|
|
||||||
"Wird nur ein Ausgang aktiviert und mit mehreren Items konfiguriert\\, so kann die 1:1 Checkbox angeklickt werden. In diesem Falle werden nur Items gemäß der Filtereinstellung angenommen und in der Reihenfolge\\, wie die Items im Filter eingetragen sind\\, in definierte Positionen im Ziel-Inventar abgelegt. Damit kann weder das Inventar des Verteilers noch des Zielblocks volllaufen. Dies funktioniert für Autocrafter\\, Industrieofen und Elektronikfabrik.\n"..
|
|
||||||
"Mit dieser Betriebsart lassen sich andere Maschinen wie bspw. der Autocrafter exakt gemäß dem eingestellten Rezept bestücken. \n"..
|
|
||||||
"\n"..
|
|
||||||
"Dies funktioniert nur nur\\, wenn die Inventare des Verteilers und des Zielblocks zuvor frei sind.\n"..
|
|
||||||
"\n"..
|
|
||||||
"Die Verarbeitungsleistung beträgt 12 Items alle 4 s.\n"..
|
"Die Verarbeitungsleistung beträgt 12 Items alle 4 s.\n"..
|
||||||
"\n"..
|
"\n"..
|
||||||
"\n"..
|
"\n"..
|
||||||
|
@ -452,6 +452,10 @@ techage.manual_EN.aText = {
|
|||||||
"\n"..
|
"\n"..
|
||||||
"The processing power of a TA2 distributor is 4 items every 2 s\\, whereby the distributor tries to distribute the 4 items to the open outputs.\n"..
|
"The processing power of a TA2 distributor is 4 items every 2 s\\, whereby the distributor tries to distribute the 4 items to the open outputs.\n"..
|
||||||
"\n"..
|
"\n"..
|
||||||
|
"If the same item is configured multiple times in one filter\\, the long term distribution ratio will be influenced accordingly.\n"..
|
||||||
|
"\n"..
|
||||||
|
"Please note that the distribution is a probabilistic process. This means that the distribution rations won't be matched exactly\\, but only in the long term.\n"..
|
||||||
|
"\n"..
|
||||||
"\n"..
|
"\n"..
|
||||||
"\n",
|
"\n",
|
||||||
"The gravel washer is a more complex machine with the goal of washing Usmium nuggets out of sieved gravel. A TA2 rinser with axis drive\\, a hopper\\, a chest and running water are required for the installation.\n"..
|
"The gravel washer is a more complex machine with the goal of washing Usmium nuggets out of sieved gravel. A TA2 rinser with axis drive\\, a hopper\\, a chest and running water are required for the installation.\n"..
|
||||||
@ -1041,7 +1045,7 @@ techage.manual_EN.aText = {
|
|||||||
"\n"..
|
"\n"..
|
||||||
"\n"..
|
"\n"..
|
||||||
"\n",
|
"\n",
|
||||||
"The function of the TA3 distributor corresponds to that of TA2 with another operating mode.\n"..
|
"The function of the TA3 distributor corresponds to that of TA2.\n"..
|
||||||
"The processing power is 12 items every 4 s.\n"..
|
"The processing power is 12 items every 4 s.\n"..
|
||||||
"\n"..
|
"\n"..
|
||||||
"\n"..
|
"\n"..
|
||||||
|
@ -132,6 +132,10 @@ Einstellbar ist die Betriebsart über die "blockiere" Checkbox.
|
|||||||
|
|
||||||
Der Verarbeitungsleistung eines TA2 Verteilers beträgt 4 Items alle 2 s, wobei der Verteiler dabei versucht, die 4 Items auf die offenen Ausgänge zu verteilen.
|
Der Verarbeitungsleistung eines TA2 Verteilers beträgt 4 Items alle 2 s, wobei der Verteiler dabei versucht, die 4 Items auf die offenen Ausgänge zu verteilen.
|
||||||
|
|
||||||
|
Wird dasselbe Item in einem Filter mehrfach hinterlegt, so beeinflusst dies das langfristige Verteilungsverhältnis entsprechend.
|
||||||
|
|
||||||
|
Bitte beachte, dass die Verteilung ein probabilistischer Vorgang ist, d.h. die Verhältnisse werden nicht exakt, sondern nur langfristig eingehalten.
|
||||||
|
|
||||||
[ta2_distributor|image]
|
[ta2_distributor|image]
|
||||||
|
|
||||||
|
|
||||||
|
@ -133,6 +133,10 @@ The operating mode can be set using the "blocking mode" checkbox.
|
|||||||
|
|
||||||
The processing power of a TA2 distributor is 4 items every 2 s, whereby the distributor tries to distribute the 4 items to the open outputs.
|
The processing power of a TA2 distributor is 4 items every 2 s, whereby the distributor tries to distribute the 4 items to the open outputs.
|
||||||
|
|
||||||
|
If the same item is configured multiple times in one filter, the long term distribution ratio will be influenced accordingly.
|
||||||
|
|
||||||
|
Please note that the distribution is a probabilistic process. This means that the distribution rations won't be matched exactly, but only in the long term.
|
||||||
|
|
||||||
[ta2_distributor|image]
|
[ta2_distributor|image]
|
||||||
|
|
||||||
|
|
||||||
|
@ -726,15 +726,7 @@ Die Verarbeitungsleistung beträgt 6 Items alle 2 s.
|
|||||||
|
|
||||||
### TA3 Verteiler / Distributor
|
### TA3 Verteiler / Distributor
|
||||||
|
|
||||||
Die Funktion des TA3 Verteilers entspricht der von TA2 mit einer weiteren Betriebart.
|
Die Funktion des TA3 Verteilers entspricht der von TA2.
|
||||||
|
|
||||||
**1:1 Bestückungsfunktion**
|
|
||||||
|
|
||||||
Wird nur ein Ausgang aktiviert und mit mehreren Items konfiguriert, so kann die 1:1 Checkbox angeklickt werden. In diesem Falle werden nur Items gemäß der Filtereinstellung angenommen und in der Reihenfolge, wie die Items im Filter eingetragen sind, in definierte Positionen im Ziel-Inventar abgelegt. Damit kann weder das Inventar des Verteilers noch des Zielblocks volllaufen. Dies funktioniert für Autocrafter, Industrieofen und Elektronikfabrik.
|
|
||||||
Mit dieser Betriebsart lassen sich andere Maschinen wie bspw. der Autocrafter exakt gemäß dem eingestellten Rezept bestücken.
|
|
||||||
|
|
||||||
Dies funktioniert nur nur, wenn die Inventare des Verteilers und des Zielblocks zuvor frei sind.
|
|
||||||
|
|
||||||
Die Verarbeitungsleistung beträgt 12 Items alle 4 s.
|
Die Verarbeitungsleistung beträgt 12 Items alle 4 s.
|
||||||
|
|
||||||
[ta3_distributor|image]
|
[ta3_distributor|image]
|
||||||
|
@ -709,7 +709,7 @@ The processing power is 6 items every 2 s.
|
|||||||
|
|
||||||
### TA3 Distributor
|
### TA3 Distributor
|
||||||
|
|
||||||
The function of the TA3 distributor corresponds to that of TA2 with another operating mode.
|
The function of the TA3 distributor corresponds to that of TA2.
|
||||||
The processing power is 12 items every 4 s.
|
The processing power is 12 items every 4 s.
|
||||||
|
|
||||||
[ta3_distributor|image]
|
[ta3_distributor|image]
|
||||||
|
Loading…
Reference in New Issue
Block a user