Inksaver

TurtleUtils (20241208.1000)

Dec 8th, 2024
34
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
Lua 103.64 KB | Source Code | 0 0
  1. local version       = 20241208.1000
  2. local Turtle        = require("lib.clsTurtle")
  3. local T             = Turtle(false)
  4. local Items         = require("lib.data.items")
  5. local Vector2       = require("lib.Vector2")
  6. local metals        = {"iron", "gold", "copper", "netherite", "red_alloy"}
  7. local categories    = {"building", "colored", "combat", "computercraft", "food", "functional", "ingredients", "morered", "natural", "redstone", "tools"}
  8. local timber        = {"dark_oak", "oak", "birch", "spruce", "jungle", "acacia", "mangrove", "cherry", "bamboo", "crimson", "warped"}
  9. local stone         = {"cobblestone", "red_sandstone", "sandstone","end_stone", "blackstone", "stone", "deepslate","granite", "diorite", "andesite", "netherrack", "nether", "basalt", "mud", "dark_prismarine", "prismarine"}
  10. local minerals      = {"iron", "gold", "copper", "netherite", "diamond", "emerald", "coal", "redstone", "lapis_lazuli", "quartz", "amethyst"}
  11. local ingots        = {"iron", "gold", "copper", "netherite"}
  12. local itemColours   = {"white", "light_gray", "gray", "brown", "red", "orange", "yellow", "lime", "green", "cyan", "light_blue", "blue", "purple", "magenta", "pink"}
  13. local currentCursor = { x = 0, y = 0, fg = colors.white, bg = colors.black, blink = true}
  14. local nuggetItems   = {"minecraft:iron_axe", "minecraft:iron_chestplate", "minecraft:iron_helmet", "minecraft:iron_hoe", "minecraft:iron_pickaxe",
  15.                      "minecraft:iron_boots", "minecraft:iron_leggings", "minecraft:iron_shovel","minecraft:iron_sword"}
  16. local fuelValues    = {["lava_bucket"] = 100, ["coal_block"] = 80, ["dried_kelp_block"] = 20, ["blaze_rod"] = 12, ["coal"] = 8, ["log"] = 1.5, ["wood"] = 1.5, ["planks"] = 1.5}
  17. local nbt = {["021f1ac06ec4e4c75d0e0bf67c0712dc"] = "Silk Touch", ["704a1bcdf9953c791651a77b1fe78891"] = "Mending"}
  18.  
  19.  
  20. local U = {}
  21. U.fc = 0                        -- function count for debugging purposes
  22. U.bedrock = 0
  23. U.ceiling = 255
  24. U.multiButtonData = nil
  25. --U.R = nil                     -- assigned and populated from tk2.lua main()
  26. U.turtleName = ""
  27. U.isStorageConfigured = false
  28. U.barrelObjects = nil           -- list of barrel objects returned by {periheral.find("minecraft:barrel")} in U.wrapModem(R)
  29. U.chestObjects = nil            -- list of chest objects returned by {periheral.find("minecraft:chest")} in U.wrapModem(R)
  30. U.barrelNames = nil             -- list of barrel Names returned by {periheral.getName(barrelObject[#])}
  31. U.chestNames = nil              -- list of chest Names returned by {periheral.getName(chestObject[#])}
  32. U.barrelItems = nil             -- list of items and the barrels where they are usually found
  33. U.chestItems = nil              -- list of items and the chests where they are usually found
  34. U.itemDatabase = nil            -- database of all Items, their name, displayName and crafting recipe
  35. U.crafts = nil                  -- table of displayNames of all items that can be crafted. populated in scene:Craft.S.enter()from U.fillCrafts()
  36. U.smelts = nil                  -- table of displayNames of all items that can be smelted. populated in scene:Smelt.S.enter()from U.fillSmelts()
  37. U.smeltFrom = nil               -- table of items and their smelt sources eg {["minecraft:stone"] = {"minecraft:cobblestone"}}
  38. U.crafter = nil                 -- placeholder for single Crafter object
  39. U.smelters = {}                 -- populated in main().setupSmelters()
  40. U.allItems = {}                 -- populated in main()
  41. U.data = nil
  42. U.furnaceCount = 0
  43. U.blastCount = 0
  44. U.smokerCount = 0
  45. U.blastFurnace = {"ancient_debris","iron","gold","copper"}
  46. U.smoker = {"mutton", "rabbit", "chicken", "potato", "cod", "salmon", "beef", "porkchop", "kelp"}
  47. U.window = nil
  48. U.dialog = nil
  49. U.smeltersActive = false
  50. U.smeltFromCraft = false
  51. U.windowAction = ""
  52. U.connected = false
  53. U.dialogActive = false
  54. U.dialogData = nil
  55. U.stack1 = {"bed", "shulker", "decorated_pot", "minecart", "boat", "shovel", "pickaxe", "axe", "hoe", "fishing_rod", "flint_n_steel", "*bucket", "spyglass", "book_n_quill",
  56.             "elytra", "shears", "horn", "music", "soup", "stew", "cake", "water_bottle", "potion", "enchanted_book", "helmet", "chestplate", "leggings", "boots",
  57.             "bow", "crossbow", "armor", "cap", "tunic", "pants"}
  58. U.stack16 = {"ender_pearl", "snowball", "bucket", "egg", "sign", "honey_bottle", "banner", "armor_stand", "bucket"}
  59. --[[
  60.     ccTweaked events
  61.     char            local event, character = os.pullEvent("char")
  62.    
  63.     key             local event, key, is_held = os.pullEvent("key")
  64.     key_up          local event, key = os.pullEvent("key_up")
  65.    
  66.     mouse_click     local event, button, x, y = os.pullEvent("mouse_click")
  67.     mouse_drag      local event, button, x, y = os.pullEvent("mouse_drag")
  68.     mouse_scroll    local event, direction, x, y = os.pullEvent("mouse_scroll")  direction -1 = up, 1 = down
  69.     mouse_up        local event, button, x, y = os.pullEvent("mouse_up")
  70.     monitor_touch   local event, side, x, y = os.pullEvent("monitor_touch")
  71. ]]
  72.  
  73. local colours = {}
  74. table.insert(colours, 1,"white")
  75. table.insert(colours, 2, "orange")
  76. table.insert(colours, 4, "magenta")
  77. table.insert(colours, 8, "lightBlue")
  78. table.insert(colours, 16, "yellow")
  79. table.insert(colours, 32, "lime")
  80. table.insert(colours, 64, "pink")
  81. table.insert(colours, 128, "gray")
  82. table.insert(colours, 256, "lightGray")
  83. table.insert(colours, 512, "cyan")
  84. table.insert(colours, 1024, "purple")
  85. table.insert(colours, 2048, "blue")
  86. table.insert(colours, 4096, "brown")
  87. table.insert(colours, 8192, "green")
  88. table.insert(colours, 16384, "red")
  89. table.insert(colours, 32768, "black")
  90.  
  91. -- GENERAL UTILITIES
  92. function string:count(c)
  93.     --[[ count number of occurences of c ]]
  94.     local _,n = self:gsub(c,"")
  95.     return n
  96. end
  97.  
  98. function string:endsWith(ending)
  99.     --[[ get ending character of a string ]]
  100.     return ending == "" or self:sub(-#ending) == ending
  101. end
  102.  
  103. function string:split(sSeparator, nMax, bRegexp, noEmpty)
  104.     --[[return a table split with sSeparator. noEmpty removes empty elements
  105.         use: tblSplit = SplitTest:split('~',[nil], [nil], false) or tblSplit = string.split(SplitTest, '~')]]  
  106.     assert(sSeparator ~= '','separator must not be empty string')
  107.     assert(nMax == nil or nMax >= 1, 'nMax must be >= 1 and not nil')
  108.     if noEmpty == nil then noEmpty = true end
  109.  
  110.     local aRecord = {}
  111.     local newRecord = {}
  112.     -- self refers to the 'string' being split
  113.     if self:len() > 0 then
  114.         local bPlain = not bRegexp
  115.         nMax = nMax or -1
  116.  
  117.         local nField, nStart = 1, 1
  118.         local nFirst,nLast = self:find(sSeparator, nStart, bPlain)
  119.         while nFirst and nMax ~= 0 do
  120.             aRecord[nField] = self:sub(nStart, nFirst-1)
  121.             nField = nField+1
  122.             nStart = nLast+1
  123.             nFirst,nLast = self:find(sSeparator, nStart, bPlain)
  124.             nMax = nMax-1
  125.         end
  126.         aRecord[nField] = self:sub(nStart)
  127.        
  128.         if noEmpty then --split on newline preserves empty values
  129.             for i = 1, #aRecord do
  130.                 if aRecord[i] ~= "" then
  131.                     table.insert(newRecord, aRecord[i])
  132.                 end
  133.             end
  134.         else
  135.             newRecord = aRecord
  136.         end
  137.     end
  138.    
  139.     return newRecord
  140. end
  141.  
  142. function U.charCount(text, char)
  143.     return text:count(char)
  144. end
  145.  
  146. function U.clear(reset)
  147.     reset = reset or false
  148.    
  149.     if reset then
  150.         term.setTextColor(colors.white)
  151.         term.setBackgroundColor(colors.black)
  152.     end
  153.     term.clear()
  154.     term.setCursorPos(1, 1)
  155. end
  156.  
  157. function U.colourText(col, row, text)
  158.     --[[
  159.         This uses char ` to separate colour strings and ¬ to separate text/background colours              
  160.         example text = `lg¬black` `lg¬purple` `lg¬black`Furnace `lg¬red` `lg¬black`Blast Furnace `lg¬green` `lg¬black`Smoker
  161.     ]]
  162.  
  163.     term.setCursorPos(col, row)
  164.     local lineParts = text:split("`")
  165.     for i = 1, #lineParts do
  166.         part = lineParts[i]                     -- eg "red;black" or "This is a line of red text on black background"
  167.         if part:find("¬") ~= nil then
  168.             local fgbg = part:split("¬")
  169.             if fgbg[1] == "lg" then
  170.                 fgbg[1] = "lightGray"
  171.             end
  172.             if fgbg[2] == "lg" then
  173.                 fgbg[2] = "lightGray"
  174.             end
  175.             term.setTextColor(colors[fgbg[1]])
  176.             term.setBackgroundColor(colors[fgbg[2]])
  177.         else                                    -- not a colour command so print it out without newline
  178.             term.write(part)
  179.         end
  180.     end
  181. end
  182.  
  183. function U.contains(list, item)
  184.     if list ~= nil then
  185.         for k,v in pairs(list) do
  186.             if v == item then
  187.                 return true
  188.             end
  189.         end
  190.     end
  191.     return false
  192. end
  193.  
  194. function U.copyTable(tbl)
  195.     local temp = {}
  196.     for _, v in ipairs(tbl) do
  197.         table.insert(temp, v)
  198.     end
  199.     return temp
  200. end
  201.  
  202. function U.getColour(number)
  203.     return colours[number]
  204. end
  205.  
  206. function U.indexOf(list, item)
  207.     for i,v in ipairs(list) do
  208.         if v == item then
  209.             return i
  210.         end
  211.     end
  212.     return -1
  213. end
  214.  
  215. function U.isIngot(item)
  216.     for _,v in ipairs(ingots) do
  217.         if item:find(v) ~= nil then
  218.             return true, v
  219.         end
  220.     end
  221.     return false, ""
  222. end
  223.  
  224. function U.isMineral(item)
  225.     for _,v in ipairs(minerals) do
  226.         if item:find(v) ~= nil then
  227.             return true, v
  228.         end
  229.     end
  230.     return false, ""
  231. end
  232.  
  233. function U.isStone(item)
  234.     for _,v in ipairs(stone) do
  235.         if item:find(v) ~= nil then
  236.             return true, v
  237.         end
  238.     end
  239.     return false, ""
  240. end
  241.  
  242. function U.isTableEmpty(aTable)
  243.     if next(aTable) == nil then
  244.         return true
  245.     end
  246.     return false
  247. end
  248.  
  249. function U.isTimber(item)
  250.     for _,v in ipairs(timber) do
  251.         if item:find(v) ~= nil then
  252.             return true, v
  253.         end
  254.     end
  255.     return false, ""
  256. end
  257.  
  258. function U.padCenter(text, length, char)
  259.     return U.padCentre(text, length, char)
  260. end
  261.  
  262. function U.padCentre(text, length, char)
  263.     --[[Pads str to length with char on left and right]]
  264.     text = tostring(text)
  265.     char = char or " "
  266.     while #text < length do
  267.         text = char..text..char
  268.     end
  269.     if #text > length then
  270.         text = text:sub(1, length)
  271.     end
  272.    
  273.     return text
  274. end
  275.  
  276. function U.padLeft(text, length, char)
  277.     --[[Pads str to length with char from left]]
  278.     text = tostring(text)
  279.     char = char or " "
  280.     while #text < length do
  281.         text = char..text
  282.     end
  283.     if #text > length then
  284.         text = text:sub(1, length)
  285.     end
  286.    
  287.     return text
  288. end
  289.  
  290. function U.padRight(text, length, char)
  291.     --[[
  292.     Pads string to length len with chars from right
  293.     test = lib.padRight("test", 10, "+") -> "test++++++"]]
  294.     text = tostring(text)
  295.     char = char or " "
  296.     while #text < length do
  297.         text = text.. char
  298.     end
  299.     if #text > length then
  300.         text = text:sub(1, length)
  301.     end
  302.    
  303.     return text
  304. end
  305.  
  306. function U.restoreCursor(withColours)
  307.     withColours = withColours or false
  308.     term.setCursorPos(currentCursor.x, currentCursor.y )
  309.     term.setCursorBlink(currentCursor.blink )
  310.     if withColours then
  311.         term.setTextColor(currentCursor.fg)
  312.         term.setBackgroundColor(currentCursor.bg )
  313.     end
  314. end
  315.  
  316. function U.removeDigits(text)
  317.     -- eg " 1 64 cobbled_deepslate     " slot and count
  318.     -- eg "62 cobblestone              " count only, no slot
  319.     local slot = 0
  320.     local count = 0
  321.     text = U.trim(text)                         -- remove leading and trailing spaces
  322.     if tonumber(text:sub(1,1)) ~= nil then      -- starts with a number
  323.         local start = text:find(" ")            -- eg start = 3
  324.         slot = text:sub(1, start - 1)           -- eg slot = 1
  325.         text = U.trim(text:sub(start + 1))      -- eg "64 cobbled_deepslate" or "cobblestone"
  326.         if tonumber(text:sub(1,1)) ~= nil then  -- starts with a number
  327.             start = text:find(" ")              -- eg start = 3
  328.             count = text:sub(1, start - 1)      -- eg count = 64
  329.             text = text:sub(start + 1)          -- eg "cobbled_deepslate"
  330.         else
  331.             count = slot                        -- original number was count, not slot
  332.             slot = 0                            -- reset slot
  333.         end
  334.     end
  335.    
  336.     return text, tonumber(count), tonumber(slot)                    -- eg cobbled_deepslate, 64, 1 or cobblestone, 62, 0
  337. end
  338.  
  339. function U.setCurrentCursor(withColours)
  340.     withColours = withColours or false
  341.     currentCursor.x, currentCursor.y = term.getCursorPos()
  342.     currentCursor.blink = term.getCursorBlink()
  343.     if withColours then
  344.         currentCursor.fg = term.getTextColor()
  345.         currentCursor.bg = term.getBackgroundColor()
  346.     end
  347. end
  348.  
  349. function U.split(text, separator)
  350.     separator = separator or " "
  351.     return text:split(separator)
  352. end
  353.  
  354. function U.stackSize(item)
  355.     for _, v in ipairs(U.stack1) do
  356.         if item:find(v) ~= nil then
  357.             return 1
  358.         end
  359.     end
  360.     for _, v in ipairs(U.stack16) do
  361.         if item:find(v) ~= nil then
  362.             return 16
  363.         end
  364.     end
  365.     return 64
  366. end
  367.  
  368. function U.tableConcat(tbl, sep)
  369.     local output = ""
  370.     for i,value in pairs(tbl) do
  371.         output = output .. tostring(value)
  372.         if i ~= #tbl then
  373.             output = output .. sep
  374.         end
  375.     end
  376.  
  377.     return output
  378. end
  379.  
  380. function U.tableContains(tableName, value, exactMatch)
  381.     exactMatch = exactMatch or false
  382.     for k, v in ipairs(tableName) do
  383.         if exactMatch then
  384.             if v == value then
  385.                 return true
  386.             end
  387.         else
  388.             if v:find(value) ~= nil then
  389.                 return true
  390.             end
  391.         end
  392.     end
  393.     return false
  394. end
  395.  
  396. function U.trim(text)
  397.     --[[ trim leading and trailing spaces ]]
  398.     if text ~= nil then
  399.         return (text:gsub("^%s*(.-)%s*$", "%1"))
  400.     end
  401.     return text
  402. end
  403.  
  404. function U.unpack(tbl)
  405.     local output = "{"
  406.     for key, value in pairs(tbl) do
  407.         if type(value) == "table" then
  408.             for k,v in pairs(value) do
  409.                 output = output.."k = "..k..", v = "..v.."; "
  410.             end
  411.         else
  412.             output = "key = "..key..", value = "..tostring(value)
  413.         end
  414.     end
  415.     return output.."}"
  416. end
  417.  
  418. function U.writeTraceTable(description, tbl)
  419.     local output = {}
  420.     for key, value in pairs(tbl) do
  421.         local text = ""
  422.         if type(value) == "table" then
  423.             for k,v in pairs(value) do
  424.                 text = text.."k = "..k..", v = "..v.."; "
  425.             end
  426.         else
  427.             text = "key = "..key..", value = "..value
  428.         end
  429.         table.insert(output, text)
  430.     end
  431.     _G.Log:writeTraceTable(description, output)
  432. end
  433. --TURTLE UTILITIES
  434. function U.useSticksAsFuel()
  435.     local slot = T:getItemSlot("minecraft:stick")
  436.     if slot > 0 then -- use any sticks to refuel
  437.         turtle.select(slot)
  438.         turtle.refuel()
  439.     end
  440.     slot = T:getItemSlot("minecraft:mangrove_roots")
  441.     if slot > 0 then -- use any roots to refuel
  442.         turtle.select(slot)
  443.         turtle.refuel()
  444.     end
  445. end
  446.  
  447. -- NETWORK UTILITIES
  448. function U.addToStorageList(storageType, itemKey, storageName, writeToFile)
  449.     -- itemKey is a table, so is passed byRef. No need to return a value
  450.     -- eg itemKey: [ "minecraft:dark_oak_sapling" ] = {"minecraft:barrel_94", "minecraft:barrel_96"}
  451.     -- storageName = "minecraft:barrel_99"
  452.     local itemTable = {}
  453.     if storageType == "chest" then
  454.         itemTable = U.chestItems[itemKey]   -- eg [ "minecraft:dark_oak_sapling" ] = {"minecraft:barrel_94", "minecraft:barrel_96"}
  455.     else
  456.         itemTable = U.barrelItems[itemKey]
  457.     end
  458.     if itemTable == nil then                -- key does not match. This item not previously stored
  459.         if storageType == "chest" then
  460.             U.chestItems[itemKey] = {storageName}   -- eg U.chestItems[minecraft:diorite] = {chest_105}
  461.         else
  462.             U.barrelItems[itemKey] = {storageName}
  463.         end
  464.     else
  465.         for _, storage in ipairs(itemTable) do  -- is "minecraft:barrel_99" already in the list?
  466.             if storage == storageName then
  467.                 return  -- exit function
  468.             end
  469.         end
  470.         -- not found so add to table. return not required as funcion is ended
  471.         table.insert(itemTable, storageName)    -- add to table eg table[ "minecraft:dark_oak_sapling" ] = {"minecraft:barrel_94", "minecraft:barrel_96",, "minecraft:barrel_99"}
  472.     end
  473.     if writeToFile then
  474.         U.updateList(storageType)
  475.     end
  476. end
  477.  
  478. function U.attachModem()
  479.     -- modem cannot be "attached". Has to be player right-click!
  480.     -- place on  top or next to a modem and ask player to right-click
  481.     T:clear()
  482.     menu.colourPrint("Please right-click on the modem(s) I am next to or above/below"..
  483.                     "\nThe centre square should be lit red.\n"..
  484.                     "If embedded use narrow gap at side\n", colors.red)
  485.     local event, side = os.pullEvent("peripheral")
  486.     for timer = 5, 0, -1 do
  487.         -- text, fg, bg, width, isInput, cr
  488.         menu.colourWrite("Thanks. continuing in ".. timer.." seconds", colors.lime, colors.black, 0, false, true)
  489.         sleep(1)
  490.     end
  491. end
  492.  
  493. function U.checkInventory(inventory, itemName, itemsPerSlot, matchPart)
  494.     --[[
  495.     Find an item already in an inventory
  496.     inventory = The wrapped inventory or it's name
  497.     itemName = The name of the item to find.
  498.     return no of items already present, and storage space for additional
  499.     ]]
  500.     itemsPerSlot = itemsPerSlot or 64
  501.     matchPart = matchPart or ""
  502.     local contents = nil
  503.     local numSlots = 0
  504.     if type(inventory) == "string" then
  505.         contents = peripheral.call(inventory, "list")
  506.         numSlots = peripheral.call(inventory, "size")
  507.         --assert(contents ~= nil, "Nil contents from inventory "..tostring(inventory))
  508.         _G.Log:saveToLog("U.checkInventory('"..inventory.."', itemName = "..itemName..", itemsPerSlot = "..itemsPerSlot..", matchPart = "..tostring(matchPart))
  509.     else
  510.         contents = inventory.list()
  511.         numSlots = inventory.size()
  512.     end
  513.     local inStock = 0
  514.     local partMatch = false
  515.     local canStore = 0
  516.     if contents ~= nil then
  517.         _G.Log:saveToLog("#slots in use = "..#contents)
  518.         canStore = (numSlots - #contents) * itemsPerSlot    -- capacity of empty slots
  519.         for slot, item in pairs(contents) do
  520.             if item.name == itemName then
  521.                 inStock = inStock  + item.count
  522.                 canStore = canStore + itemsPerSlot - item.count
  523.             else
  524.                 if matchPart ~= "" then -- eg check for "cobblestone" or "slab"
  525.                     if item.name:find(matchPart) ~= nil then
  526.                         partMatch = true
  527.                     end
  528.                 end
  529.             end
  530.         end
  531.     end
  532.     return inStock, canStore, partMatch -- eg 1, 3647, false if contains only 1 matching item in otherwise empty chest
  533. end
  534.  
  535. function U.emptyInventory(barrels, chests, sticksAsFuel)
  536.     --[[U.emptyInventory({"sapling", "propagule", "dirt", "crafting"}, {"all"}, true)]]
  537.     if not T:isEmpty() then
  538.         if sticksAsFuel then
  539.             U.useSticksAsFuel()
  540.         end
  541.         for _, item in ipairs(barrels) do
  542.             U.sendItemToNetworkStorage("barrel", item, 0)
  543.         end
  544.         for _, item in ipairs(chests) do
  545.             U.sendItemToNetworkStorage("chest", item, 0)
  546.         end
  547.     end
  548. end
  549.  
  550. function U.loadStorageLists()
  551.     local lib = {}
  552.    
  553.     function lib.createList(storageType)
  554.         local total = 0
  555.         local locations = {}
  556.         local storageNames = U.barrelNames
  557.         if storageType == "chest" then
  558.             storageNames = U.chestNames
  559.         end
  560.         for _, store in pairs(storageNames) do                  -- eg "minecraft:chest_1"
  561.             T:clear()              
  562.             _G.Log:saveToLog("Checking "..store.. " contents")      -- eg "checking minecraft:chest_1 contents"
  563.             local contents = peripheral.call(store, "list")     -- list of items / slots for this chest
  564.             for slot, item in pairs(contents) do                -- for each item check if this storage is listed
  565.                 if locations[item.name] == nil then             -- this item not yet found
  566.                     locations[item.name] = {store}              -- add to table eg locations["minecraft:cobblestone"] = {"minecraft:chest_1"}
  567.                 else                                            -- already has at least 1 location
  568.                     U.addToStorageList(storageType, locations[item.name], store, false)
  569.                 end
  570.             end
  571.             total = total + 1
  572.         end
  573.         _G.Log:saveToLog("found ".. total.." "..storageType)
  574.         local output = textutils.serialize(locations)       -- serialise to json ready to write to file
  575.         local fileName = "lib/data/"..storageType.."Items.lua"          -- barrelList.lua or chestList.lua
  576.         local outputHandle = fs.open(fileName, "w")         -- open file
  577.         outputHandle.writeLine("return")                    -- start file with return
  578.         outputHandle.write(output)                          -- add serialised table
  579.         outputHandle.close()                                -- close file
  580.        
  581.         return locations
  582.     end
  583.    
  584.     function lib.listNeedUpdating(storageType, list)
  585.         -- see if named chest/barrel in list is found in fresh peripheral.find
  586.         -- turtle may have moved to a different network
  587.         -- list = eg [ "minecraft:stick" ] = {"minecraft:barrel_91","minecraft:barrel_114"}
  588.         local rawStorage = nil                      -- peripheral find can return duplicate values
  589.         local using = "minecraft:barrel"            -- storageType may be barrel, barrels, minecraft:barrel
  590.         local found = false
  591.         if storageType:find("chest") ~= nil then
  592.             using = "minecraft:chest"
  593.         end
  594.         rawStorage = {peripheral.find(using)}
  595.        
  596.         if not U.isTableEmpty(rawStorage) then      -- chests / barrels are attached see if they match
  597.             for item, storeList in pairs(list) do       -- existing storage names can be found here
  598.                 for key, value in ipairs(rawStorage) do -- look in the fresh list of storage names to see if there are missing entries
  599.                     local name = peripheral.getName(value)             
  600.                     for _, storageName in ipairs(storeList) do  -- check each storage name found
  601.                         if storageName == name then     -- recorded name matches, check next one
  602.                             found = true
  603.                             break
  604.                         end
  605.                     end
  606.                     if found then break end
  607.                 end
  608.                 if not found then
  609.                     -- no match in existing list for this storage: list needs updating
  610.                     return true-- list does not match
  611.                 end
  612.             end
  613.         end
  614.         return false    -- list is ok
  615.     end
  616.    
  617.     -- populate these module scope variables:
  618.     local message = U.wrapModem(R)
  619.     if message == "Modem not found" then return message end
  620.     local redo = false
  621.     if U.barrelItems == nil then    -- module scope variable not yet loaded
  622.         _G.Log:saveToLog("U.barrelItems == nil")
  623.         if fs.exists("lib/data/barrelItems.lua") then
  624.             _G.Log:saveToLog("U.barrelItems = require('lib.data.barrelItems)")
  625.             U.barrelItems = require("lib.data.barrelItems") -- module scope variable
  626.             if U.barrelItems == nil then
  627.                 redo = true
  628.             else
  629.                 _G.Log:saveToLog("U.barrelItems ~= nil")
  630.                 redo = lib.listNeedUpdating("barrel", U.barrelItems)
  631.             end
  632.         else
  633.             _G.Log:saveToLog("U.barrelItems = lib.createList('barrel'")
  634.             U.barrelItems = lib.createList("barrel")
  635.         end
  636.     end
  637.     if U.chestItems == nil then -- module scope variable not yet loaded
  638.         _G.Log:saveToLog("U.chestItems == nil")
  639.         if fs.exists("lib/data/chestItems.lua") then
  640.             _G.Log:saveToLog("U.chestItems = require('lib.data.chestItems')")
  641.             U.chestItems = require("lib.data.chestItems")   -- module scope variable
  642.             if U.chestItems == nil then
  643.                 redo = true
  644.             else
  645.                 _G.Log:saveToLog("U.chestItems ~= nil")
  646.                 redo = lib.listNeedUpdating("chest", U.chestItems)
  647.             end
  648.         else
  649.             _G.Log:saveToLog("U.chestItems = lib.createList('chest'")
  650.             U.chestItems = lib.createList("chest")
  651.         end
  652.     end
  653.     if redo then
  654.         U.barrelItems = lib.createList("barrel")
  655.         U.chestItems = lib.createList("chest")
  656.     end
  657.     if fs.exists("lib/data/items.lua") then
  658.         _G.Log:saveToLog("loading items database")
  659.         U.itemDatabase = require ("lib/data/items")
  660.     end
  661.     U.isStorageConfigured = true
  662.     return ""
  663. end
  664.  
  665. function U.findEmptySlot(list, size)
  666.     --[[ adapted from https://github.com/cc-tweaked/CC-Tweaked/discussions/1552
  667.     Find the first empty slot in a chest.
  668.     list = list of items in the chest/barrel/dropper
  669.     size = The size of the inventory
  670.     return integer? slot The slot number of the first empty slot, or nil if none are empty.
  671.     ]]
  672.     for slot = 1, size do
  673.         if not list[slot] then
  674.             return slot
  675.         end
  676.     end
  677.    
  678.     return nil
  679. end
  680.  
  681. function U.findItemCountInInventory(list, itemName, exactMatch)
  682.     --[[
  683.     Find an item in an inventory
  684.     list = The list of items in the inventory
  685.     itemName = The name of the item to find.
  686.     return integer? The slot number of the item, or nil if not found.
  687.     ]]
  688.     exactMatch = exactMatch or false
  689.     if type(list) == "string" then
  690.         list = peripheral.call(list, "list")
  691.     end
  692.     local retValue = nil
  693.     local count = 0
  694.     for slot, item in pairs(list) do
  695.         local found = false
  696.         if exactMatch then
  697.             if item.name == itemName then found = true end
  698.         else
  699.             if item.name:find(itemName) ~= nil then found = true end
  700.         end
  701.         if found then
  702.             if retValue == nil then
  703.                 retValue = {}
  704.             end
  705.             table.insert(retValue,{slot, item.count}) -- eg {1, 64}
  706.             count = count + item.count
  707.         end
  708.     end
  709.     return count, retValue -- either nil or eg {{1, 64},{4, 22}}
  710. end
  711.  
  712. function U.findItemInInventory(inventory, itemName, exactMatch)
  713.     --[[ adapted from https://github.com/cc-tweaked/CC-Tweaked/discussions/1552
  714.     Find an item in an inventory
  715.     inventory = name of inventory or wrapped peripheral
  716.     itemName = The name of the item to find.
  717.     return integer?, integer? The slot number and count of the item, or nil if not found.
  718.     ]]
  719.     exactMatch = exactMatch or false
  720.     if type(inventory) == "string" then
  721.         _G.Log:saveToLog("U.findItemInInventory("..inventory..", "..itemName..", exactMatch = "..tostring(exactMatch)..")", true)
  722.         contents = peripheral.call(inventory, "list")
  723.     else    -- should be supplied with .list() already
  724.         _G.Log:saveToLog("U.findItemInInventory(<inventory>, "..itemName..", exactMatch = "..tostring(exactMatch)..")", true)
  725.         contents = inventory
  726.     end
  727.     --_G.Log:saveToLog("contents = "..table.concat(contents, ", "))
  728.     --utils.writeTraceTable("contents = ", contents)
  729.     if contents ~= nil then
  730.         for slot, item in pairs(contents) do
  731.             --_G.Log:saveToLog("item.name = "..item.name..", item.count = "..item.count)
  732.             if exactMatch then
  733.                 if item.name == itemName then
  734.                     _G.Log:saveToLog("Item found in "..slot..", quantity = "..item.count)
  735.                     return slot, item.count
  736.                 end
  737.             else
  738.                 if (item.name):find(itemName) ~= nil then
  739.                     _G.Log:saveToLog("Matching Item found in "..slot..", quantity = "..item.count)
  740.                     return slot, item.count
  741.                 end
  742.             end
  743.         end
  744.     end
  745.     _G.Log:saveToLog("Item not found")
  746.     return 0,0
  747. end
  748.  
  749. function U.getItemFromNetwork(storageType, itemRequired, countRequired, toTurtleSlot, ignoreStock)
  750.     local lib = {}
  751.    
  752.     function lib.getItem(storageTable, itemRequired, countRequired, toTurtleSlot, sent)
  753.         local exit = false
  754.         for k, storageName in pairs(storageTable) do    -- eg {"minecraft:barrel_17", "minecraft:barrel_18"...}
  755.             local available, data = U.findItemCountInInventory(storageName, itemRequired, false)        -- either nil or eg {{1, 64},{4, 22}}
  756.             _G.Log:saveToLog("U.findItemCountInInventory("..storageName..", "..itemRequired..", false")
  757.             if data ~= nil then
  758.                 for i = 1, #data do
  759.                     local request = countRequired
  760.                     if countRequired > 64 then
  761.                         request = 64
  762.                     end
  763.                     local received = U.sendItemsToTurtle(storageName, data[i][1], request, toTurtleSlot)    -- request items, returns number sent
  764.                     if received == nil then received = 0 end
  765.                     sent = sent + received
  766.                     _G.Log:saveToLog("received = "..received..", request = "..request.." from "..storageName..", sent = "..sent)
  767.                     if sent >= countRequired then
  768.                         exit = true
  769.                         break
  770.                     end         -- job done, no need to check remaining slots
  771.                     countRequired = countRequired - sent            -- this line not reached if sent >= count
  772.                 end
  773.             end
  774.             --if sent >= countRequired then break end               -- no need to check other storage
  775.             --if countRequired <= 0 then break end
  776.             if exit then
  777.                 return sent, countRequired
  778.             end
  779.         end
  780.        
  781.         return sent, countRequired
  782.     end
  783.    
  784.     -- eg slot, count = U.getItemFromNetwork("barrel", "minecraft:crafting_table", 1)
  785.     -- storageType either "chest" or "barrel"
  786.     if countRequired == nil then return 0,0 end
  787.     --if toTurtleSlot not specified then nil = use any slot
  788.     ignoreStock = ignoreStock or false  -- return only no of items obtained from storage
  789.     _G.Log:saveToLog("U.getItemFromNetwork(storageType = "..storageType..", itemRequired = ".. itemRequired..
  790.                 ", countRequired = ".. countRequired..", toTurtleSlot = "..tostring(toTurtleSlot)..", ignoreStock = "..tostring(ignoreStock))
  791.     -- Must be next to a modem: MUST remove crafting table if modem on that side. Other tools ok
  792.     local sent = 0
  793.     local turtleSlot, turtleCount = T:getItemSlot(itemRequired) -- check local stock
  794.     if not ignoreStock then -- take account of existing items and reduce count accordingly
  795.         countRequired = countRequired - turtleCount
  796.     end
  797.     local savedItems = U.chestItems
  798.     if storageType == "barrel" then
  799.         savedItems = U.barrelItems
  800.     --elseif storageType == "chest" then
  801.         --savedItems = chestItems
  802.     end
  803.     local message = U.wrapModem(R)  -- list of chest/barrel peripherals, name of turtle, list of storage names
  804.     --if turtleName == "Modem not found" then return 0, nil, nil, turtleName end
  805.     if countRequired > 0 then                       -- not enough in stock, or ignore current stock
  806.         -- check if item in storageLists
  807.         local testStores = nil
  808.         if savedItems[itemRequired] ~= nil then -- only works with full item names
  809.             _G.Log:saveToLog("savedItems key in list: "..textutils.serialise(savedItems[itemRequired], {compact = true}))
  810.             testStores = savedItems[itemRequired]
  811.         else
  812.             for key, value in pairs(savedItems) do
  813.                 if key:find(itemRequired)~= nil then
  814.                     _G.Log:saveToLog("savedItems found in list: "..textutils.serialise(value, {compact = true}))
  815.                     testStores = value
  816.                     break
  817.                 end
  818.             end
  819.         end
  820.         if testStores == nil then   -- no match in storage lists
  821.             _G.Log:saveToLog("Unable to find recorded storage, using all "..storageType.."s")
  822.             sent, countRequired = lib.getItem(storageNames, itemRequired, countRequired, toTurtleSlot, sent)
  823.         else -- match found, list of storage availble -- eg {"minecraft:barrel_17", "minecraft:barrel_18"...}
  824.             _G.Log:saveToLog("Using recorded list alias 'testStores'")
  825.             sent, countRequired = lib.getItem(testStores, itemRequired, countRequired, toTurtleSlot, sent)
  826.         end
  827.     end
  828.         -- slotData.lastSlot, total, slotData -- integer, integer, table
  829.     local data = {}
  830.     turtleSlot, turtleCount, data = T:getItemSlot(itemRequired)
  831.     _G.Log:saveToLog("turtleSlot = "..turtleSlot..", turtleCount = "..turtleCount..", sent = "..sent) --..", data = "..textutils.serialise(data))
  832.     if ignoreStock then
  833.         return turtleSlot, sent -- 0 -> count
  834.     else
  835.         return turtleSlot, turtleCount  -- 0 -> count
  836.     end
  837. end
  838.  
  839. function U.getSlotCapacity(slot)
  840.     return turtle.getItemSpace(slot) + turtle.getItemCount(slot)
  841. end
  842.  
  843. function U.getSlotContains(inventoryName, inSlot)
  844.     local list = peripheral.call(inventoryName, "list")
  845.     for slot, item in pairs(list) do
  846.         if inSlot == slot then
  847.             return item.name
  848.         end
  849.     end
  850.     return ""
  851. end
  852.  
  853. function U.moveItem(inventoryName, itemName, toSlot)
  854.     --[[ adapted from https://github.com/cc-tweaked/CC-Tweaked/discussions/1552
  855.     Move a specific item to specific slot eg 1, moving other items out of the way if needed.
  856.     inventoryName = The name of the chest/barrel/dropper to search.
  857.     itemName = The name of the item to find.
  858.     toSlot optional. default is slot 1
  859.     return boolean success Whether or not the item was successfully moved to toSlot (or already existed there)
  860.     ]]
  861.     toSlot = toSlot or 1
  862.     local list = peripheral.call(inventoryName, "list")
  863.     local size = peripheral.call(inventoryName, "size")
  864.     local slot = U.findItemInInventory(list, itemName)
  865.  
  866.   -- If the item didn't exist, or is already in the first slot, we're done.
  867.     if not slot then
  868.         _G.Log:saveToLog("U.moveItem(): Item not found")
  869.         return false
  870.     end
  871.     if slot == toSlot then
  872.         return true
  873.     end
  874.  
  875.     -- If an item is blocking the first slot (we already know it's not the one we want), we need to move it.
  876.     if list[toSlot] then
  877.         _G.Log:saveToLog("U.moveItem() Slot "..toSlot.." occupied, moving..")
  878.         local emptySlot = U.findEmptySlot(list, size)
  879.  
  880.         -- If there are no empty slots, we can't move the item.
  881.         if not emptySlot then
  882.             _G.Log:saveToLog("U.moveItem(): No empty slots")
  883.             return false
  884.         end
  885.  
  886.         -- Move the item to the first empty slot.
  887.        
  888.         if not U.moveItemStack(inventoryName, toSlot, emptySlot) then
  889.             _G.Log:saveToLog("U.moveItem(): Failed to move item to slot " .. emptySlot)
  890.             return false
  891.         end
  892.  
  893.         _G.Log:saveToLog("U.moveItem(): Moved item to slot " .. emptySlot)
  894.     end
  895.  
  896.     -- Move the item to slot 1.
  897.     if not U.moveItemStack(inventoryName, slot, toSlot) then
  898.         _G.Log:saveToLog("U.moveItem(): Failed to move item to slot "..toSlot)
  899.         return false
  900.     end
  901.  
  902.     _G.Log:saveToLog("U.moveItem(): Moved item to slot "..toSlot)
  903.     return true
  904. end
  905.  
  906. function U.moveItemsFromTurtle(toInventoryName, fromTurtleSlot, quantity, toSlot)
  907.     --[[
  908.     Move quantity of an item from one inventory to another. Turtles MUST use attachedInventory.pullItems()
  909.     eg U.moveItemsFromTurtle(turtleName, chestName, turtleSlot, turtleCount, nil)
  910.     turtleName:         The name of the turtle (via getLocalName())
  911.     toInventoryName:    The name of the inventory to move items into.
  912.     fromTurtleSlot:     The slot to move from. must be pre-determined for the item required
  913.     quantity:           The amount to transfer (nil for full stack)
  914.     toSlot:             The slot to move to. (nil will use any available slot(s))
  915.     ]]
  916.     return peripheral.call(toInventoryName, "pullItems", U.turtleName, fromTurtleSlot, quantity, toSlot)
  917. end
  918.  
  919. function U.moveItemStack(inventoryName, fromSlot, toSlot)
  920.     --[[ adapted from https://github.com/cc-tweaked/CC-Tweaked/discussions/1552
  921.     Move an item from one slot to another in a given inventory.
  922.     inventoryName The name of the inventory to move items in.
  923.     fromSlot The slot to move from.
  924.     toSlot The slot to move to.
  925.     ]]
  926.     return peripheral.call(inventoryName, "pushItems", inventoryName, fromSlot, nil, toSlot)
  927. end
  928.  
  929. function U.sendItemToNetworkStorage(storageType, itemToSend, amountToSend, fromSlot)
  930.     -- used to remove items from turtle inventory
  931.     -- Must be next to a modem: MUST remove crafting table if modem on that side. Other tools ok
  932.     -- fromSlot is given if emptying a specific slot
  933.     local peripheralNames = U.chestNames
  934.     if storageType == "barrel"then
  935.         peripheralNames = U.barrelNames
  936.     end  
  937.     local lib = {}
  938.    
  939.     function lib.sendItem(savedItems, peripheralNames, turtleSlot, item, slotCount, itemsPerSlot)
  940.         local storageToUse = ""
  941.         local storageList = lib.getStorageFromList(savedItems, item, slotCount, itemsPerSlot)   -- try from savedList
  942.         if storageList == nil then  -- no match found, but use first one found with U.wrapModem
  943.             _G.Log:saveToLog("No storage with matching items found, using first empty chest")
  944.             storageToUse = lib.findEmptyStorage(peripheralNames, item, itemsPerSlot, slotCount)
  945.         else
  946.             _G.Log:saveToLog("Storage with matching items found, checking capacity")
  947.             storageToUse  = lib.checkCapacity(storageList, item, slotCount, itemsPerSlot)
  948.             if storageToUse == "" then  -- no capacity in known storage list, so start a new one
  949.                 storageToUse = lib.findEmptyStorage(peripheralNames, item, itemsPerSlot, slotCount)
  950.             end
  951.         end
  952.         --U.moveItemsFromTurtle(turtleName, toInventoryName, fromTurtleSlot, quantity, toSlot)
  953.         _G.Log:saveToLog("U.moveItemsFromTurtle(turtleName = "..U.turtleName..", storageToUse = "..tostring(storageToUse)..", slot = "..tostring(turtleSlot)..", slotCount = "..tostring(slotCount)..")")
  954.         U.moveItemsFromTurtle(storageToUse, turtleSlot, slotCount)
  955.     end
  956.    
  957.     function lib.findEmptyStorage(peripheralNames, itemName, itemsPerSlot, itemCount)
  958.         for store = 1, #peripheralNames do
  959.             inStock, canStore, partMatch = U.checkInventory(peripheralNames[store], itemName, itemsPerSlot, "")
  960.             if canStore > itemCount then
  961.                 return peripheralNames[store]
  962.             end
  963.         end
  964.         return nil
  965.     end
  966.    
  967.     function lib.getStorageFromList(savedItems, item, sendAmount, itemsPerSlot)
  968.         if savedItems[item] == nil then                             -- no record of this item stored
  969.             _G.Log:saveToLog("lib.getStorageFromList() "..item.." not found")
  970.             local parts = T:getNames(item)                              -- eg minecraft:jungle_planks = "minecraft", "jungle", "planks"
  971.             for part = #parts, 1, -1 do                                 -- iterate "planks", "jungle", "minecraft"
  972.                 local searchTerm = parts[part]
  973.                 if searchTerm ~= "minecraft" and searchTerm ~= "mysticalagriculture" then
  974.                     for itemName, storageList in pairs(savedItems) do   -- iterate items used as keys eg minecraft:jungle_log matches jungle
  975.                         if itemName:find(searchTerm) ~= nil then        -- partial match eg "sapling" found in "minecraft:oak_sapling"
  976.                             _G.Log:saveToLog("lib.getStorageFromList() matched "..searchTerm.." with "..itemName)
  977.                             return storageList                          -- eg {"minecraft:chest_22", "minecraft:chest_23"}
  978.                         end
  979.                     end
  980.                 end
  981.             end
  982.         else
  983.             _G.Log:saveToLog("lib.getStorageFromList() ["..item.."] found")
  984.             return savedItems[item] -- list of chests with this item available
  985.         end
  986.         return nil
  987.     end
  988.    
  989.     function lib.checkCapacity(storageList, item, sendAmount, itemsPerSlot)
  990.         -- find a chest/barrel with sufficient capacity from list of storage
  991.         for store = 1, #storageList do
  992.             local inStock, canStore, partMatch = U.checkInventory(storageList[store], item, itemsPerSlot, "")
  993.             if canStore > sendAmount then
  994.                 return storageList[store]
  995.             end
  996.         end
  997.         return ""
  998.     end
  999.    
  1000.     function lib.send(storageType, peripheralNames, savedItems, itemToSend,  sourceSlot, slotCount, itemsPerSlot)
  1001.         local newStore = false
  1002.         local storageToUse = ""
  1003.         local storageList = lib.getStorageFromList(savedItems, itemToSend, slotCount, itemsPerSlot)
  1004.         if storageList == nil then
  1005.             storageToUse = lib.findEmptyStorage(peripheralNames, itemToSend, itemsPerSlot, slotCount)
  1006.             U.addToStorageList(storageType, itemToSend, storageToUse, true)
  1007.             return U.moveItemsFromTurtle(storageToUse, sourceSlot, slotCount)  
  1008.         else
  1009.             local storageToUse  = lib.checkCapacity(storageList, itemToSend, slotCount, itemsPerSlot)
  1010.             if storageToUse == "" then  -- no capacity in known storage list, so start a new one
  1011.                 storageToUse = lib.findEmptyStorage(peripheralNames, itemToSend, itemsPerSlot, slotCount)
  1012.                 newStore = true
  1013.             end
  1014.             _G.Log:saveToLog("sent = U.moveItemsFromTurtle(U.turtleName = "..U.turtleName..", storageToUse = "..storageToUse..", sourceSlot = "..sourceSlot..", slotCount = ".. slotCount)
  1015.             if newStore then
  1016.                 U.addToStorageList(storageType, itemToSend, storageToUse, true)
  1017.             end
  1018.             return U.moveItemsFromTurtle(storageToUse, sourceSlot, slotCount)  
  1019.         end
  1020.     end
  1021.    
  1022.     amountToSend = amountToSend or 0                        -- 0 = remove all of this item
  1023.     local totalSent = 0                                     -- track quantity sent
  1024.     local minSend = 0                                       -- minimum amount to send
  1025.     if amountToSend > 0 then minSend = amountToSend end     -- update minimum to send
  1026.     local message = U.wrapModem()
  1027.     if message ~= "" then
  1028.         return 0, message
  1029.     end
  1030.     --local storageToUse  = ""
  1031.     _G.Log:saveToLog("U.sendItemToNetworkStorage(storageType = '"..storageType.."', itemToSend = '"..itemToSend.."', amountToSend = "..amountToSend..")")
  1032.     local savedItems = nil
  1033.     if storageType == "barrel" then
  1034.         savedItems = U.barrelItems
  1035.     elseif storageType == "chest" then
  1036.         savedItems = U.chestItems
  1037.     end
  1038.    
  1039.     if itemToSend == "all" then -- empty Turtle, so item names not relevant
  1040.         _G.Log:saveToLog("itemToSend = all")
  1041.         repeat
  1042.             local item, turtleSlot, slotCount, itemsPerSlot = "", 0, 0, 64
  1043.             for slot = 1, 16 do
  1044.                 item, slotCount = T:getSlotContains(slot)
  1045.                 if slotCount > 0 then
  1046.                     turtleSlot = slot
  1047.                     _G.Log:saveToLog("for slot = 1, 16 do: item = "..item..", slotCount = "..slotCount)
  1048.                     itemsPerSlot = U.getSlotCapacity(slot)  -- most items capacity is 64 per slot
  1049.                     _G.Log:saveToLog("sending'"..item.."' from slot "..slot..", quantity = "..slotCount)
  1050.                     break
  1051.                 end
  1052.             end
  1053.             if turtleSlot > 0 then
  1054.                 lib.sendItem(savedItems, peripheralNames, turtleSlot, item, slotCount, itemsPerSlot)
  1055.             end
  1056.         until turtleSlot == 0
  1057.         return 0    -- exit function
  1058.     elseif fromSlot ~= nil then
  1059.         -- send 'amountToSend' from 'fromSlot'
  1060.         local slotCount = turtle.getItemCount(fromSlot)
  1061.         local itemsPerSlot = U.getSlotCapacity(fromSlot)
  1062.         return lib.send(storageType, peripheralNames, savedItems, itemToSend, fromSlot, slotCount, itemsPerSlot)
  1063.     else
  1064.         repeat  -- until item no longer present in inventory or requested amount has been sent
  1065.             local sourceSlot, total, data = T:getItemSlot(itemToSend)   -- which slot and how much of itemToSend is in turtle?
  1066.             local slotCount = data.leastCount
  1067.             _G.Log:saveToLog("T:getItemSlot('"..itemToSend.."' sourceSlot = "..sourceSlot..", total = "..total..")")
  1068.             if sourceSlot == 0 then
  1069.                 _G.Log:saveToLog(itemToSend.." not found in turtle inventory")
  1070.                 return 0    -- exit function
  1071.             else
  1072.                 local itemsPerSlot = U.getSlotCapacity(sourceSlot)  -- most items capacity is 64 per slot
  1073.                 itemToSend = data.leastName                             -- full name of item with lowest itemCount
  1074.                 _G.Log:saveToLog("U.sendItemToNetworkStorage("..itemToSend..", sourceSlot = "..sourceSlot..", slotCount = "..slotCount) --..", data = "..textutils.serialise(data)..")")
  1075.                 if sourceSlot > 0 then                                  -- item is present in turtle inventory
  1076.                     local sent = lib.send(storageType, peripheralNames, savedItems, itemToSend,  sourceSlot, slotCount, itemsPerSlot)
  1077.                     totalSent = totalSent + sent
  1078.                     if minSend > 0 and totalSent >= minSend then
  1079.                         return totalSent
  1080.                     end
  1081.                     if amountToSend > 0 then    -- sending specified amount
  1082.                         amountToSend = amountToSend - sent
  1083.                     end
  1084.                     if newStore then
  1085.                         U.addToStorageList(storageType, itemToSend, storageToUse, true)
  1086.                     end
  1087.                 end
  1088.             end
  1089.         until sourceSlot == 0
  1090.     end
  1091.    
  1092.     return totalSent
  1093. end
  1094.  
  1095. function U.sendItemsToCrafter(crafterName, fromInventoryName, fromInventorySlot, quantity, toCrafterSlot)
  1096.     --[[
  1097.     fromInventoryName:  The name of the inventory to move items from.
  1098.     fromInventorySlot:  The slot to move from. must be pre-determined for the item required
  1099.     quantity:           The amount to transfer (nil for full stack)
  1100.     toCrafterSlot:      The slot to move to. (nil will use any available slot(s))
  1101.     ]]
  1102.     return peripheral.call(fromInventoryName, "pushItems", crafterName, fromInventorySlot, quantity, toCrafterSlot)
  1103. end
  1104.  
  1105. function U.sendItemsToTurtle(fromInventoryName, fromInventorySlot, quantity, toTurtleSlot)
  1106.     --U.sendItemsToTurtle(turtleName, storageName, storageSlot, count, toTurtleSlot)
  1107.     --[[
  1108.     Move quantity of an item from one inventory to another. Turtles MUST use attachedInventory.pushItems()
  1109.     eg U.sendItemsToTurtle(turtleName, chestName, chestSlot, chestCount, 16) -- move to slot 16 so must be empty
  1110.     fromInventoryName:  The name of the inventory to move items from.
  1111.     fromInventorySlot:  The slot to move from. must be pre-determined for the item required
  1112.     quantity:           The amount to transfer (nil for full stack)
  1113.     toTurtleSlot:       The slot to move to. (nil will use any available slot(s))
  1114.     ]]
  1115.     return peripheral.call(fromInventoryName, "pushItems", U.turtleName, fromInventorySlot, quantity, toTurtleSlot)
  1116. end
  1117.  
  1118. function U.transferItem(fromInventoryName, toInventoryName, itemName, quantity, toSlot)
  1119.     --[[
  1120.     Move a specific number of an item from one inventory to another
  1121.     fromInventoryName:  The name of the chest/barrel/dropper to search.
  1122.     toInventoryName:    The name of the receiving inventory (chest/barrel/dropper/smelter)
  1123.     itemName:           The name of the item to find.
  1124.     toSlot:             optional. nil picks any slot
  1125.     return:             boolean success Whether or not the item was successfully moved to toSlot (or already existed there)
  1126.     ]]
  1127.     --_G.Log:saveToLog("U.transferItem(from: "..fromInventoryName..", to: "..toInventoryName..", itemName = "..itemName..", quantity = "..tostring(quantity)..", toSlot = "..tostring(toSlot))
  1128.     local list = peripheral.call(fromInventoryName, "list")
  1129.     local size = peripheral.call(fromInventoryName, "size")
  1130.     --_G.Log:saveToLog("U.transferItem() size = "..size..", list = \n"..textutils.serialize(list))
  1131.     local count, data = U.findItemCountInInventory(list, itemName)  -- either nil or eg {{1, 64},{4, 22}}
  1132.     --_G.Log:saveToLog("U.transferItem() data = "..textutils.serialize(data, {compact = true}))
  1133.     local remaining = quantity          -- eg 22 items needed
  1134.  
  1135.     if data == nil then -- Item not found
  1136.         return quantity -- return amount requested = nothing sent
  1137.     end
  1138.    
  1139.     local fromSlot = 0
  1140.     local count = 64
  1141.     local available = 0
  1142.     for _, v in pairs(data) do  -- eg {1, 64},{2, 64}
  1143.         if v[2] < count and v[2] >= quantity then
  1144.             fromSlot = v[1]
  1145.             count = v[2]
  1146.         else
  1147.             available = available + v[2]
  1148.         end
  1149.     end
  1150.     if fromSlot > 0 then                        -- now have slot with min required quantity
  1151.         _G.Log:saveToLog("U.transferItem() from: "..fromInventoryName..", to: "..toInventoryName..", fromSlot: "..fromSlot..", toSlot: "..tostring(toSlot)..", quantity: "..tostring(quantity))
  1152.         U.transferItems(fromInventoryName, toInventoryName, fromSlot, toSlot, quantity)
  1153.         return 0
  1154.     else                                    -- available must be at least 1
  1155.         for i = 1, #data do                 -- itreate all slots containg at least 1 item
  1156.             fromSlot = data[i][1]           -- eg slot 1
  1157.             local send = data[i][2]         -- eg 10 items
  1158.             if remaining - send < 0 then    -- eg 22 - 10 = 12
  1159.                 send = remaining
  1160.                 remaining = 0
  1161.             else
  1162.                 remaining = remaining - send-- eg remaining = 22 - 10 = 12
  1163.             end
  1164.             U.transferItems(fromInventoryName, toInventoryName, fromSlot, toSlot, send)
  1165.             if remaining <= 0 then          -- all required items transferred
  1166.                 return 0
  1167.             end
  1168.         end
  1169.     end
  1170.    
  1171.     return remaining                        -- return remaining items to be found from a  different inventory
  1172. end
  1173.  
  1174. function U.transferItems(fromInventoryName, toInventoryName, fromSlot, toSlot, quantity)
  1175.     --[[
  1176.     Move quantity of an item from one inventory to another
  1177.     fromInventoryName:  The name of the inventory to move items from.
  1178.     toInventoryName:    The name of the inventory to move items into.
  1179.     fromSlot:           The slot to move from. must be pre-determined for the item required
  1180.     toSlot:             The slot to move to. (nil will use any available slot(s))
  1181.     quantity:           The amount to transfer (nil for full stack)
  1182.     ]]
  1183.     _G.Log:saveToLog("U.transferItems(from: "..fromInventoryName..", to: "..toInventoryName..", fromSlot: "..fromSlot..", toSlot: "..tostring(toSlot)..", quantity: "..tostring(quantity)..")")
  1184.     return peripheral.call(fromInventoryName, "pushItems", toInventoryName, fromSlot, quantity, toSlot)
  1185. end
  1186.  
  1187. function U.transferItemToTurtle(availableStorage, availableStorageKeys, crafterData)
  1188.     -- U.transferItemToTurtle(<availableStorage>, data = {{1, 64},{4, 22}}, "crafter_01", <crafterData>)
  1189.     -- availableStorage.minecraft:chest_114 = {count = 86, data = {{1, 64},{4, 22}},
  1190.     -- availableStorage.minecraft:chest_115 = {count = 1024, data = {{1, 64},{2, 64},{3, 64}, ... }
  1191.     -- crafterData = {{2,64}, {4,64}, {6,64}, {8,64}} 64 items in each of 4 slots in the crafter
  1192.     -- glitch? in crafter inventory, cannot add sequentially to existing items.
  1193.     -- send to turtle slot first, then transfer
  1194.     _G.Log:saveToLog("U.transferItemToTurtle(availableStorage = "..textutils.serialise(availableStorage, {compact = true})..
  1195.                 "\navailableStorageKeys = "..textutils.serialise(availableStorageKeys, {compact = true})..
  1196.                 "\n"..U.turtleName..", crafterData = "..textutils.serialise(crafterData, {compact = true}))
  1197.                
  1198.     local total = 0
  1199.     local numSlots = 0
  1200.     local sent = 0
  1201.     for _, v in ipairs(crafterData) do                              -- how many items required in total?
  1202.         total = total + v[2]                                        -- how many slots does it go in
  1203.         numSlots = numSlots + 1
  1204.     end
  1205.     for _, availableStorageKey in ipairs(availableStorageKeys) do   -- eg {minecraft:chest_114, minecraft:chest_115}
  1206.         local storageName = availableStorageKey                     -- eg minecraft:chest_114
  1207.         local object = availableStorage[storageName]                -- availableStorage.minecraft:chest_114 = {count = 90, data = {{14,64},{15,26}}
  1208.         local storageData = object.data                             -- eg data = {{14,64},{15,26}}
  1209.         local storageCount = object.count                           -- eg count = 90
  1210.         for _, crafterSlotData in ipairs(crafterData) do            -- eg {{2,22}, {4,22}, {6,22}, {8,22}} -> iteration 1 = {2, 22} iterate crafter slots to be filled
  1211.             local toCrafterSlot = crafterSlotData[1]                -- eg slot 2 in turtle
  1212.             local amountToSend = crafterSlotData[2]                 -- eg place 22 items in slot 2
  1213.             _G.Log:saveToLog("storageData = "..textutils.serialise(storageData, {compact = true}))
  1214.             _G.Log:saveToLog("crafterSlotData = "..textutils.serialise(crafterSlotData, {compact = true}))
  1215.             for i = 1, #storageData do                              -- {{14,64},{15,26}}                   
  1216.                 local slotData = storageData[i]                     -- {14,64}
  1217.                 local availableToSend = slotData[2]                 -- 64
  1218.                 local fromStorageSlot = slotData[1]                 -- 14
  1219.                 local confirmedSent = 0
  1220.                
  1221.                 _G.Log:saveToLog("i = "..i..", slotData = "..textutils.serialise(slotData, {compact = true}))
  1222.                 if availableToSend >= amountToSend then
  1223.                     _G.Log:saveToLog("availableToSend ("..availableToSend..") >= amountToSend: ("..amountToSend.."), current value of sent = "..sent)
  1224.                     _G.Log:saveToLog("?confirmedSent = peripheral.call("..storageName..", 'pushItems', "..U.turtleName..
  1225.                                 ", from slot "..fromStorageSlot..", amountToSend = "..
  1226.                                 amountToSend..", to turtle slot "..toCrafterSlot)
  1227.                     confirmedSent = peripheral.call(storageName, "pushItems", U.turtleName, fromStorageSlot, amountToSend, toCrafterSlot)
  1228.                     sent = sent + confirmedSent
  1229.                     _G.Log:saveToLog("verified confirmedSent = "..confirmedSent..", sent = "..sent)
  1230.                     slotData[2] = slotData[2] - confirmedSent
  1231.                     crafterSlotData[2] = 0
  1232.                     _G.Log:saveToLog("slotData[2] = "..slotData[2]..", crafterSlotData[2] = "..crafterSlotData[2])
  1233.                 else
  1234.                     _G.Log:saveToLog("availableToSend ("..availableToSend..") < amountToSend: ("..amountToSend.."), current value of sent = "..sent)
  1235.                     _G.Log:saveToLog("?confirmedSent = peripheral.call("..storageName..", 'pushItems', "..U.turtleName..
  1236.                                 ", from slot "..fromStorageSlot..", availableToSend = "..
  1237.                                 availableToSend..", to turtle slot "..toCrafterSlot)
  1238.                     -- taking items from multiple storage slots requires loading into turtle first
  1239.                     confirmedSent = peripheral.call(storageName, "pushItems", U.turtleName, fromStorageSlot, availableToSend, toCrafterSlot)
  1240.                     sent = sent + confirmedSent
  1241.                     _G.Log:saveToLog("verified confirmedSent = "..confirmedSent..", sent = "..sent)
  1242.                     amountToSend = amountToSend - confirmedSent
  1243.                     slotData[2] = slotData[2] - confirmedSent
  1244.                     crafterSlotData[2] = amountToSend
  1245.                     _G.Log:saveToLog("slotData[2] = "..slotData[2]..", crafterSlotData[2] = "..crafterSlotData[2])
  1246.                 end
  1247.                
  1248.                 if crafterSlotData[2] == 0 then
  1249.                     _G.Log:saveToLog("crafterSlotData[2]("..crafterSlotData[2]..") == 0: breaking\n")
  1250.                     break   -- already sent correct amount
  1251.                 end
  1252.             end
  1253.         end
  1254.         if sent >= total then
  1255.             _G.Log:saveToLog("sent("..sent..") >= total ("..total.."): breaking\n")
  1256.             break
  1257.         end
  1258.     end
  1259.  
  1260.     return sent
  1261. end
  1262.  
  1263. function U.pullItems(fromName, fromSlot, quantity, toName, toSlot)
  1264.     quantity = quantity or 1
  1265.     toSlot = toSlot or nil
  1266.     if type(fromName) == "table" then           -- already a wrapped peripheral
  1267.         fromName = peripheral.getName(fromName)
  1268.     end
  1269.     if type(toName) == "string" then
  1270.         toName = peripheral.wrap(toName)
  1271.     end
  1272.     toName.pullItems(fromName, fromSlot, quantity, toSlot)
  1273. end
  1274.  
  1275. function U.updateList(storageType)
  1276.     local output = ""
  1277.     if storageType == "barrel" then
  1278.         output = textutils.serialize(U.barrelItems)     -- serialise to json ready to write to file
  1279.     elseif storageType == "chest" then
  1280.         output = textutils.serialize(U.chestItems)      -- serialise to json ready to write to file
  1281.     end
  1282.     local fileName = "lib/data/"..storageType.."Items.lua"          -- barrelList.lua or chestList.lua
  1283.     local outputHandle = fs.open(fileName, "w")         -- open file
  1284.     outputHandle.writeLine("return")                    -- start file with return
  1285.     outputHandle.write(output)                          -- add serialised table
  1286.     outputHandle.close()                                -- close file
  1287. end
  1288.  
  1289. function U.updateStorageFile(storageType)
  1290.     -- storageType should be "chest" or "barrel"
  1291.     local output  = nil
  1292.     if storageType == "chest" then
  1293.         output = textutils.serialize(U.chestItems)                  -- serialise to json ready to write to file
  1294.     else
  1295.         output = textutils.serialize(U.barrelItems)                 -- serialise to json ready to write to file
  1296.     end
  1297.     local outputHandle = fs.open("lib/data/"..storageType.."Items.lua", "w")        -- open file
  1298.     outputHandle.writeLine("return")                                -- start file with return
  1299.     outputHandle.write(output)                                      -- add serialised table
  1300.     outputHandle.close()                               
  1301. end
  1302.  
  1303. function U.wrapModem()
  1304.     --[[To move turtle inventory items use the target peripheral:
  1305.         local modem = peripheral.wrap("front")      -- wrap modem next to turtle (beware crafting table!)
  1306.         local turtleName = modem.getNameLocal()     -- get name of the turtle
  1307.         local barrel = peripheral.find("barrel")    -- find barrel name you want to receive goods
  1308.         barrel.pushItems(turtleName, 1, 1)          -- push items FROM turtle to barrel  pushItems(toName, fromSlot , limit , toSlot)
  1309.         barrel.pullItems(turtleName, fromSlot , limit , toSlot)
  1310.     ]]
  1311.     local modem = peripheral.find("modem")      -- find modem
  1312.     if modem == nil then
  1313.         return "Modem not found"
  1314.     end
  1315.     _G.Log:saveToLog("network.wrapModem: U.turtleName = "..U.turtleName)
  1316.     if U.turtleName == "" then                      -- modem not already wrapped
  1317.         -- populate module level variables barrelObjects, barrelNames, chestObjects, chestNames
  1318.         _G.Log:saveToLog("creating U. lists/objects")
  1319.         U.chestObjects = {}
  1320.         U.chestNames  = {}
  1321.         U.barrelObjects = {}
  1322.         U.barrelNames  = {}
  1323.         U.turtleName = modem.getNameLocal()         -- get name of the turtle (module scope variable)
  1324.         _G.Log:saveToLog("U.turtleName (updated) = "..U.turtleName)
  1325.         local rawStorage = nil                      -- peripheral find can return duplicate values
  1326.         rawStorage = {peripheral.find("minecraft:barrel")}
  1327.         for k, value in ipairs(rawStorage) do
  1328.             local name = peripheral.getName(value)
  1329.             if not U.tableContains(U.barrelNames, name, true) then  -- use exact match as checking peripherals
  1330.                 table.insert(U.barrelObjects, value)
  1331.                 table.insert(U.barrelNames, name)
  1332.             end
  1333.         end
  1334.         rawStorage = {peripheral.find("minecraft:chest")}
  1335.         for k, value in ipairs(rawStorage) do
  1336.             local name = peripheral.getName(value)
  1337.             if not U.tableContains(U.chestNames, name, true) then   -- use exact match as checking peripherals
  1338.                 table.insert(U.chestObjects, value)
  1339.                 table.insert(U.chestNames, name)
  1340.             end
  1341.         end
  1342.         --_G.Log:saveToLog("U.barrelNames = "..textutils.serialise(U.barrelNames, {compact = true}))
  1343.         table.sort(U.barrelNames, function(a,b) return tonumber(a:sub(18)) < tonumber(b:sub(18)) end)
  1344.         table.sort(U.chestNames, function(a,b) return tonumber(a:sub(17)) < tonumber(b:sub(17)) end)
  1345.         --_G.Log:saveToLog("sorted U.barrelNames = "..textutils.serialise(U.barrelNames, {compact = true}))
  1346.         table.sort(U.barrelObjects, function(a,b) return tonumber(peripheral.getName(a):sub(18)) < tonumber(peripheral.getName(b):sub(18)) end)
  1347.         table.sort(U.chestObjects, function(a,b) return tonumber(peripheral.getName(a):sub(17)) < tonumber(peripheral.getName(b):sub(17)) end)
  1348.     end
  1349.     return ""
  1350. end
  1351. --SCENE UTILITIES
  1352. function U.fillCrafts()
  1353.     U.crafts = {}
  1354.     local lib = {}
  1355.     function lib.isPresent(value)
  1356.         for i = 1, #U.crafts do
  1357.             if U.crafts[i] == value then
  1358.                 return true
  1359.             end
  1360.         end
  1361.         return false
  1362.     end
  1363.    
  1364.    
  1365.     for c = 1, #categories do
  1366.         for _, v in ipairs(Items[categories[c]]) do
  1367.             if v.recipes ~= nil then
  1368.                 if not lib.isPresent(v.displayName) then
  1369.                     table.insert(U.crafts, v.displayName)
  1370.                 end
  1371.             end
  1372.         end
  1373.     end
  1374. end
  1375.  
  1376. function U.fillSmelts()
  1377.     --[[ Items.lua example
  1378.     {
  1379.       name = "minecraft:iron_ingot",
  1380.       displayName = "Iron Ingot",
  1381.       recipe = {"minerals","ingot"},
  1382.       smelt = {"minecraft:raw_iron"}
  1383.     },
  1384.     ]]
  1385.     U.smeltFrom = {}    -- will contain list of items derived from smelt with a table of source items
  1386.     U.smelts = {}       -- will contain list of display names of all items that can be smelted
  1387.     for c = 1, #categories do
  1388.         for _, v in ipairs(Items[categories[c]]) do
  1389.             if v.smelt ~= nil then
  1390.                 table.insert(U.smelts, v.displayName)           -- eg "Iron Ingot"
  1391.                 U.smeltFrom[v.name] = {}
  1392.                 for k, source in ipairs(v.smelt) do
  1393.                     table.insert(U.smeltFrom[v.name], source)   -- eg ["minecraft:coal"] = {"minecraft:coal_ore", "minecraft:deepslate_coal_ore"}
  1394.                 end
  1395.             end
  1396.         end
  1397.     end
  1398. end
  1399.  
  1400. function U.findRecipe(from, key)
  1401.     -- eg "displayName", "Oak Stairs", "minecraft:oak_stairs" --> searches in Items only NOT in storage
  1402.     local recipe = {}
  1403.     for c = 1, #categories do
  1404.         for _, v in ipairs(Items[categories[c]]) do
  1405.             if from == "displayName" then
  1406.                 if v.displayName == key then                    -- eg "Oak Stairs"
  1407.                     if v.recipe ~= nil then                     -- eg  {"wood", "stairs"} or {{"colored","bed"},{"colored", "directColoured"}}
  1408.                         for _, value in ipairs(v.recipe) do
  1409.                             table.insert(recipe, value)
  1410.                         end
  1411.                         --_G.Log:saveToLog("\tU.findRecipe: '"..key.."'\n"..textutils.serialize(recipe, {compact = true}))
  1412.                         return recipe                           -- {"wood", "stairs"}
  1413.                     end
  1414.                 end
  1415.             elseif from == "name" then
  1416.                 if v.name == key then                           -- eg "minecraft:oak_stairs"
  1417.                     if v.recipe ~= nil then                     -- eg  {"wood", "stairs"} or {{"colored","bed"},{"colored", "directColoured"}}
  1418.                         for _, value in ipairs(v.recipe) do
  1419.                             table.insert(recipe, value)
  1420.                         end
  1421.                         --_G.Log:saveToLog("\tU.findRecipe '"..key.."' = {'"..recipe[1].."', '"..recipe[2].."'}")
  1422.                         return recipe                           -- {"wood", "stairs"} or {{"colored","bed"},{"colored", "directColoured"}}
  1423.                     end
  1424.                 end
  1425.             end
  1426.         end
  1427.     end
  1428.    
  1429.     return nil 
  1430. end
  1431.  
  1432. function U.findSmeltItem(item)
  1433.     return U.smeltFrom[item]
  1434. end
  1435.  
  1436. function U.findSmeltSources(smeltItem)
  1437.     -- key eg minecraft:iron_ingot or minecraft:coal
  1438.     --_G.Log:saveToLog("1:U.findSmeltSources("..smeltItem..")")
  1439.     local smeltSources = nil
  1440.     local retValue = {}
  1441.     local matches = nil
  1442.     local found = false
  1443.     for c = 1, #categories do
  1444.         for _, v in pairs(Items[categories[c]]) do
  1445.             if v.name == smeltItem then
  1446.                 --_G.Log:saveToLog("2:U.findSmeltSources("..smeltItem..") v.smelt = \n"..textutils.serialize(v.smelt))
  1447.                 smeltSources = v.smelt
  1448.                 found = true
  1449.                 break
  1450.                 --return v.smelt -- eg {"minecraft:raw_iron"} or {"log","wood"}
  1451.             end
  1452.         end
  1453.         if found then break end
  1454.     end
  1455.     if smeltSources ~= nil then
  1456.         for k,v in pairs(smeltSources) do
  1457.             if v:find(":") == nil then
  1458.                 -- find all types of eg "log" from Items
  1459.                 local key = v
  1460.                 if v == "golden_item" then
  1461.                     key = "golden"
  1462.                 elseif v == "iron_item" then
  1463.                     key = "iron"
  1464.                 end
  1465.                 --_G.Log:saveToLog("3:U.findSmeltSource("..smeltItem..") key = "..key)
  1466.                 matches = U.searchAllItems(key)
  1467.                 --_G.Log:saveToLog("4:U.findSmeltSource("..smeltItem..") matches = "..textutils.serialize(matches, {compact = true}))
  1468.                 for _, v1 in ipairs(matches) do
  1469.                     table.insert(retValue, v1)
  1470.                 end
  1471.             else
  1472.                 table.insert(retValue, v)
  1473.             end
  1474.         end
  1475.         return retValue
  1476.     end
  1477.    
  1478.     return nil
  1479. end
  1480.  
  1481. function U.findSmeltLocations(storageType, sources, smeltItem)
  1482.     -- {log, wood} or {iron_item} or {golden_item} or {minecraft:cobblestone"}
  1483.     --local sources = {}
  1484.     local lib = {}
  1485.    
  1486.     function lib.cleanAdd(storageType, sources, smeltItem)
  1487.         local temp = U.findItemInInventory(storageType, smeltItem, false)   -- eg {"minecraft:chest_19", "minecraft:chest_20"}
  1488.         _G.Log:saveToLog("U.findSmeltLocations(smeltItems) lib.cleanAdd , temp = \n"..textutils.serialize(temp))
  1489.        
  1490.         for k, v in pairs(temp) do  -- eg {"minecraft:chest_19", "minecraft:chest_20"}
  1491.             _G.Log:saveToLog("U.findSmeltLocations(smeltItems) lib.cleanAdd , k = "..k..", v = "..v)
  1492.             local found = false
  1493.             for k1,v1 in pairs(sources) do
  1494.                 _G.Log:saveToLog("U.findSmeltLocations(smeltItems) lib.cleanAdd , k1 = "..k1..", v1 = "..v1)
  1495.                 if v == v1 then
  1496.                     found = true
  1497.                 end
  1498.             end
  1499.             if not found then
  1500.                 table.insert(sources, v)
  1501.                 _G.Log:saveToLog("U.findSmeltLocations(smeltItems) lib.cleanAdd table.insert v = "..v)
  1502.             end
  1503.         end
  1504.        
  1505.         return sources
  1506.     end
  1507.        
  1508.     if smeltItem:find(":") == nil then
  1509.         if smeltItem == "log" or smeltItem == "wood" then
  1510.             sources = lib.cleanAdd(storageType, sources, smeltItem)
  1511.             _G.Log:saveToLog("U.findSmeltLocations(smeltItems) added "..smeltItem..", sources = \n"..textutils.serialize(sources))
  1512.             --table.insert(sourcesue, U.findItemInInventory(v, false))
  1513.         elseif smeltItem == "golden_item" then
  1514.             sources = lib.cleanAdd(storageType, sources, "golden")
  1515.             --table.insert(sourcesue, U.findItemInInventory("golden", false))
  1516.         elseif smeltItem == "iron_item" then
  1517.             sources = lib.cleanAdd(storageType, sources, "iron")
  1518.             --table.insert(sourcesue, U.findItemInInventory("iron", false))
  1519.         end
  1520.     else
  1521.         sources = lib.cleanAdd(storageType, sources, smeltItem)
  1522.         --table.insert(sourcesue, v)
  1523.     end
  1524.     return sources
  1525. end
  1526.  
  1527. function U.getAvailable(storageType, item)
  1528.     --[[
  1529.     eg reqItem = "minecraft:oak_trapdoor"
  1530.     locations table example
  1531.     [ "minecraft:cobblestone" ] =
  1532.     {
  1533.         "minecraft:chest_60",
  1534.         "minecraft:chest_62",
  1535.         "minecraft:chest_61",
  1536.         "minecraft:chest_1",
  1537.     }
  1538.     ]]
  1539.     if item:find(":") == nil then
  1540.         item = "minecraft:"..item
  1541.     end
  1542.  
  1543.     local quantity = 0
  1544.     local total = 0
  1545.     local availableList = {}
  1546.     local currentItemLocations = {}
  1547.     local storage = U.barrelObjects
  1548.     if storageType == "chest" then
  1549.         storage = U.chestObjects
  1550.     end
  1551.    
  1552.     if storage[item] ~= nil then
  1553.         currentItemLocations = storage[item]                -- eg U.locations["minecraft:oak_log"] = {"chest_38"}
  1554.         for _, chestName in ipairs(currentItemLocations) do
  1555.             --_G.Log:saveToLog("S.populateAvailable(self,".. self.currentItem..") - chestName = ".. chestName)
  1556.             local chest = peripheral.wrap(chestName)            -- eg "minecraft:chest_60"
  1557.             local contents = chest.list()                       -- list of items / slots for this chest
  1558.             quantity = 0
  1559.             for slot, item in pairs(contents) do                -- for each item check if this storage is listed
  1560.                 if item.name == item then
  1561.                     quantity = quantity + item.count
  1562.                     total = total + item.count
  1563.                 end
  1564.             end
  1565.             table.insert(availableList, chestName:sub(11).." "..quantity.." "..item:sub(11))
  1566.             --_G.Log:saveToLog("S.populateAvailable(self,".. self.currentItem..") - table.insert: "..chestName:sub(11).." "..quantity.." "..self.currentItem:sub(11))
  1567.         end
  1568.     end
  1569.    
  1570.     return availableList, total
  1571. end
  1572.  
  1573. function U.getCategoryTable(category)
  1574.     return Items[category]
  1575. end
  1576.  
  1577. function U.getCraftDetails(value)
  1578.     -- value can be recipe table OR string item name
  1579.     local ratio = 1
  1580.     local recipe = nil
  1581.     local item = ""
  1582.     local craftRecipe = nil
  1583.     --_G.Log:saveToLog("\tU.getCraftDetails("..tostring(value)..")")
  1584.     if type(value) == "string" then
  1585.         item = value -- value is string
  1586.         if value:find(":") ~= nil then
  1587.             recipe = U.findRecipe("name", value)
  1588.         else
  1589.             recipe = U.findRecipe("displayName", value)
  1590.         end
  1591.     else
  1592.         item = value[2]
  1593.     end
  1594.     --[[ eg for spruce sign  makes 6. quantity = 2 ratio = 3
  1595.     craftingRecipe =
  1596.     {
  1597.         {
  1598.             [minecraft:spruce_planks].quantity = 2,
  1599.             [minecraft:spruce_planks].slots = {1,2,3,5,6,7}
  1600.         },
  1601.         {
  1602.             [minecraft:stick].quantity = 2,
  1603.             [minecraft:stick].slots = {10}
  1604.         }
  1605.     }
  1606.     ]]
  1607.     if recipe == nil then
  1608.         _G.Log:saveToLog("\tU.getCraftDetails("..item..").recipe = nil, ratio = 1")
  1609.         return recipe, craftRecipe, ratio
  1610.     end
  1611.     -- eg recipe = {"wood", "stairs"} or {"wood","minecraft:bamboo_planks"} or {{"colored","carpet"}, {"colored","directColoured"}}
  1612.     if type(recipe[1]) == "table" then  -- {{"colored","carpet"}, {"colored","directColoured"}}
  1613.         craftRecipe = {}
  1614.         _G.Log:saveToLog("\tU.getCraftDetails("..item..").recipe[1] = table: "..textutils.serialise(recipe[1], {compact = true}))
  1615.         for k, value in ipairs(recipe) do                   -- eg {"colored","carpet"}
  1616.             local testRecipe = Recipes[value[1]][value[2]]  -- eg {"","","","wool","wool","","","","","3",}
  1617.             _G.Log:saveToLog("\tU.getCraftDetails("..item..").testRecipe["..k.."] = "..textutils.serialise(testRecipe, {compact = true}))
  1618.             local temp = {}
  1619.             for _, name in ipairs(testRecipe) do
  1620.                 table.insert(temp, name)
  1621.             end
  1622.             table.insert(craftRecipe, temp)
  1623.             ratio = tonumber(temp[10])
  1624.         end
  1625.     else    -- {"wood","minecraft:bamboo_planks"}
  1626.         local testRecipe = Recipes[recipe[1]][recipe[2]]
  1627.         assert(testRecipe ~= nil,"Recipe for "..item .."('"..recipe[1].."', '"..recipe[2].."') does not exist")
  1628.         craftRecipe = {}
  1629.         for _, name in ipairs(testRecipe) do
  1630.             table.insert(craftRecipe, name)
  1631.         end
  1632.         ratio = tonumber(craftRecipe[10])
  1633.     end
  1634.  
  1635.     if ratio == nil then
  1636.         _G.Log:saveToLog("\tU.getCraftDetails("..item..").recipe = "..textutils.serialise(recipe, {compact = true}).. ", default ratio = 1")
  1637.         return recipe, craftRecipe, 1
  1638.     end
  1639.     _G.Log:saveToLog("\tU.getCraftDetails("..item..").return: recipe = ".. textutils.serialise(recipe, {compact = true})..", ratio (tonumber) = "..tostring(ratio)..", craftRecipe = "..textutils.serialise(craftRecipe, {compact = true}))
  1640.     --_G.Log:saveToLog("\tU.getCraftDetails("..item..").craftRecipe = "..textutils.serialise(craftRecipe, {compact = true}))
  1641.     return recipe, craftRecipe, ratio   -- recipe, craftRecipe, ratio
  1642.     -- {'wood', 'minecraft:bamboo_block'},
  1643.     -- {"minecraft:bamboo","minecraft:bamboo","minecraft:bamboo","minecraft:bamboo","minecraft:bamboo","minecraft:bamboo","minecraft:bamboo","minecraft:bamboo","minecraft:bamboo","1"},
  1644.     -- 1
  1645. end
  1646.  
  1647. function U.getCraftingRecipe(value)
  1648.     -- eg recipe = {"wood", "stairs"} or recipe = "minecraft:oak_stairs"
  1649.     if value == nil then return nil end
  1650.     local recipe = nil
  1651.     if type(value) == "string " then
  1652.         if value:find(":") ~= nil then
  1653.             recipe = U.findRecipe("name", value)
  1654.         else
  1655.             recipe = U.findRecipe("displayName", value)
  1656.         end
  1657.     end
  1658.     if recipe == nil then return nil end
  1659.    
  1660.     if type(recipe[1]) == "string" then
  1661.         local temp = Recipes[recipe[1]][recipe[2]]
  1662.         local retValue = {}
  1663.         for _, v in ipairs(temp) do
  1664.             table.insert(retValue, v)
  1665.         end
  1666.     else
  1667.         local retValue = {}
  1668.         for k,v in pairs(recipe) do -- eg {{"colored","bed"},{"colored", "directColoured"}}
  1669.             local temp = Recipes[v[1]][v[2]]
  1670.             local temp2 = {}
  1671.             for k1, v1 in ipairs(temp) do
  1672.                 table.insert(temp2, v1)
  1673.             end
  1674.             table.insert(retValue, temp2)
  1675.         end
  1676.     end
  1677.     return retValue
  1678.     -- eg {"planks", "", "", "planks", "planks", "", "planks", "planks", "planks"}
  1679.     -- or {{"", "", "", "wool", "wool", "wool", "planks", "planks", "planks", "1"}, {"", "", "", "item", "dye", "", "", "", "", "1"}}
  1680. end
  1681.  
  1682. function U.getCurrentCursor(withColours)
  1683.     withColours = withColours or false
  1684.     if withColours then
  1685.         return currentCursor.x, currentCursor.y, currentCursor.fg, currentCursor.bg
  1686.     else
  1687.         return currentCursor.x, currentCursor.y
  1688.     end
  1689. end
  1690.  
  1691. function U.getEmptyChests(limit)
  1692.     limit = limit or 1
  1693.     local emptyChests = {}
  1694.     local count = 0
  1695.     for i = 1, #U.chestNames do
  1696.         local isEmpty = true
  1697.         for slot, item in pairs(U.chestNames[i].list()) do
  1698.             if item.count > 0 then
  1699.                 isEmpty = false
  1700.                 break
  1701.             end
  1702.         end
  1703.         if isEmpty then
  1704.             table.insert(emptyChests, peripheral.getName(U.chestNames[i]))
  1705.             count = count + 1
  1706.             if count >= limit then
  1707.                 return emptyChests
  1708.             end
  1709.         end
  1710.     end
  1711. end
  1712.  
  1713. function U.getIngredientFullName(recipe, item, key)
  1714.     -- eg recipe = {"wood", "stairs"}
  1715.     -- eg item = "minecraft:oak_stairs"
  1716.     -- eg key = "planks" from recipes["wood"]["stairs"] = {"planks", "", "", "planks", "planks", "", "planks", "planks", "planks"}
  1717.     -- eg convert "planks" to "minecraft:oak_planks" for use in "minecraft:oak_stairs"
  1718.     if key:find(":") ~= nil then                    -- defined item --> return
  1719.         return key
  1720.     end
  1721.     local category, itemType = "", ""
  1722.     if type(recipe) == "table" then
  1723.         category = recipe[1]    -- eg wood, stone, minerals
  1724.         itemType = recipe[2]    -- eg planks, stairs, slab, pressure_plate. all items containing : will have already returned
  1725.         --_G.Log:saveToLog("\tU.getIngredientFullName.recipe = ".. textutils.serialise(recipe, {compact = true}))
  1726.     else
  1727.         category = recipe
  1728.         --_G.Log:saveToLog("\tU.getIngredientFullName.recipe = ".. recipe)
  1729.     end
  1730.     --_G.Log:saveToLog("\tU.getIngredientFullName.category = "..category..", key = "..key)
  1731.     local data = U.split(item, ":")                 -- eg "minecraft:oak_stairs" --> "minecraft", "oak_stairs"
  1732.     local words = U.split(data[2], "_")             -- eg "red_candle" --> "red", "candle"
  1733.     --local words = U.split(modType[2], "_")        -- eg "oak_stairs" --> "oak", "stairs" or "oak_fence_gate" --> "oak", "fence", "gate"
  1734.     modType = data[1]..":"                          -- eg "minecraft" --> "minecraft:" now string not table
  1735.     local _, woodType =  U.isTimber(item)
  1736.     local _, stoneType = U.isStone(item)
  1737.     local itemColour = U.getItemColour(item)
  1738.     local _, mineralType = U.isMineral(item)
  1739.     local _, ingotType = U.isIngot(item)
  1740.     local retValue = ""
  1741.     --_G.Log:saveToLog("\tU.getIngredientFullName woodType = "..woodType..", stoneType = "..stoneType..", itemColour = "..itemColour..", mineralType = "..mineralType..", ingotType = "..ingotType)
  1742.     if category == "wood" then                          -- eg "minecraft:oak_stairs" or "minecraft:oak_fence_gate"
  1743.         -- keys = stem, log, planks
  1744.         local modifier = ""
  1745.         if item:find("stripped") ~= nil then
  1746.             modifier = "stripped_"
  1747.         end
  1748.         retValue = modType..modifier..woodType.."_"..key
  1749.     elseif category == "redstone" then
  1750.         -- keys = planks, wood_slab
  1751.         -- for simplicity return only oak_planks, oak_slabs
  1752.         if key == "wood_slab" then
  1753.             retValue = "minecraft:oak_slab"
  1754.         elseif key == "planks" then
  1755.             retValue "minecraft:oak_planks"
  1756.         end
  1757.     elseif category == "stone" then
  1758.         -- keys = stone
  1759.         -- eg recipe = {"stone", "stairs"} = {"stone", "", "", "stone", "stone", "", "stone", "stone", "stone"},
  1760.         -- eg item = "minecraft:polished_granite_stairs"
  1761.         if itemType == "polished" then
  1762.             return modType..stoneType
  1763.         end
  1764.         local modifier1 = ""
  1765.         if item:find("cobbled") ~= nil then
  1766.             modifier1 = "cobbled_"
  1767.         elseif item:find("cracked") ~= nil then
  1768.             modifier1 = "cracked_"
  1769.         end
  1770.         local modifier2 = ""
  1771.         if item:find("polished") ~= nil then
  1772.             modifier2 = "polished_"
  1773.         elseif item:find("smooth") ~= nil then
  1774.             modifier2 = "smooth_"
  1775.         elseif item:find("mossy") ~= nil then
  1776.             modifier2 = "mossy_"
  1777.         elseif item:find("chiseled") ~= nil then
  1778.             modifier2 = "chiseled_"
  1779.         elseif item:find("cut") ~= nil then
  1780.             modifier2 = "cut_"
  1781.         end
  1782.         local modifier3 = ""
  1783.         if item:find("brick") ~= nil then       -- eg "minecraft:stone_brick_stairs"
  1784.             if item:find("bricks") == nil then  -- excludes "minecraft:stone_bricks"
  1785.                 modifier3 = "_bricks"
  1786.             end
  1787.         elseif item:find("tile") ~= nil then    -- eg "minecraft:deepslate_tile_stairs"
  1788.             if item:find("tiles") == nil then   -- excludes "minecraft:deepslate_tiles"
  1789.                 modifier3 = "_tiles"
  1790.             end
  1791.         end
  1792.         retValue = modType..modifier1..modifier2..stoneType..modifier3
  1793.         --_G.Log:saveToLog("\tU.getIngredientFullName().retValue = "..retValue)
  1794.         if retValue:find("_") == 11 then        -- "minecraft:_bricks"
  1795.             --_G.Log:saveToLog("\tU.getIngredientFullName().retValue.gsub('_', '', 1) "..retValue:gsub("_", "", 1))
  1796.             retValue = retValue:gsub("_", "", 1)        -- replace 1 instance of "_"
  1797.         end
  1798.     elseif category == "minerals" then                  -- eg {"minerals","minecraft:iron_trapdoor"} --> { "", "", "","minecraft:iron_ingot", "minecraft:iron_ingot", "","minecraft:iron_ingot", "minecraft:iron_ingot", ""},  
  1799.         -- names = {"coal", "iron", "gold", "redstone", "emerald", "lapis_lazuli", "diamond", "netherite", "quartz", "amethyst", "copper"}
  1800.         -- keys =  block, ingot, mineral, copperType
  1801.         -- eg recipe = {"minerals","ingot"}, item = "minecraft:iron_ingot", item = "block": from {"block", "", "", "", "", "", "", "", "", ""},
  1802.            
  1803.         if key == "block" then                          -- get ingots as recipe for iron_block --> iron_ingot
  1804.             retValue = modType..mineralType.."_ingot"       -- "minecraft:iron_ingot"
  1805.         elseif key == "ingot" or key == "mineral" then
  1806.             if ingotType == "" then
  1807.                 retValue = modType..mineralType             -- "minecraft:coal"
  1808.             else
  1809.                 retValue = modType..mineralType.."_ingot"   -- "minecraft:iron_ingot"
  1810.             end
  1811.         elseif key == "copperType" then                 -- ["stairs"] = {"copperType", "", "", "copperType", "copperType", "", "copperType", "copperType", "copperType"}
  1812.             local modifier1 = ""
  1813.             if item:find("waxed") ~= nil then
  1814.                 modifier1 = "waxed_"
  1815.             end
  1816.            
  1817.             local modifier2 = ""
  1818.             if item:find("exposed") ~= nil then
  1819.                 modifier2 = "exposed_"
  1820.             elseif item:find("weathered") ~= nil then
  1821.                 modifier2 = "weathered_"
  1822.             elseif item:find("oxidized") ~= nil then
  1823.                 modifier2 = "oxidized_"
  1824.             end
  1825.            
  1826.             local modifier3 = ""
  1827.             if item:find("cut") ~= nil then
  1828.                 modifier3 = "cut_"
  1829.             end
  1830.             if words[#words] == "copper" then
  1831.                 retValue = modType..modifier1..modifier2.."copper_block"
  1832.             else
  1833.                 retValue = modType..modifier1..modifier2..modifier3.."copper"
  1834.             end
  1835.         end
  1836.     elseif category == "colored" then
  1837.         -- itemType = wool, carpet, coloured, directColoured, glass_pane, shulker_box, candle, banner, bed
  1838.         -- keys = wool, item, dye, glass, planks
  1839.         if key == "item" then
  1840.             if item:find("glass") ~= nil then
  1841.                 retValue = "minecraft:glass"
  1842.             elseif item:find("shulker") ~= nil then
  1843.                 retValue = "minecraft:shulker_box"
  1844.             elseif item:find("bed") ~= nil then
  1845.                 retValue = "minecraft:white_bed"
  1846.             end
  1847.             if itemColour ~= "" then
  1848.                 retValue = modType.."white_"..words[#words]     -- eg wool, carpet
  1849.             end
  1850.         elseif key == "glass" then
  1851.             retValue = "minecraft:glass"
  1852.         elseif key == "wool" then
  1853.             retValue = "minecraft:"..itemColour.."_wool"
  1854.         elseif key == "planks" then
  1855.             retValue = "minecraft:*_planks"
  1856.         elseif key == "dye" then
  1857.             if itemColour ~= "" then
  1858.                 retValue = modType..itemColour.."_dye"
  1859.             end
  1860.         else
  1861.             if itemType == "carpet" then
  1862.                 retValue = modType..itemColour.."_wool"
  1863.             elseif itemType == "glass_pane" then
  1864.                 retValue = modType..itemColour.."_glass"
  1865.             elseif itemType == "banner" then
  1866.                 retValue = modType..itemColour.."_wool"
  1867.             elseif itemType == "coloured" or itemType =="directColoured" then
  1868.                 --"Wool, Shulker Box, Bed, Candle
  1869.                 if key == "item" then
  1870.                     retValue = item
  1871.                 elseif key == "dye" then
  1872.                     retValue = modType..itemColour.."_dye"
  1873.                 end
  1874.             end
  1875.         end
  1876.     elseif category == "natural" then
  1877.         retValue = key
  1878.     elseif category == "functional" then
  1879.         -- keys = planks, log, stone, sherd, stone_slab, wood_slab, wool, dye, item
  1880.         if key == "planks" or key == "log" then
  1881.             if woodType == "" then
  1882.                 retValue = modType.."oak_"..key
  1883.             else
  1884.                 retValue = modType..woodType.."_"..key
  1885.             end
  1886.         elseif key == "stone" then
  1887.             if stoneType == "" then
  1888.                 retValue = modType.."cobblestone"
  1889.             else
  1890.                 retValue = modType..stoneType
  1891.             end
  1892.         elseif key == "wood_slab" then
  1893.             if woodType == "" then
  1894.                 retValue = modType.."oak_slab"
  1895.             else
  1896.                 retValue = modType..woodType.."_slab"
  1897.             end
  1898.         elseif key == "slab" then
  1899.             if stoneType == "" then
  1900.                 retValue = modType.."cobblestone_slab"
  1901.             else
  1902.                 retValue = modType..stoneType.."_slab"
  1903.             end
  1904.         elseif key == "wool" or key == "dye" then
  1905.             -- eg wool --> minecraft:red_wool to create minecraft:red_bed from planks and wool
  1906.             if itemColour == "" then
  1907.                 retValue = modType.."white_"..key
  1908.             else
  1909.                 retValue = modType..itemColour.."_"..key
  1910.             end
  1911.         elseif key == "item" then
  1912.             -- item used eg minecraft:red_candle, recipe = ["directColoured"] = {"", "", "", "item", "dye", "", "", "", ""}
  1913.             -- item = "minecraft:candle"   
  1914.             retValue = modType..words[2]                    -- eg minecraft:candle
  1915.         elseif key == "sherd" then
  1916.             retValue = "minecraft:pottery_sherd"                    -- any type of sherd can be mixed, or bricks
  1917.         end
  1918.     elseif category == "tools" then
  1919.         -- keys = planks, boat
  1920.         if key == "boat" or key == "planks" then
  1921.             retValue = "minecraft:*_"..key              -- eg "minecraft:oak_boat
  1922.         end
  1923.     elseif category == "combat" then
  1924.         if key == "planks" then
  1925.             retValue = "minecraft:*_planks"
  1926.         elseif key == "potion" then
  1927.             retValue = "minecraft:potion"
  1928.         end
  1929.     elseif category == "food" then
  1930.         if key == "flower" then
  1931.             retValue = "minecraft:dandelion"
  1932.         else
  1933.             retValue = key
  1934.         end
  1935.     elseif category == "ingredients" then
  1936.         -- keys = planks
  1937.         retValue = "minecraft:*_planks"
  1938.     elseif category == "computercraft" then
  1939.         retValue = key
  1940.         if key == "stone" then
  1941.             retValue = "minecraft:stone"
  1942.         elseif key == "turtle" then
  1943.             retValue = "computercraft:turtle"
  1944.         end
  1945.     elseif category == "morered" then
  1946.         retValue = key
  1947.     end
  1948.     _G.Log:saveToLog("\t\titem = "..item..", key = "..key..", full name = "..retValue)
  1949.     return retValue
  1950. end
  1951.  
  1952. function U.getIngredientList(value)
  1953.     -- project is an object of the Project class OR an item name eg "minecraft:oak_stairs"
  1954.     -- project = item name when called from Craft:S:onListClick(listbox), where quantity is unknown
  1955.     local itemsRequired = nil
  1956.     local project = nil
  1957.     local recipe = nil
  1958.     local craftRecipe = nil
  1959.     local quantity = 0
  1960.     local craftItem = ""
  1961.     local ratio = 0
  1962.    
  1963.     local lib = {}
  1964.    
  1965.     function lib.getItemDetails(currentCraftRecipe, recipe, craftItem)
  1966.         local required = nil
  1967.         for i = 1, #currentCraftRecipe - 1 do                                               -- iterate all items in {"planks", "", "", "planks", "planks", "", "planks", "planks", "planks"}
  1968.             local itemName = currentCraftRecipe[i]                                          -- eg "planks"
  1969.             if itemName ~= "" then                                                  -- NOT an empty ingredient
  1970.                 if type(itemName) == "table" then
  1971.                     itemName = itemName[1]
  1972.                 end
  1973.                 --_G.Log:saveToLog("\t\tcall fullName recipe = "..textutils.serialise(recipe, {compact = true})..", craftItem = "..craftItem..", itemName-->key = "..tostring(itemName))
  1974.                 itemName = U.getIngredientFullName(recipe, craftItem, itemName)     -- eg item = planks ->  "minecraft:oak_planks"
  1975.                 --_G.Log:saveToLog("U.getIngredientList().U.getIngredientFullName("..i..") = "..item)
  1976.                 currentCraftRecipe[i] = itemName
  1977.                 if required == nil then
  1978.                     required = {}
  1979.                 end
  1980.                 if required[itemName] == nil then                   -- not already in the table
  1981.                     required[itemName] = 1                          -- new table entry with value of 1
  1982.                 else
  1983.                     required[itemName] = required[itemName] + 1 -- increment count
  1984.                 end
  1985.             end
  1986.         end
  1987.         return required
  1988.     end
  1989.    
  1990.     function lib.updateTable(tblItemsRequired)
  1991.         local temp = {} -- build a table of new values for each item name
  1992.         _G.Log:saveToLog("\tU.getIngredientList() calculating updated quantities")
  1993.         for item, count in pairs(tblItemsRequired) do
  1994.             _G.Log:saveToLog("\t\titem: "..item..", count: "..count)
  1995.             local a, b, itemRatio = U.getCraftDetails(item)         -- eg 4 for planks
  1996.             _G.Log:saveToLog("\t\titem: "..item..", count: "..count..", itemRatio = "..itemRatio)
  1997.             --local required = count * quantity
  1998.             --self.quantity = math.ceil(self.quantity / self.ratio) * self.ratio
  1999.             local required =  math.ceil(count / itemRatio) * itemRatio
  2000.             _G.Log:saveToLog("\t\trequired = "..required)
  2001.             temp[item] = required
  2002.         end
  2003.         -- write altered values back to itemsRequired
  2004.         for item, count in pairs(temp) do
  2005.             tblItemsRequired[item] = count
  2006.         end
  2007.         return tblItemsRequired
  2008.     end
  2009.    
  2010.     if type(value) == "table" then                                  -- called from Project object
  2011.         _G.Log:saveToLog("\tU.getIngredientList(project):")
  2012.         project = value
  2013.         craftItem = project:getItem()                               -- eg "minecraft:oak_stairs"
  2014.         recipe, craftRecipe, ratio = U.getCraftDetails(craftItem)
  2015.         project:setRecipe(recipe)
  2016.         project:setCraftingRecipe(craftRecipe)
  2017.        
  2018.         --recipe = project:getRecipe()                              -- eg {"wood", "stairs"}
  2019.         --craftRecipe = project:getCraftingRecipe()                 -- eg {"planks", "", "", "planks", "planks", "", "planks", "planks", "planks", "4"}
  2020.         quantity = project:getQuantity()                            -- eg 4, 128
  2021.        
  2022.         --ratio = project:getRatio()                                    -- eg 4 for stairs
  2023.         --itemsRequired = project.itemsRequired                     -- direct link to object internal values
  2024.         --recipe, craftRecipe, ratio = U.getCraftDetails(value)
  2025.     else                                                            -- called directly from Craft.lua to provide info only eg "minecraft:oak_stairs"
  2026.         _G.Log:saveToLog("\tU.getIngredientList(value: "..value.."):")
  2027.         --recipe = U.findRecipe("name", value)                      -- eg {"wood", "stairs"} or {{"colored","bed"},{"colored", "directColoured"}}
  2028.         --craftRecipe = U.getCraftingRecipe(recipe)                 -- eg {"planks", "", "", "planks", "planks", "", "planks", "planks", "planks", "4"}
  2029.         quantity = 1                                                -- not set so use 1 as default
  2030.         craftItem = value                                           -- eg "minecraft:oak_stairs"
  2031.         --ratio = U.getCraftRatio(craftItem)
  2032.         recipe, craftRecipe, ratio = U.getCraftDetails(value)
  2033.     end
  2034.    
  2035.     if itemsRequired == nil then
  2036.         itemsRequired = {}
  2037.     end
  2038.    
  2039.     if craftItem:find("_block") ~= nil and craftItem:find("purpur") == nil and craftItem:find("quartz") == nil then
  2040.         -- minecraft:redstone_block can be crafted to and from minecraft:redstone
  2041.         -- minecraft:iron_block can be crafted to and from minecraft:iron_ingot
  2042.         -- do NOT iterate ingredients beyond level 1
  2043.        
  2044.         -- eg craftRecipe for block of iron = {"mineral", "mineral", "mineral", "mineral", "mineral", "mineral", "mineral", "mineral", "mineral", "1"},    
  2045.         _G.Log:saveToLog("\tU.getIngredientList.block = "..craftItem..", recipe = "..textutils.serialise(recipe, {compact = true})..", craftRecipe = "..textutils.serialise(craftRecipe, {compact = true}))
  2046.         local ingredient = U.getIngredientFullName(recipe, craftItem, craftRecipe[1])   -- eg ({"minerals","block"}, minecraft:iron_block, "mineral"
  2047.         itemsRequired[ingredient] = 9 * quantity
  2048.         -- can have multiple choices eg to make a chest, any plaks will work
  2049.         _G.Log:saveToLog("\tcraftItem: "..craftItem..", quantity: "..quantity..", ratio: "..tostring(ratio))
  2050.     else
  2051.         --if recipe ~= nil then _G.Log:saveToLog("\t\trecipe = "..textutils.serialise(recipe, {compact = true})) end
  2052.         --if craftRecipe ~= nil then _G.Log:saveToLog("\t\tcraftRecipe = "..textutils.serialise(craftRecipe, {compact = true})) end
  2053.         _G.Log:saveToLog("\t\tquantity: "..quantity..", craftItem: "..craftItem..", ratio: "..tostring(ratio))
  2054.         -- craftRecipe can consist of 2 or more sub-tables {{"", "", "", "wool", "wool", "wool", "planks", "planks", "planks", "1"}, {"", "", "", "item", "dye", "", "", "", "", "1"}}
  2055.         if craftRecipe ~= nil then
  2056.             if type(craftRecipe[1]) == "table" then -- 2 recipes involved
  2057.                 --for _, currentCraftRecipe in pairs(craftRecipe)
  2058.                 for i = 1, #craftRecipe do
  2059.                     local required = lib.getItemDetails(craftRecipe[i], recipe[i], craftItem)
  2060.                     if required ~= nil then
  2061.                         lib.updateTable(required)
  2062.                         table.insert(itemsRequired, required)
  2063.                     end
  2064.                 end
  2065.             else
  2066.                 local required = lib.getItemDetails(craftRecipe, recipe, craftItem)
  2067.                 if required ~= nil then
  2068.                     lib.updateTable(required)
  2069.                     table.insert(itemsRequired, required)
  2070.                 end
  2071.             end
  2072.             _G.Log:saveToLog("\tU.getIngredientList().itemsRequired = "..textutils.serialise(itemsRequired, {compact = true}))
  2073.         end
  2074.     end
  2075.    
  2076.     --if project == nil then
  2077.         _G.Log:saveToLog("\tU.getIngredientList().return (itemsRequired) = "..textutils.serialise(itemsRequired, {compact = true}))
  2078.         return itemsRequired, ratio -- eg nil if no crafting recipe
  2079.                             -- eg oak stairs: {["minecraft:oak_planks"] = 6}
  2080.                             -- eg redstone repeater: {["minecraft:stone"] = 3, ["minecraft:redstone_torch"] = 2, ["minecraft:redstone"] = 1}
  2081.     --else
  2082.         --_G.Log:saveToLog("\tU.getIngredientList().updating project (itemsRequired) = "..textutils.serialise(itemsRequired, {compact = true}))
  2083.         --project:setItemsRequired(itemsRequired)
  2084.         --project:setRatio(ratio)
  2085.     --end
  2086. end
  2087.  
  2088. function U.getInput(withTimer, interval)
  2089.     --will return on both mouse and key events unless mouseOnly
  2090.     withTimer = withTimer or false
  2091.     interval = interval or 3
  2092.    
  2093.     local timer_id
  2094.     while true do
  2095.         if withTimer then
  2096.             timer_id = os.startTimer(interval)
  2097.             --_G.Log:saveToLog("U.getInput() timer_id "..timer_id)
  2098.         end
  2099.         U.data = {os.pullEvent()}
  2100.         --_G.Log:saveToLog("U.getInput() data = "..textutils.serialize(U.data, {compact = true}))
  2101.         if U.data[1]:find("mouse") ~= nil or U.data[1] == "monitor_touch" then
  2102.             if withTimer then
  2103.                 os.cancelTimer(timer_id)
  2104.             end
  2105.             return U.data   -- eg {event = "mouse_click", button = 1, x = 4, y = 4)
  2106.         elseif U.data[1] == "key" then
  2107.             local key = keys.getName(U.data[2])
  2108.             if key == "enter" or key == "backspace" then
  2109.                 if withTimer then
  2110.                     os.cancelTimer(timer_id)
  2111.                 end
  2112.                 return U.data -- eg {event = "key", key = 8 (backspace) or 13 (return), nil, nil
  2113.             end
  2114.         elseif U.data[1] == "char" then -- returns all characters including symbols and shift
  2115.             if withTimer then
  2116.                 os.cancelTimer(timer_id)
  2117.             end
  2118.             return U.data
  2119.         elseif U.data[1] == "timer" and withTimer then
  2120.             return nil
  2121.         end
  2122.     end
  2123. end
  2124.  
  2125. function U.getInventoryContents(inventory, withSlots)
  2126.     --  an inventory object with a .list() property eg barrel, chest
  2127.     withSlots = withSlots or false
  2128.     local list = nil
  2129.     local contents = nil
  2130.     if inventory == nil then return end
  2131.     if type(inventory) == "string" then
  2132.         list = peripheral.call(inventory, "list")
  2133.     else
  2134.         list = inventory.list()
  2135.     end
  2136.     for slot, item in pairs(list) do
  2137.         if contents == nil then
  2138.             contents = {}
  2139.         end
  2140.         local name = item.name
  2141.         if name:find("minecraft:") ~= nil then
  2142.             name = name:sub(11)
  2143.         end
  2144.         -- setup strSlot to 2 characters eg 1 = " 1", 11 = "11"
  2145.         local strSlot = tostring(slot)
  2146.         if slot <10 then
  2147.             strSlot = " "..strSlot
  2148.         end
  2149.         local strCount = tostring(item.count)
  2150.         if item.count < 10 then
  2151.             strCount = " ".. strCount
  2152.         end
  2153.         if withSlots then
  2154.             table.insert(contents, strSlot.." "..strCount.." "..name)
  2155.         else
  2156.             table.insert(contents, strCount.." "..name)
  2157.         end
  2158.         --    slot count name
  2159.         -- eg " 1  4 oak_slab"
  2160.         -- eg "12 64 cobblestone_stairs"
  2161.     end
  2162.    
  2163.     return contents
  2164. end
  2165.  
  2166. function U.getInventoryObject(inventoryName)
  2167.     -- return the wrapped inventory eg local barrel = U.getInventoryObject(inputBarrelName)
  2168.     return peripheral.wrap(inventoryName)
  2169. end
  2170.  
  2171. function U.getItemCategory(from, key)
  2172.     for c = 1, #categories do
  2173.         for _, v in ipairs(Items[categories[c]]) do
  2174.             if from == "displayName" then
  2175.                 if v.displayName:find(key) ~= nil then
  2176.                     return categories[c]
  2177.                 end
  2178.             elseif from == "name" then
  2179.                 if key:find("computercraft") ~= nil then
  2180.                     return "computercraft"
  2181.                 elseif  key:find("morered") ~= nil then
  2182.                     return "morered"
  2183.                 else
  2184.                     if v.name == key then
  2185.                         return categories[c]
  2186.                     end
  2187.                 end
  2188.             end
  2189.         end
  2190.     end
  2191.     return nil
  2192. end
  2193.  
  2194. function U.getItemStock(item)
  2195.     -- return quantity of selected item in all storage locations
  2196.     if item:find(":") == nil then                       -- displayName used instead
  2197.         item, _ = U.searchItem("displayName", item)     -- convert to full item name
  2198.     end
  2199.     local inventories = U.findItemInInventory("chest", item, true)  -- find any chests containing this item  eg {"minecraft:chest_38"}
  2200.     if inventories == nil then                          -- item not in storage
  2201.         inventories = U.findItemInInventory("barrel", item, true)   -- find any chests containing this item  eg {"minecraft:chest_38"}
  2202.         if inventories == nil then
  2203.             return 0
  2204.         end
  2205.     end
  2206.     return  U.getStock(item, inventories)
  2207. end
  2208.  
  2209. function U.getRecipeData(from, key)
  2210.     for c = 1, #categories do
  2211.         for _, v in ipairs(Items[categories[c]]) do
  2212.             if from == "displayName" then
  2213.                 --_G.Log:saveToLog("checking: "..v.displayName)
  2214.                 if v.displayName == key then
  2215.                     if v.recipes ~= nil then
  2216.                         for k,recipe in pairs(v.recipes) do
  2217.                             recipe.priority = 1
  2218.                         end
  2219.                     end
  2220.                     return v.name, v.displayName, v.recipes
  2221.                 end
  2222.             elseif from == "name" then
  2223.                 if v.name == key then
  2224.                     if v.recipes ~= nil then
  2225.                         for k,recipe in pairs(v.recipes) do
  2226.                             recipe.priority = 1
  2227.                         end
  2228.                     end
  2229.                     return v.name, v.displayName, v.recipes
  2230.                 end
  2231.             end
  2232.         end
  2233.     end
  2234. end
  2235.  
  2236. function U.getSlotOneContains(inventoryName)
  2237.     local list = peripheral.call(inventoryName, "list")
  2238.     for slot, item in pairs(list) do
  2239.         if slot == 1 then
  2240.             return item.name
  2241.         end
  2242.     end
  2243.     return ""
  2244. end
  2245.  
  2246. function U.getStock(itemName, inventories)
  2247.     -- return quantity of selected item found in storage supplied eg {"minecraft:chest_38"}
  2248.     local count = 0
  2249.     --_G.Log:saveToLog("U.getStock("..itemName..", inventories = \n"..textutils.serialize(inventories))
  2250.     for _, inventory in pairs(inventories) do
  2251.         local chest = peripheral.wrap(inventory)
  2252.         for slot, item in pairs(chest.list()) do
  2253.             --_G.Log:saveToLog("U.getStock() slot = "..slot..", item = "..item.name..", count = "..item.count)
  2254.             if item.name == itemName then
  2255.                 count = count + item.count
  2256.             end
  2257.         end
  2258.     end
  2259.    
  2260.     return count
  2261. end
  2262.  
  2263. function U.populateAllItems()
  2264.     U.allItems = {}
  2265.     for category, tbl in pairs(Items) do
  2266.         for _, item in ipairs(tbl) do
  2267.             table.insert(U.allItems, item.displayName)
  2268.         end
  2269.     end
  2270.     table.sort(U.allItems)
  2271. end
  2272.  
  2273. function U.refineTemplate(category, itemToCraft, key)
  2274.     -- eg category = "wood", itemToCraft = "minecraft:oak_stairs", key = "planks"
  2275.     return U.getIngredientFullName(category, itemToCraft, key)
  2276. end
  2277.  
  2278. function U.emptySmelters()
  2279.     for i = 1, #U.smelters do
  2280.         U.smelters[i]:emptyOutput(true) -- check if empty and drop output items into output chest
  2281.     end
  2282. end
  2283.  
  2284. function U.refuelSmelter(smelter)
  2285.     local lib = {}
  2286.    
  2287.     function lib.getFuelValue(item)
  2288.         for k, v in pairs(fuelValues) do
  2289.             if item:find(k) ~= nil then -- eg find "planks" in "minecraft:bamboo_planks"
  2290.                 return v
  2291.             end
  2292.         end
  2293.         return 0
  2294.     end
  2295.    
  2296.     --[[
  2297.     lava 100, coal block 180, dried kelp 20, blaze rod 12, coal 8, log, planks, most wood items 1.5
  2298.     tools, sign, boat = 1, slab 0.75, sapling, bowl,stick, button, wool = 0.5; bamboo, scaffolding 0.25
  2299.     ]]
  2300.     --_G.Log:saveToLog("U.refuelSmelter() fuelType = "..smelter.fuelType)
  2301.     if smelter.fuelType == "minecraft:bucket" then
  2302.         --exchange empty bucket with full one
  2303.         if smelter.lavaFinished then
  2304.             -- find any bucket in storage to transfer empty bucket out
  2305.             local lavaAvailable = false
  2306.             local chestList = U.findItemInInventory("chest", "bucket", false)
  2307.             if #chestList > 0 then
  2308.                 U.transferItem(smelter.name, chestList[1], "minecraft:bucket", 1)
  2309.                 chestList = U.findItemInInventory("chest", "minecraft:lava_bucket", true)
  2310.                 if #chestList > 0 then
  2311.                     U.transferItem(chestList[1], smelter.name, "minecraft:lava_bucket", 1)
  2312.                     lavaAvailable = true
  2313.                 end
  2314.             else    -- no bucket or lava bucket available
  2315.                 U.transferItem(smelter.name, U.outputBarrelName, "minecraft:bucket", 1) -- move bucket to output, use any planks to refuel
  2316.             end
  2317.             if not lavaAvailable then
  2318.                 chestList = U.findItemInInventory("chest", "planks", false)
  2319.                 for i = 1, #chestList do
  2320.                     local remaining = U.transferItem(chestList[i], smelter.name, "planks", 64, 2)
  2321.                     if remaining == 0 then
  2322.                         return true
  2323.                     end
  2324.                 end
  2325.             end
  2326.         end
  2327.     else
  2328.         if smelter.fuelType == "" then                                  -- no fuel present in fuel slot
  2329.             smelter.fuelType = "minecraft:bamboo_planks"                -- use default
  2330.         end
  2331.         _G.Log:saveToLog("U.refuelSmelter() fuelType = "..smelter.fuelType)
  2332.         local chestList = U.findItemInInventory("chest", smelter.fuelType, true)    -- find where it is stored
  2333.         local fuelRequired = math.ceil(smelter:getQuantity() / lib.getFuelValue(smelter.fuelType))      -- dried kelp ratio is 20 smelt items per fuel item
  2334.         _G.Log:saveToLog("U.refuelSmelter() smelter:getQuantity() = "..smelter:getQuantity()..", lib.getFuelValue(smelter.fuelType) = "..lib.getFuelValue(smelter.fuelType))
  2335.         if chestList ~= nil then
  2336.             _G.Log:saveToLog("U.refuelSmelter() "..smelter.fuelType..", chestList = "..textutils.serialize(chestList, {compact = true}))
  2337.             for i = 1, #chestList do
  2338.                 _G.Log:saveToLog("U.refuelSmelter() U.transferItem("..chestList[i]..", "..smelter.name..", "..smelter.fuelType..", ".. fuelRequired..", 2")
  2339.                 local remaining = U.transferItem(chestList[i], smelter.name, smelter.fuelType, fuelRequired, 2)
  2340.                 if remaining == 0 then
  2341.                     return true
  2342.                 end
  2343.             end
  2344.         end
  2345.     end
  2346.    
  2347.     return false
  2348. end
  2349.  
  2350. function U.addToSmelter(storageList, sourceItem, index, count)
  2351.     -- add count items to smelter[index]
  2352.     --local smelter = U.smelters[index]
  2353.     local toInventoryName = U.smelters[index]:getName()
  2354.     _G.Log:saveToLog("U.addToSmelter(storageList = "..textutils.serialise(storageList, {compact = true})..", sourceItem = "..sourceItem)
  2355.     _G.Log:saveToLog("U.addToSmelter(index = "..index..". smelter name = "..toInventoryName..", count = "..count..")")
  2356.     U.smelters[index]:reset()
  2357.     for i = 1, #storageList do
  2358.         local remaining = U.transferItem(storageList[i], toInventoryName, sourceItem, count, 1)
  2359.         _G.Log:saveToLog("U.smelters["..index.."]:addQuantity("..count..")")
  2360.         U.smelters[index]:addQuantity(count)
  2361.         -- any storage chest emptied will automatically be removed fron U.locations
  2362.         if remaining <= 0 then
  2363.             break
  2364.         end
  2365.     end
  2366.     -- now check fuel
  2367.     --_G.Log:saveToLog("U.addToSmelter("..index..") fuelLevel = "..smelter.fuelLevel)
  2368.     --if smelter.fuelLevel < count then
  2369.         if not U.refuelSmelter(U.smelters[index]) then
  2370.             -- display warning about fuel levels
  2371.             --U.messageBox({"Refuelling smelters critical","No suitable fuel resources found","Click to craft planks"}, 4, "centre", "center", colors.red, colors.black)
  2372.             --U.window.setVisible(true)
  2373.             --U.windowAction = "Craft"
  2374.             --sleep(3)
  2375.         end
  2376.     --end
  2377. end
  2378.  
  2379. function U.sendToSmelters(storageList, sourceItem, quantity)
  2380.     -- calculate how many smelters needed for max efficiency
  2381.     -- blast furnace = ores, raw metals, armor and tools
  2382.     -- smoker food only including kelp. (except chorus fruit)
  2383.     -- quantity is total amount required
  2384.     local useSmoker = false
  2385.     local useBlast = false
  2386.     for k, v in ipairs(U.smoker) do
  2387.         if sourceItem:find(v) ~= nil then
  2388.             useSmoker = true
  2389.             break
  2390.         end
  2391.     end
  2392.     for k, v in ipairs(U.blastFurnace) do
  2393.         if sourceItem:find(v) ~= nil then
  2394.             useBlast= true
  2395.             break
  2396.         end
  2397.     end
  2398.     local smelters = {}
  2399.     for i = 1, #U.smelters do
  2400.         if not U.smelters[i].active then    -- smelter is free for use
  2401.             U.smelters[i]:setInUse(false)
  2402.             if U.smelters[i].type == "smoker" and useSmoker then
  2403.                 table.insert(smelters, i)
  2404.                 U.smelters[i]:setInUse(true)
  2405.             elseif U.smelters[i].type == "blast" and useBlast then
  2406.                 table.insert(smelters, i)
  2407.                 U.smelters[i]:setInUse(true)
  2408.             elseif U.smelters[i].type == "furnace" then
  2409.                 table.insert(smelters, i)
  2410.                 U.smelters[i]:setInUse(true)
  2411.             end
  2412.         end
  2413.     end
  2414.     -- smelters is a list of Smelter indices
  2415.     local portion = math.ceil(quantity / #smelters) -- eg 64 needed using 16 smelters = 4 per smelter
  2416.     _G.Log:saveToLog("U.sendToSmelters() smelters = "..textutils.serialise(smelters, {compact = true})..", portion = "..portion)
  2417.     local remaining = quantity
  2418.     for i = 1, #smelters do
  2419.         if remaining <= portion then    -- last smelter to be filled, portion could be smaller than remaining
  2420.             U.addToSmelter(storageList, sourceItem, smelters[i], remaining) -- note smelters[i] = index of relevant U.smelters
  2421.             break
  2422.         else
  2423.             U.addToSmelter(storageList, sourceItem, smelters[i], portion)
  2424.         end
  2425.         remaining = remaining - portion
  2426.     end
  2427.     U.smeltersActive  = true
  2428. end
  2429.  
  2430. function U.updateSmelters()
  2431.     -- called from main() when Smelt scene is active
  2432.     local anyActive = false
  2433.     for i = 1, #U.smelters do
  2434.         --_G.Log:saveToLog("U.updateSmelters() "..i)
  2435.         U.smelters[i]:update()  -- check each smelter fuel level, fuel type, input and output count
  2436.         if U.smelters[i].active then
  2437.             anyActive = true
  2438.         end
  2439.     end
  2440.     return anyActive
  2441. end
  2442.  
  2443. function U.removeFromStorageFile(storageType, itemName, chestName)
  2444.     local storage = U.barrelObjects
  2445.     if storageType == "chest" then
  2446.         storage = U.chestObjects
  2447.     end
  2448.    
  2449.     local data = storage[itemName]  -- eg {"minecraft:chest_67", "minecraft:chest_68"}
  2450.     for i, value in ipairs(data) do  
  2451.         if chestName == value then
  2452.             table.remove(data, i)
  2453.             break
  2454.         end
  2455.     end
  2456.     if #data == 0 then
  2457.         storage[itemName] = nil     -- remove from locations
  2458.         U.updateStorageFile()
  2459.     end
  2460. end
  2461.  
  2462. function U.searchItem(from, key, currentCategory, strict)
  2463.     -- eg "Oak Trapdoor" searches in Items only NOT in storage
  2464.     strict = strict or false
  2465.     if currentCategory == nil or currentCategory == "all" then
  2466.         for c = 1, #categories do
  2467.             for _, v in ipairs(Items[categories[c]]) do
  2468.                 if from == "displayName" then
  2469.                     if strict then
  2470.                         if v.displayName == key then
  2471.                             return v.name, v.displayName, v.recipes, v.smelt
  2472.                         end
  2473.                     else
  2474.                         if v.displayName:find(key) ~= nil then
  2475.                             return v.name, v.displayName, v.recipes, v.smelt
  2476.                         end
  2477.                     end
  2478.                 elseif from == "name" then
  2479.                     if strict then
  2480.                         if key:find(":") == nil then
  2481.                             key = "minecraft:"..key
  2482.                         end
  2483.                         if v.name == key then
  2484.                             return v.name, v.displayName, v.recipes, v.smelt
  2485.                         end
  2486.                     else
  2487.                         if v.name:find(key) ~= nil then
  2488.                             return v.name, v.displayName, v.recipes, v.smelt
  2489.                         end
  2490.                     end
  2491.                 end
  2492.             end
  2493.         end
  2494.     else
  2495.         for _, v in ipairs(Items[currentCategory]) do
  2496.             if from == "displayName" then
  2497.                 if strict then
  2498.                     if v.displayName == key then
  2499.                         return v.name, v.displayName, v.recipes, v.smelt
  2500.                     end
  2501.                 else
  2502.                     if v.displayName:find(key) ~= nil then
  2503.                         return v.name, v.displayName, v.recipes, v.smelt
  2504.                     end
  2505.                 end
  2506.             elseif from == "name" then
  2507.                 if strict then
  2508.                     if v.name == key then
  2509.                         return v.name, v.displayName, v.recipes, v.smelt
  2510.                     end
  2511.                 else
  2512.                     if v.name:find(key) ~= nil then
  2513.                         return v.name, v.displayName, v.recipes, v.smelt
  2514.                     end
  2515.                 end
  2516.             end
  2517.         end
  2518.     end
  2519.     return "", "", nil, nil
  2520. end
  2521.  
  2522. function U.searchAllItems(key)
  2523.     -- eg "log" returns {"minecraft:xxx_log", etc} in Items only NOT in storage
  2524.     -- exclude 'wooden' from search for 'wood'
  2525.     local matches = {}
  2526.     for c = 1, #categories do
  2527.         for _, v in ipairs(Items[categories[c]]) do
  2528.             if v.name:find(key) ~= nil then
  2529.                 if key == "wood" then
  2530.                     if v.name:find("wooden") == nil then
  2531.                         table.insert(matches, v.name)
  2532.                     end
  2533.                 elseif key == "iron" then
  2534.                     if U.tableContains(nuggetItems, v.name) then
  2535.                         table.insert(matches, v.name)
  2536.                     end
  2537.                 else
  2538.                     table.insert(matches, v.name)
  2539.                 end
  2540.             end
  2541.         end
  2542.     end
  2543.     return matches
  2544. end
  2545.  
  2546. function U.sendToOutput(currentItemLocations, itemName, required)
  2547.     -- eg 2 chests {{ "minecraft:chest_17" = 128,}  {"minecraft:chest_19" = 26}}
  2548.     local count = 0
  2549.     for _, chestName in ipairs(currentItemLocations) do            
  2550.         _G.Log:saveToLog("U.sendToOutput() - chestName = "..chestName)
  2551.         local chest = peripheral.wrap(chestName)                            -- v = "minecraft:chest_17"
  2552.         for slot, item in pairs(chest.list()) do
  2553.             _G.Log:saveToLog("U.sendToOutput() - slot ="..slot..", item.name = "..item.name..", item.count = "..item.count)
  2554.             if item.name == itemName then
  2555.                 if item.count >= required then                              -- v.quantity = 128 >= required
  2556.                     _G.Log:saveToLog("U.sendToOutput() - chest.pushItems("..U.outputBarrelName..", "..required..")")
  2557.                     chest.pushItems(U.outputBarrelName, slot, required)         -- remove exact requirements and break
  2558.                     count = required
  2559.                     break
  2560.                 else
  2561.                     _G.Log:saveToLog("U.sendToOutput() - chest.pushItems("..U.outputBarrelName..", "..item.count..")")
  2562.                     chest.pushItems(U.outputBarrelName, slot, item.count)       -- remove entire barrel contents of item
  2563.                     required = required - item.count                        -- reduce required then re-enter loop with next chest
  2564.                     count = count + item.count
  2565.                 end
  2566.             end
  2567.         end
  2568.     end
  2569.     return count
  2570. end
  2571. -- UI UTILITIES
  2572. function U.getItemColour(item)
  2573.     if item:find("light_gray") ~= nil then
  2574.         return "light_gray"
  2575.     end
  2576.     if item:find("light_blue") ~= nil then
  2577.         return "light_blue"
  2578.     end
  2579.     for _,v in ipairs(itemColours) do
  2580.         if item:find(v) ~= nil and item:find("redstone") == nil then            --  colour in item has matched. ignore redstone
  2581.             --_G.Log:saveToLog("\tU.getItemColour(item).return = "..v)
  2582.             return v
  2583.         end
  2584.     end
  2585.     --_G.Log:saveToLog("\tU.getItemColour(item).return = string.empty")
  2586.     return ""
  2587. end
  2588.  
  2589. function U.messageBox(message, size, alignH, alignV, fg, bg, buttons)
  2590.     --[[
  2591.     message  string
  2592.     size 1 to 5 int OR Vector2()
  2593.     alignH  left, centre, right
  2594.     alignV  top, centre, bottom
  2595.     buttons table {"ok", "yes", "no", "cancel"}
  2596.     ]]
  2597.     -- 5  sizes: 1,2,3,4,5
  2598.     size = size or Vector2(8, 3)
  2599.     alignH = alignH or "centre"
  2600.     alignV = alignV or "centre"
  2601.     fg = fg or colors.white
  2602.     bg = bg or colors.black
  2603.     if type(size) == "number" then
  2604.         size = Vector2(math.floor(size * 3 * 2.7), size * 3)    -- 8x3, 16x6, 24x9, 32x12, 40x15
  2605.     end
  2606.  
  2607.     local pos = Vector2(0, 0)
  2608.     if alignH:find("cent") ~= nil then
  2609.         pos.x = math.floor((51 - size.x) / 2)
  2610.     elseif alignH == "right" then
  2611.         pos.x = 51 - size.x
  2612.     end
  2613.    
  2614.     if alignV:find("cent") ~= nil then
  2615.         pos.y = math.floor((19 - size.y) / 2)
  2616.     elseif alignV == "right" then
  2617.         pos.y = 19 - size.y
  2618.     end
  2619.    
  2620.     if type(message) == "string" then
  2621.         message = {message}
  2622.     end
  2623.    
  2624.     U.window = window.create(term.current(), pos.x, pos.y, size.x, size.y, false)
  2625.     U.window.setBackgroundColour(bg)
  2626.     U.window.clear()
  2627.     U.window.setBackgroundColour(colors.white)
  2628.     U.window.setCursorPos(1,1)
  2629.     U.window.write((" "):rep(size.x))
  2630.    
  2631.     for row = 2, size.y - 1 do
  2632.         U.window.setCursorPos(1, row)
  2633.         U.window.write(" ")
  2634.         U.window.setCursorPos(size.x, row)
  2635.         U.window.write(" ")
  2636.     end
  2637.     U.window.setCursorPos(1,size.y)
  2638.     U.window.write((" "):rep(size.x))
  2639.    
  2640.     U.window.setBackgroundColour(bg)
  2641.     U.window.setTextColour(fg)
  2642.     local offset = math.floor((size.y - #message) / 2)
  2643.     for i = 1, #message do
  2644.         U.window.setCursorPos(2, offset + i)
  2645.         U.window.write(U.padCentre(message[i], size.x - 2, " "))
  2646.     end
  2647. end
  2648.  
  2649. function U.mouseOver(mx, my, rx, ry, rw, rh)
  2650.     --[[ Is mouse in a 20 x 3 button placed on column 5 row 2?
  2651.     +------------------------------>
  2652.     |                                row 1
  2653.     |    ....................        row 2 20 spaces in background colour
  2654.     |    .   Button Text    .        row 3 20 characters including spaces in text colour on backgound colour
  2655.     |    ....................        row 4 20 spaces in background colour
  2656.     |                                row 5
  2657.     V
  2658.          56789111111111122222222223
  2659.               012345678901234567890
  2660.              
  2661.     mx = col 5                  - mouse is on left column of button
  2662.     my = row 2                  - mouse is on top row of button
  2663.     rx = button left col eg 5   - mouse is valid
  2664.     ry = button top row eg 2    - mouse is valid
  2665.     rw = width in columns eg 20 -
  2666.     rh = height in rows eg 3
  2667.     check the following:
  2668.     1. is mouse column < button start col OR > button end col?
  2669.     2. is mouse row < button top row OR > button bottom row?
  2670.     3. If either 1 or 2 is true then the mouse is NOT in the button
  2671.     example mouse x = 4, mouse y = 24
  2672.     ]]
  2673.     --[[
  2674.     long version to explain logic
  2675.     example mx = 4, my = 24 bottom left of button: rx = 5, ry = 2, rw = 20 rh = 3
  2676.     if mx < rx or mx >= rx  + rw then  -- rx + rw = 5 + 20 = 25--> mx = 24 so NOT >= 25
  2677.         return false
  2678.     end
  2679.     if my < ry or my >= ry  + rh then -- ry + rh = 2 + 3 = 5 --> my = 4 so NOT >= 5
  2680.         return false
  2681.     end
  2682.     return true
  2683.     ]]
  2684.     if tonumber(mx) == nil or tonumber(my) == nil then return false end
  2685.     return  not (mx < rx or mx >= rx + rw or my < ry or my >= ry + rh) -- short version
  2686. end
  2687.  
  2688. function U.onScrBarClick(scrollbar, clickUp, clickDown, listPos, listbox)
  2689.     --[[_G.Log:writeTraceTable("\tS:onScrBarClick(scrollbar, clickUp, clickDown, listPos)",
  2690.                             {
  2691.                             "clickUp = "..tostring(clickUp)..", clickDown = "..tostring(clickDown)..", listPos = "..tostring(listPos),
  2692.                             "scrollbar.name = " ..scrollbar.name,
  2693.                             ".isVertical = "    ..tostring(scrollbar.isVertical),
  2694.                             ".groovePos: x = "  ..scrollbar.groovePos.x..", y = "..scrollbar.groovePos.y,
  2695.                             ".grooveSize: w = " ..scrollbar.grooveSize.x..", h = "..scrollbar.grooveSize.y,
  2696.                             ".handlePos: x = "  ..scrollbar.handlePos.x..", y = "..scrollbar.handlePos.y,
  2697.                             ".handleSize: w = " ..scrollbar.handleSize.x..", h = "..scrollbar.handleSize.y
  2698.                             })]]
  2699.                            
  2700.     local sb = scrollbar
  2701.  
  2702.     if clickUp then         -- clicked on top or left of scrollbar
  2703.         if sb.isVertical then
  2704.             listbox.listIndex = listbox.listIndex - 1
  2705.             if listbox.listIndex < 1 then
  2706.                 listbox.listIndex = 1
  2707.             end
  2708.             --_G.Log:saveToLog("S:onScrBarClick(clickUp, listbox.listIndex = "..listbox.listIndex)
  2709.         else
  2710.             listbox.columnIndex = listbox.columnIndex - 1
  2711.             if listbox.columnIndex < 1 then
  2712.                 listbox.columnIndex = 1
  2713.             end
  2714.             --_G.Log:saveToLog("S:onScrBarClick(clickUp, listbox.columnIndex = "..listbox.columnIndex)
  2715.         end
  2716.     elseif clickDown then   -- clicked on bottom or right of scrollbar
  2717.         -- move handle down / right by 1
  2718.         if sb.isVertical then
  2719.             local maxIndex = 1
  2720.             if listbox.dataHeight > listbox.size.y then
  2721.                 maxIndex = listbox.dataHeight - listbox.size.y
  2722.             end
  2723.             listbox.listIndex = listbox.listIndex + 1
  2724.             if listbox.listIndex > maxIndex then
  2725.                 listbox.listIndex = maxIndex
  2726.             end
  2727.             --_G.Log:saveToLog("S:onScrBarClick(clickDown, listbox.listIndex = "..listbox.listIndex)
  2728.         else
  2729.             local maxColIndex = 1
  2730.             if listbox.dataWidth > listbox.size.x then
  2731.                 maxColIndex = listbox.dataWidth - listbox.size.x
  2732.             end
  2733.             listbox.columnIndex = listbox.columnIndex + 1
  2734.             if listbox.columnIndex > maxColIndex then
  2735.                 listbox.columnIndex = maxColIndex
  2736.             end
  2737.             --_G.Log:saveToLog("S:onScrBarClick(clickDown, listbox.columnIndex = "..listbox.columnIndex)
  2738.         end
  2739.     else
  2740.         -- clicked some position on groove, but not inside handle
  2741.         -- calculate position in list in proportion using listPos eg 0.5
  2742.         if sb.isVertical then
  2743.             listbox.listIndex = math.ceil(listbox.dataHeight * listPos)
  2744.             --_G.Log:saveToLog("S:onScrBarClick(clickDown, listbox.listIndex = "..listbox.listIndex)
  2745.         else
  2746.             listbox.columnIndex = math.ceil(listbox.dataWidth * listPos)
  2747.             --_G.Log:saveToLog("S:onScrBarClick(clickDown, listbox.columnIndex = "..listbox.columnIndex)
  2748.         end
  2749.     end
  2750.     sb:setHandle(listbox)
  2751. end
  2752.  
  2753. function U.onScrBarDrag(scrollbar, listbox)
  2754.     local sb = scrollbar
  2755.     local mouseStartPos = sb.mouseDragStart -- Vector2
  2756.     local mouseEndPos = sb.mouseDragEnd
  2757.     local moveX = mouseEndPos.x - mouseStartPos.x   -- negative = move left eg 10 to 5: 5-10 = -5
  2758.     local moveY = mouseEndPos.y - mouseStartPos.y   -- negative = move up   eg 10 to 5: 5-10 = -5
  2759.     local maxIndex = math.max(listbox.dataHeight - listbox.size.y, listbox.size.y)
  2760.     local maxColIndex = math.max(listbox.dataWidth - listbox.size.x, listbox.size.x)
  2761.  
  2762.     --_G.Log:saveToLog("S:onScrBarDrag("..scrollbar.name..") - moveX = "..moveX..", moveY = "..moveY)
  2763.     if sb.isVertical then
  2764.         listbox.listIndex = listbox.listIndex + moveY
  2765.         if listbox.listIndex < 1 then
  2766.             listbox.listIndex = 1
  2767.         elseif listbox.listIndex > maxIndex then
  2768.             listbox.listIndex = maxIndex
  2769.         end
  2770.     else
  2771.         listbox.columnIndex = listbox.columnIndex + moveX
  2772.         if listbox.columnIndex < 1 then
  2773.             listbox.columnIndex = 1
  2774.         elseif listbox.columnIndex > maxColIndex then
  2775.             listbox.columnIndex = maxColIndex
  2776.         end
  2777.     end
  2778.     sb:setHandle(listbox)
  2779. end
  2780.  
  2781. return U
Add Comment
Please, Sign In to add comment