Merge pull request #57 from Thomas--S/distributor-ratio

Allow to configure item ratios in distributor filters
This commit is contained in:
Joachim Stolberg 2021-05-09 19:29:04 +02:00 committed by GitHub
commit c920910d75
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
7 changed files with 81 additions and 48 deletions

View File

@ -26,6 +26,8 @@ local SRC_INV_SIZE = 8
local STANDBY_TICKS = 3 local STANDBY_TICKS = 3
local COUNTDOWN_TICKS = 4 local COUNTDOWN_TICKS = 4
local CYCLE_TIME = 4 local CYCLE_TIME = 4
local FILTER_ITEM_LIMIT_PER_STACK = 12
local FILTER_ITEM_LIMIT = 36
local INFO = [[Turn port on/off or read its state: command = 'port', payload = red/green/blue/yellow{=on/off}]] local INFO = [[Turn port on/off or read its state: command = 'port', payload = red/green/blue/yellow{=on/off}]]
@ -35,19 +37,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
@ -55,6 +44,8 @@ local function filter_settings(pos)
local filter = minetest.deserialize(meta:get_string("filter")) or {false,false,false,false} local filter = minetest.deserialize(meta:get_string("filter")) or {false,false,false,false}
local ItemFilter = {} -- {<item:name> = {dir,...}] local ItemFilter = {} -- {<item:name> = {dir,...}]
local OpenPorts = {} -- {dir, ...} local OpenPorts = {} -- {dir, ...}
local counter = 0 -- counts total item number in filter configuration
-- collect all filter settings -- collect all filter settings
for idx,slot in ipairs(SlotColors) do for idx,slot in ipairs(SlotColors) do
if filter[idx] == true then if filter[idx] == true then
@ -69,7 +60,13 @@ local function filter_settings(pos)
if not ItemFilter[name] then if not ItemFilter[name] then
ItemFilter[name] = {} ItemFilter[name] = {}
end end
table.insert(ItemFilter[name], out_dir) for _ = 1, math.min(FILTER_ITEM_LIMIT_PER_STACK, stack:get_count()) do
if counter > FILTER_ITEM_LIMIT then
break
end
table.insert(ItemFilter[name], out_dir)
counter = counter + 1
end
end end
end end
end end
@ -171,6 +168,20 @@ local function formspec(self, pos, nvm)
end end
end end
local function get_total_filter_items_number(pos, except_listname, except_index)
local inv = M(pos):get_inventory()
local total = 0
for _, listname in ipairs(SlotColors) do
local list = inv:get_list(listname)
for idx, stack in ipairs(list) do
if not (listname == except_listname and idx == except_index) then
total = total + stack:get_count()
end
end
end
return total
end
local function allow_metadata_inventory_put(pos, listname, index, stack, player) local function allow_metadata_inventory_put(pos, listname, index, stack, player)
local inv = M(pos):get_inventory() local inv = M(pos):get_inventory()
local list = inv:get_list(listname) local list = inv:get_list(listname)
@ -181,8 +192,10 @@ 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])
local max_items_to_limit = FILTER_ITEM_LIMIT - get_total_filter_items_number(pos, listname, index)
stack:set_count(math.min(FILTER_ITEM_LIMIT_PER_STACK, stack:get_count(), max_items_to_limit))
inv:set_stack(listname, index, stack) inv:set_stack(listname, index, stack)
return 0 return 0
end end
@ -197,7 +210,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 +224,19 @@ 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(inv:get_stack(to_list, to_index))
local max_items_to_limit = FILTER_ITEM_LIMIT - get_total_filter_items_number(pos, to_list, to_index)
stack:set_count(math.min(FILTER_ITEM_LIMIT_PER_STACK, stack:get_count(), max_items_to_limit))
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 elseif from_list ~= "src" and to_list ~= "src" then
return 1 return math.min(stack:get_count(), FILTER_ITEM_LIMIT_PER_STACK - inv:get_stack(to_list, to_index):get_count())
else else
return 0 return stack:get_count()
end end
end end
@ -240,17 +257,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 +306,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)

View File

@ -453,6 +453,12 @@ 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"..
"In den Filtern beträgt die maximale Stackgröße 12\\; insgesamt können höchstens 36 Items konfiguriert werden.\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 +1062,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"..

View File

@ -452,6 +452,12 @@ 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"..
"The maximum stack size in the filters is 12\\; in total\\, not more than 36 items can be configured.\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 +1047,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"..

View File

@ -132,6 +132,12 @@ 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.
In den Filtern beträgt die maximale Stackgröße 12; insgesamt können höchstens 36 Items konfiguriert werden.
[ta2_distributor|image] [ta2_distributor|image]

View File

@ -133,6 +133,12 @@ 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.
The maximum stack size in the filters is 12; in total, not more than 36 items can be configured.
[ta2_distributor|image] [ta2_distributor|image]

View File

@ -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]

View File

@ -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]