Inksaver

tk.lua toolkit (requires libs):2025/03/09

Apr 20th, 2020 (edited)
4,917
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
Lua 529.25 KB | None | 0 0
  1. version = 20250309.0830
  2.  
  3. local tkVersion = version -- otherwise over-written by clsTurtle when loaded
  4. --[[
  5.     **********Toolkit v2**********
  6.     https://pastebin.com/UFvjc1bw
  7.     Last edited: see version YYYYMMDD.HHMM
  8.     if NOT online:
  9.         Make sure you create a folder 'lib' and place menu.lua and clsTurtle.lua into it
  10.     else
  11.         lib folder will be created and files obtained automatically!
  12.     end
  13. ]]
  14.  
  15. args = {...} -- eg "farm", "tree"
  16.  
  17. local menu, T
  18. --[[
  19. Computercraft started with mc version 1.7.10 and went to 1.8.9
  20. ccTweaked started around mc 1.12.2 and currently at 1.21.1
  21. mc 1.18 onwards has new blocks and bedrock at -64, so needs to be taken into account.
  22. _HOST = The ComputerCraft and Minecraft version of the current computer environment.
  23. For example, ComputerCraft 1.93.0 (Minecraft 1.15.2).
  24. ]]
  25. local dbug = false -- debug is reserved word
  26. local bedrock = 0
  27. local netherBedrock = 0
  28. local ceiling = 255
  29. local deletesWater = false
  30. local brick = "minecraft:nether_brick" -- pre 1.16+ name
  31. local ccMajorVersion = _HOST:sub(15, _HOST:find("Minecraft") - 2) --eg ComputerCraft 1.93.0 (Minecraft 1.15.2)
  32. local ccMinorVersion = 0
  33. local barrelItems = nil -- list of items and the barrels where they are usually found
  34. local chestItems = nil  -- list of items and the chests where they are usually found
  35. --[[
  36. Netherite level stand on 14
  37. Chunk borders F3+G or:
  38. math.floor(x / 16) * 16 add 16 for each border. same for z
  39. ]]
  40. function string:split(sSeparator, nMax, bRegexp, noEmpty)
  41.     --[[return a table split with sSeparator. noEmpty removes empty elements
  42.         use: tblSplit = SplitTest:split('~',[nil], [nil], false) or tblSplit = string.split(SplitTest, '~')]]  
  43.     assert(sSeparator ~= '','separator must not be empty string')
  44.     assert(nMax == nil or nMax >= 1, 'nMax must be >= 1 and not nil')
  45.     if noEmpty == nil then noEmpty = true end
  46.  
  47.     local aRecord = {}
  48.     local newRecord = {}
  49.     -- self refers to the 'string' being split
  50.     if self:len() > 0 then
  51.         local bPlain = not bRegexp
  52.         nMax = nMax or -1
  53.  
  54.         local nField, nStart = 1, 1
  55.         local nFirst,nLast = self:find(sSeparator, nStart, bPlain)
  56.         while nFirst and nMax ~= 0 do
  57.             aRecord[nField] = self:sub(nStart, nFirst-1)
  58.             nField = nField+1
  59.             nStart = nLast+1
  60.             nFirst,nLast = self:find(sSeparator, nStart, bPlain)
  61.             nMax = nMax-1
  62.         end
  63.         aRecord[nField] = self:sub(nStart)
  64.        
  65.         if noEmpty then --split on newline preserves empty values
  66.             for i = 1, #aRecord do
  67.                 if aRecord[i] ~= "" then
  68.                     table.insert(newRecord, aRecord[i])
  69.                 end
  70.             end
  71.         else
  72.             newRecord = aRecord
  73.         end
  74.     end
  75.    
  76.     return newRecord
  77. end
  78.  
  79. if tonumber(ccMajorVersion) == nil then -- 1.93.0 NAN
  80.     --ccMajorVersion = ccMajorVersion:sub(1, ccMajorVersion:find(".", 3, true) - 1)
  81.     local parts = ccMajorVersion:split(".")
  82.     ccMajorVersion = parts[1]
  83.     if parts[2] ~= nil then
  84.         ccMajorVersion = ccMajorVersion.."."..parts[2]
  85.     end
  86.     ccMajorVersion = tonumber(ccMajorVersion)
  87.     if parts[3] ~= nil then
  88.         ccMinorVersion = tonumber(parts[3])
  89.     end
  90. end
  91.  
  92. local mcMajorVersion = _HOST:sub(_HOST:find("Minecraft") + 10, _HOST:find("\)") - 1) -- eg 1.18 or 1.20 -> 1.18, 1.20
  93. local mcMinorVersion = 0
  94. if tonumber(mcMajorVersion) == nil then -- 1.18.3 NAN
  95.     --mcMajorVersion = tonumber(_HOST:sub(_HOST:find("Minecraft") + 10, _HOST:find("\)") - 3)) -- eg 1.19.4 -> 1.19
  96.     local parts = mcMajorVersion:split(".")
  97.     mcMajorVersion = parts[1]
  98.     if parts[2] ~= nil then
  99.         mcMajorVersion = mcMajorVersion.."."..parts[2]
  100.     end
  101.     mcMajorVersion = tonumber(mcMajorVersion)
  102.     if parts[3] ~= nil then
  103.         mcMinorVersion = tonumber(parts[3])
  104.     end
  105. end
  106.  
  107. if mcMajorVersion < 1.7  and mcMajorVersion >= 1.18 then -- 1.12 to 1.??
  108.     bedrock = -64
  109.     ceiling = 319
  110. end
  111. if mcMajorVersion < 1.7  and mcMajorVersion >= 1.16 then -- 1.12 to 1.??
  112.     brick = "minecraft:nether_bricks"
  113. end
  114. if mcMajorVersion < 1.7  and mcMajorVersion <= 1.12 then --- turtle in source deletes it. 1.7.10 to 1.12
  115.     deletesWater = true
  116. end
  117.  
  118. local utils = {}
  119. local network = {}
  120.  
  121. function network.addToStorageList(storageType, itemKey, storageName, writeToFile)
  122.     -- itemKey is a table, so is passed byRef. No need to return a value
  123.     -- eg itemKey: [ "minecraft:dark_oak_sapling" ] = {"minecraft:barrel_94", "minecraft:barrel_96"}
  124.     -- storageName = "minecraft:barrel_99"
  125.     local itemTable = {}
  126.     if storageType == "chest" then
  127.         itemTable = chestItems[itemKey] -- eg [ "minecraft:dark_oak_sapling" ] = {"minecraft:barrel_94", "minecraft:barrel_96"}
  128.     else
  129.         itemTable = barrelItems[itemKey]
  130.     end
  131.     if itemTable == nil then                -- key does not match. This item not previously stored
  132.         if storageType == "chest" then
  133.             chestItems[itemKey] = {storageName} -- eg U.chestItems[minecraft:diorite] = {chest_105}
  134.         else
  135.             barrelItems[itemKey] = {storageName}
  136.         end
  137.     else
  138.         for _, storage in ipairs(itemTable) do  -- is "minecraft:barrel_99" already in the list?
  139.             if storage == storageName then
  140.                 return  -- exit function
  141.             end
  142.         end
  143.         -- not found so add to table. return not required as funcion is ended
  144.         table.insert(itemTable, storageName)    -- add to table eg table[ "minecraft:dark_oak_sapling" ] = {"minecraft:barrel_94", "minecraft:barrel_96",, "minecraft:barrel_99"}
  145.     end
  146.     if writeToFile then
  147.         network.updateList(storageType)
  148.     end
  149. end
  150.  
  151. function network.attachModem()
  152.     -- modem cannot be "attached". Has to be player right-click!
  153.     -- place on  top or next to a modem and ask player to right-click
  154.     T:clear()
  155.     menu.colourPrint("Please right-click on the modem(s) I am next to or above/below"..
  156.                     "\nThe centre square should be lit red.\n"..
  157.                     "If embedded use narrow gap at side\n", colors.red)
  158.     local event, side = os.pullEvent("peripheral")
  159.     for timer = 5, 0, -1 do
  160.         -- text, fg, bg, width, isInput, cr
  161.         menu.colourWrite("Thanks. continuing in ".. timer.." seconds", colors.lime, colors.black, 0, false, true)
  162.         sleep(1)
  163.     end
  164. end
  165.  
  166. function network.checkInventory(inventory, itemName, itemsPerSlot, matchPart)
  167.     --[[
  168.     Find an item already in an inventory
  169.     inventory = The wrapped inventory or it's name
  170.     itemName = The name of the item to find.
  171.     return no of items already present, and storage space for additional
  172.     ]]
  173.     itemsPerSlot = itemsPerSlot or 64
  174.     matchPart = matchPart or ""
  175.     local contents = nil
  176.     local numSlots = 0
  177.     if type(inventory) == "string" then
  178.         contents = peripheral.call(inventory, "list")
  179.         numSlots = peripheral.call(inventory, "size")
  180.         --assert(contents ~= nil, "Nil contents from inventory "..tostring(inventory))
  181.         T:saveToLog("network.checkInventory('"..inventory.."', itemName = "..itemName..", itemsPerSlot = "..itemsPerSlot..", matchPart = "..tostring(matchPart))
  182.     else
  183.         contents = inventory.list()
  184.         numSlots = inventory.size()
  185.     end
  186.     local inStock = 0
  187.     local partMatch = false
  188.     local canStore = 0
  189.     if contents ~= nil then
  190.         T:saveToLog("#slots in use = "..#contents)
  191.         canStore = (numSlots - #contents) * itemsPerSlot    -- capacity of empty slots
  192.         for slot, item in pairs(contents) do
  193.             if item.name == itemName then
  194.                 inStock = inStock  + item.count
  195.                 canStore = canStore + itemsPerSlot - item.count
  196.             else
  197.                 if matchPart ~= "" then -- eg check for "cobblestone" or "slab"
  198.                     if item.name:find(matchPart) ~= nil then
  199.                         partMatch = true
  200.                     end
  201.                 end
  202.             end
  203.         end
  204.     end
  205.     T:saveToLog("network.checkInventory return inStock = "..inStock..", canStore = "..canStore..", partMatch = "..tostring(partMatch))
  206.     return inStock, canStore, partMatch -- eg 1, 3647, false if contains only 1 matching item in otherwise empty chest
  207. end
  208.  
  209. function network.emptyInventory(R, barrels, chests, sticksAsFuel)
  210.     --[[network.emptyInventory(R, {"sapling", "propagule", "dirt", "crafting"}, {"all"}, true)]]
  211.     if not T:isEmpty() then
  212.         if sticksAsFuel then
  213.             utils.useSticksAsFuel()
  214.         end
  215.         for _, item in ipairs(barrels) do
  216.             network.sendItemToNetworkStorage(R, "barrel", item, 0)
  217.         end
  218.         for _, item in ipairs(chests) do
  219.             network.sendItemToNetworkStorage(R, "chest", item, 0)
  220.         end
  221.     end
  222. end
  223.  
  224. function network.loadStorageLists()
  225.     local lib = {}
  226.    
  227.     function lib.createList(storageType)
  228.         local storage, turtleName, storageNames = network.wrapModem(R, storageType) -- list of barrel/chest peripherals, name of turtle, names of barrels / chests
  229.         local total = 0
  230.         local locations = {}
  231.         for _, store in pairs(storageNames) do                  -- eg "minecraft:chest_1"
  232.             T:clear()              
  233.             T:saveToLog("Checking "..store.. " contents")       -- eg "checking minecraft:chest_1 contents"
  234.             local contents = peripheral.call(store, "list")     -- list of items / slots for this chest
  235.             for slot, item in pairs(contents) do                -- for each item check if this storage is listed
  236.                 if locations[item.name] == nil then             -- this item not yet found
  237.                     locations[item.name] = {store}              -- add to table eg locations["minecraft:cobblestone"] = {"minecraft:chest_1"}
  238.                 else                                            -- already has at least 1 location
  239.                     network.addToStorageList(storageType, locations[item.name], store, false)
  240.                 end
  241.             end
  242.             total = total + 1
  243.         end
  244.         T:saveToLog("found ".. total.." "..storageType)
  245.         local output = textutils.serialize(locations)       -- serialise to json ready to write to file
  246.         local fileName = storageType.."Items.lua"           -- barrelList.lua or chestList.lua
  247.         local outputHandle = fs.open(fileName, "w")         -- open file
  248.         outputHandle.writeLine("return")                    -- start file with return
  249.         outputHandle.write(output)                          -- add serialised table
  250.         outputHandle.close()                                -- close file
  251.        
  252.         return locations
  253.     end
  254.    
  255.     function lib.listNeedUpdating(storageType, list)
  256.         -- see if named chest/barrel in list is found in fresh peripheral.find
  257.         -- turtle may have moved to a different network
  258.         -- list = eg [ "minecraft:stick" ] = {"minecraft:barrel_91","minecraft:barrel_114"}
  259.         local rawStorage = {peripheral.find(storageType)}
  260.         if not utils.isTableEmpty(rawStorage) then      -- chests / barrels are attached see if they match
  261.             for item, storeList in pairs(list) do       -- existing storage names can be found here
  262.                 local found = false
  263.                 for key, value in ipairs(rawStorage) do -- look in the fresh list of storage names to see if there are missing entries
  264.                     local name = peripheral.getName(value)             
  265.                     for _, storageName in ipairs(storeList) do  -- check each storage name found
  266.                         if storageName == name then     -- recorded name matches, check next one
  267.                             found = true
  268.                             break
  269.                         end
  270.                     end
  271.                     if found then break end
  272.                 end
  273.                 if not found then
  274.                     -- no match in existing list for this storage: list needs updating
  275.                     return true-- list does not match
  276.                 end
  277.             end
  278.         end
  279.         return false    -- list is ok
  280.     end
  281.    
  282.     local _, turtleName, storageNames = network.wrapModem(R, "chest")   -- _, name of turtle, list of names of storage
  283.     if turtleName == "Modem not found" then return turtleName end
  284.     local redo = false
  285.     if barrelItems == nil then  -- not yet loaded
  286.         T:saveToLog("barrelItems == nil")
  287.         if fs.exists("barrelItems.lua") then
  288.             T:saveToLog("require('barrelItems')")
  289.             barrelItems = require("barrelItems")    -- global variable
  290.             redo = lib.listNeedUpdating("barrel", barrelItems)
  291.         else
  292.             T:saveToLog("lib.createList('barrel'")
  293.             barrelItems = lib.createList("barrel")
  294.         end
  295.     end
  296.     if chestItems == nil then   -- not yet loaded
  297.         T:saveToLog("chestItems == nil")
  298.         if fs.exists("chestItems.lua") then
  299.             T:saveToLog("require('chestItems')")
  300.             chestItems = require("chestItems")  -- global variable
  301.             redo = lib.listNeedUpdating("chest", barrelItems)
  302.         else
  303.             T:saveToLog("lib.createList('chest'")
  304.             chestItems = lib.createList("chest")
  305.         end
  306.     end
  307.     if redo then
  308.         barrelItems = lib.createList("barrel")
  309.         chestItems = lib.createList("chest")
  310.     end
  311. end
  312.  
  313. function network.findEmptySlot(list, size)
  314.     --[[ adapted from https://github.com/cc-tweaked/CC-Tweaked/discussions/1552
  315.     Find the first empty slot in a chest.
  316.     list = list of items in the chest/barrel/dropper
  317.     size = The size of the inventory
  318.     return integer? slot The slot number of the first empty slot, or nil if none are empty.
  319.     ]]
  320.     for slot = 1, size do
  321.         if not list[slot] then
  322.             return slot
  323.         end
  324.     end
  325.    
  326.     return nil
  327. end
  328.  
  329. function network.findItemCountInInventory(list, itemName, exactMatch)
  330.     --[[
  331.     Find an item in an inventory
  332.     list = The list of items in the inventory
  333.     itemName = The name of the item to find.
  334.     return integer? The slot number of the item, or nil if not found.
  335.     ]]
  336.     exactMatch = exactMatch or false
  337.     if type(list) == "string" then
  338.         list = peripheral.call(list, "list")
  339.     end
  340.     local retValue = nil
  341.     local count = 0
  342.     for slot, item in pairs(list) do
  343.         local found = false
  344.         if exactMatch then
  345.             if item.name == itemName then found = true end
  346.         else
  347.             if item.name:find(itemName) ~= nil then found = true end
  348.         end
  349.         if found then
  350.             if retValue == nil then
  351.                 retValue = {}
  352.             end
  353.             table.insert(retValue,{slot, item.count}) -- eg {1, 64}
  354.             count = count + item.count
  355.         end
  356.     end
  357.     return count, retValue -- either nil or eg {{1, 64},{4, 22}}
  358. end
  359.  
  360. function network.findItemInInventory(inventory, itemName, exactMatch)
  361.     --[[ adapted from https://github.com/cc-tweaked/CC-Tweaked/discussions/1552
  362.     Find an item in an inventory
  363.     inventory = name of inventory or wrapped peripheral
  364.     itemName = The name of the item to find.
  365.     return integer?, integer? The slot number and count of the item, or nil if not found.
  366.     ]]
  367.     exactMatch = exactMatch or false
  368.     if type(inventory) == "string" then
  369.         T:saveToLog("network.findItemInInventory("..inventory..", "..itemName..", exactMatch = "..tostring(exactMatch)..")", true)
  370.         contents = peripheral.call(inventory, "list")
  371.     else    -- should be supplied with .list() already
  372.         T:saveToLog("network.findItemInInventory(<inventory>, "..itemName..", exactMatch = "..tostring(exactMatch)..")", true)
  373.         contents = inventory
  374.     end
  375.     --T:saveToLog("contents = "..table.concat(contents, ", "))
  376.     --utils.writeTraceTable("contents = ", contents)
  377.     if contents ~= nil then
  378.         for slot, item in pairs(contents) do
  379.             --T:saveToLog("item.name = "..item.name..", item.count = "..item.count)
  380.             if exactMatch then
  381.                 if item.name == itemName then
  382.                     T:saveToLog("Item found in "..slot..", quantity = "..item.count)
  383.                     return slot, item.count
  384.                 end
  385.             else
  386.                 if (item.name):find(itemName) ~= nil then
  387.                     T:saveToLog("Matching Item found in "..slot..", quantity = "..item.count)
  388.                     return slot, item.count
  389.                 end
  390.             end
  391.         end
  392.     end
  393.     T:saveToLog("Item not found")
  394.     return 0,0
  395. end
  396.  
  397. function network.getItemFromNetwork(storageType, itemRequired, countRequired, toTurtleSlot, ignoreStock)
  398.     local lib = {}
  399.    
  400.     function lib.getItem(storageTable, turtleName, itemRequired, countRequired, toTurtleSlot, sent)
  401.         local exit = false
  402.         for k, storageName in pairs(storageTable) do    -- eg {"minecraft:barrel_17", "minecraft:barrel_18"...}
  403.             local available, data = network.findItemCountInInventory(storageName, itemRequired, false)      -- either nil or eg {{1, 64},{4, 22}}
  404.             T:saveToLog("network.findItemCountInInventory("..storageName..", "..itemRequired..", false")
  405.             if data ~= nil then
  406.                 for i = 1, #data do
  407.                     local request = countRequired
  408.                     if countRequired > 64 then
  409.                         request = 64
  410.                     end
  411.                     local received = network.sendItemsToTurtle(turtleName, storageName, data[i][1], request, toTurtleSlot)  -- request items, returns number sent
  412.                     if received == nil then received = 0 end
  413.                     sent = sent + received
  414.                     T:saveToLog("received = "..received..", request = "..request.." from "..storageName..", sent = "..sent)
  415.                     if sent >= countRequired then
  416.                         exit = true
  417.                         break
  418.                     end         -- job done, no need to check remaining slots
  419.                     countRequired = countRequired - sent            -- this line not reached if sent >= count
  420.                 end
  421.             end
  422.             --if sent >= countRequired then break end               -- no need to check other storage
  423.             --if countRequired <= 0 then break end
  424.             if exit then
  425.                 return sent, countRequired
  426.             end
  427.         end
  428.        
  429.         return sent, countRequired
  430.     end
  431.    
  432.     -- eg slot, count = network.getItemFromNetwork("barrel", "minecraft:crafting_table", 1)
  433.     -- storageType either "chest" or "barrel"
  434.     if countRequired == nil then return 0,0 end
  435.     --if toTurtleSlot not specified then nil = use any slot
  436.     ignoreStock = ignoreStock or false  -- return only no of items obtained from storage
  437.     T:saveToLog("network.getItemFromNetwork(storageType = "..storageType..", itemRequired = ".. itemRequired..
  438.                 ", countRequired = ".. countRequired..", toTurtleSlot = "..tostring(toTurtleSlot)..", ignoreStock = "..tostring(ignoreStock))
  439.     -- Must be next to a modem: MUST remove crafting table if modem on that side. Other tools ok
  440.     local sent = 0
  441.     local turtleSlot, turtleCount = T:getItemSlot(itemRequired) -- check local stock
  442.     if not ignoreStock then -- take account of existing items and reduce count accordingly
  443.         countRequired = countRequired - turtleCount
  444.     end
  445.     local savedItems = nil
  446.     if storageType == "barrel" then
  447.         savedItems = barrelItems
  448.     elseif storageType == "chest" then
  449.         savedItems = chestItems
  450.     end
  451.     local storage, turtleName, storageNames = network.wrapModem(R, storageType) -- list of chest/barrel peripherals, name of turtle, list of storage names
  452.     --if turtleName == "Modem not found" then return 0, nil, nil, turtleName end
  453.     if countRequired > 0 then                       -- not enough in stock, or ignore current stock
  454.         -- check if item in storageLists
  455.         local testStores = nil
  456.         if savedItems[itemRequired] ~= nil then -- only works with full item names
  457.             T:saveToLog("savedItems key in list: "..textutils.serialise(savedItems[itemRequired], {compact = true}))
  458.             testStores = savedItems[itemRequired]
  459.         else
  460.             for key, value in pairs(savedItems) do
  461.                 if key:find(itemRequired)~= nil then
  462.                     T:saveToLog("savedItems found in list: "..textutils.serialise(value, {compact = true}))
  463.                     testStores = value
  464.                     break
  465.                 end
  466.             end
  467.         end
  468.         if testStores == nil then   -- no match in storage lists
  469.             T:saveToLog("Unable to find recorded storage, using all "..storageType.."s")
  470.             sent, countRequired = lib.getItem(storageNames, turtleName, itemRequired, countRequired, toTurtleSlot, sent)
  471.         else -- match found, list of storage availble -- eg {"minecraft:barrel_17", "minecraft:barrel_18"...}
  472.             T:saveToLog("Using recorded list alias 'testStores'")
  473.             sent, countRequired = lib.getItem(testStores, turtleName, itemRequired, countRequired, toTurtleSlot, sent)
  474.         end
  475.     end
  476.         -- slotData.lastSlot, total, slotData -- integer, integer, table
  477.     local data = {}
  478.     turtleSlot, turtleCount, data = T:getItemSlot(itemRequired)
  479.     T:saveToLog("turtleSlot = "..turtleSlot..", turtleCount = "..turtleCount..", sent = "..sent) --..", data = "..textutils.serialise(data))
  480.     if ignoreStock then
  481.         return turtleSlot, sent -- 0 -> count
  482.     else
  483.         return turtleSlot, turtleCount  -- 0 -> count
  484.     end
  485. end
  486.  
  487. function network.getSlotContains(inventoryName, inSlot)
  488.     local list = peripheral.call(inventoryName, "list")
  489.     for slot, item in pairs(list) do
  490.         if inSlot == slot then
  491.             return item.name
  492.         end
  493.     end
  494.     return ""
  495. end
  496.  
  497. function network.moveItem(inventoryName, itemName, toSlot)
  498.     --[[ adapted from https://github.com/cc-tweaked/CC-Tweaked/discussions/1552
  499.     Move a specific item to specific slot eg 1, moving other items out of the way if needed.
  500.     inventoryName = The name of the chest/barrel/dropper to search.
  501.     itemName = The name of the item to find.
  502.     toSlot optional. default is slot 1
  503.     return boolean success Whether or not the item was successfully moved to toSlot (or already existed there)
  504.     ]]
  505.     toSlot = toSlot or 1
  506.     local list = peripheral.call(inventoryName, "list")
  507.     local size = peripheral.call(inventoryName, "size")
  508.     local slot = network.findItemInInventory(list, itemName)
  509.  
  510.   -- If the item didn't exist, or is already in the first slot, we're done.
  511.     if not slot then
  512.         T:saveToLog("network.moveItem(): Item not found")
  513.         return false
  514.     end
  515.     if slot == toSlot then
  516.         return true
  517.     end
  518.  
  519.     -- If an item is blocking the first slot (we already know it's not the one we want), we need to move it.
  520.     if list[toSlot] then
  521.         T:saveToLog("network.moveItem() Slot "..toSlot.." occupied, moving..")
  522.         local emptySlot = network.findEmptySlot(list, size)
  523.  
  524.         -- If there are no empty slots, we can't move the item.
  525.         if not emptySlot then
  526.             T:saveToLog("network.moveItem(): No empty slots")
  527.             return false
  528.         end
  529.  
  530.         -- Move the item to the first empty slot.
  531.        
  532.         if not network.moveItemStack(inventoryName, toSlot, emptySlot) then
  533.             T:saveToLog("network.moveItem(): Failed to move item to slot " .. emptySlot)
  534.             return false
  535.         end
  536.  
  537.         T:saveToLog("network.moveItem(): Moved item to slot " .. emptySlot)
  538.     end
  539.  
  540.     -- Move the item to slot 1.
  541.     if not network.moveItemStack(inventoryName, slot, toSlot) then
  542.         T:saveToLog("network.moveItem(): Failed to move item to slot "..toSlot)
  543.         return false
  544.     end
  545.  
  546.     T:saveToLog("network.moveItem(): Moved item to slot "..toSlot)
  547.     return true
  548. end
  549.  
  550. function network.moveItemsFromTurtle(turtleName, toInventoryName, fromTurtleSlot, quantity, toSlot)
  551.     --[[
  552.     Move quantity of an item from one inventory to another. Turtles MUST use attachedInventory.pullItems()
  553.     eg network.moveItemsFromTurtle(turtleName, chestName, turtleSlot, turtleCount, nil)
  554.     turtleName:         The name of the turtle (via getLocalName())
  555.     toInventoryName:    The name of the inventory to move items into.
  556.     fromTurtleSlot:     The slot to move from. must be pre-determined for the item required
  557.     quantity:           The amount to transfer (nil for full stack)
  558.     toSlot:             The slot to move to. (nil will use any available slot(s))
  559.     ]]
  560.     return peripheral.call(toInventoryName, "pullItems", turtleName, fromTurtleSlot, quantity, toSlot)
  561. end
  562.  
  563. function network.moveItemStack(inventoryName, fromSlot, toSlot)
  564.     --[[ adapted from https://github.com/cc-tweaked/CC-Tweaked/discussions/1552
  565.     Move an item from one slot to another in a given inventory.
  566.     inventoryName The name of the inventory to move items in.
  567.     fromSlot The slot to move from.
  568.     toSlot The slot to move to.
  569.     ]]
  570.     return peripheral.call(inventoryName, "pushItems", inventoryName, fromSlot, nil, toSlot)
  571. end
  572.  
  573. function network.sendItemToNetworkStorage(R, storageType, itemToSend, amountToSend)
  574.     -- used to remove items from turtle inventory
  575.     -- Must be next to a modem: MUST remove crafting table if modem on that side. Other tools ok
  576.     local lib = {}
  577.    
  578.     function lib.sendItem(savedItems, peripheralNames, turtleName, turtleSlot, item, slotCount, itemsPerSlot)
  579.         local storageToUse = ""
  580.         local storageList = lib.getStorageFromList(savedItems, item, slotCount, itemsPerSlot)   -- try from savedList
  581.         if storageList == nil then  -- no match found, but use first one found with network.wrapModem
  582.             T:saveToLog("No storage with matching items found, using first empty chest")
  583.             storageToUse = lib.findEmptyStorage(peripheralNames, item, itemsPerSlot, slotCount)
  584.         else
  585.             T:saveToLog("Storage with matching items found, checking capacity")
  586.             storageToUse  = lib.checkCapacity(storageList, item, slotCount, itemsPerSlot)
  587.             if storageToUse == "" then  -- no capacity in known storage list, so start a new one
  588.                 storageToUse = lib.findEmptyStorage(peripheralNames, item, itemsPerSlot, slotCount)
  589.             end
  590.         end
  591.         --network.moveItemsFromTurtle(turtleName, toInventoryName, fromTurtleSlot, quantity, toSlot)
  592.         T:saveToLog("network.moveItemsFromTurtle(turtleName = "..turtleName..", storageToUse = "..tostring(storageToUse)..", slot = "..tostring(turtleSlot)..", slotCount = "..tostring(slotCount)..")")
  593.         network.moveItemsFromTurtle(turtleName, storageToUse, turtleSlot, slotCount)
  594.     end
  595.    
  596.     function lib.findEmptyStorage(peripheralNames, itemName, itemsPerSlot, itemCount)
  597.         T:saveToLog("lib.findEmptyStorage("..textutils.serialise(peripheralNames,{compact = true})..", itemName = "..itemName.. ", itemsPerSlot = "..itemsPerSlot)
  598.         for store = 1, #peripheralNames do
  599.             inStock, canStore, partMatch = network.checkInventory(peripheralNames[store], itemName, itemsPerSlot, "")
  600.             if canStore > itemCount then
  601.                 return peripheralNames[store]
  602.             end
  603.         end
  604.         return nil
  605.     end
  606.    
  607.     function lib.getStorageFromList(savedItems, item, sendAmount, itemsPerSlot)
  608.         if savedItems[item] == nil then                             -- no record of this item stored
  609.             T:saveToLog("lib.getStorageFromList() "..item.." not found")
  610.             local parts = T:getNames(item)                              -- eg minecraft:jungle_planks = "minecraft", "jungle", "planks"
  611.             for part = #parts, 1, -1 do                                 -- iterate "planks", "jungle", "minecraft"
  612.                 local searchTerm = parts[part]
  613.                 if searchTerm ~= "minecraft" and searchTerm ~= "mysticalagriculture" then
  614.                     for itemName, storageList in pairs(savedItems) do   -- iterate items used as keys eg minecraft:jungle_log matches jungle
  615.                         if itemName:find(searchTerm) ~= nil then        -- partial match eg "sapling" found in "minecraft:oak_sapling"
  616.                             T:saveToLog("lib.getStorageFromList() matched "..searchTerm.." with "..itemName)
  617.                             return storageList                          -- eg {"minecraft:chest_22", "minecraft:chest_23"}
  618.                         end
  619.                     end
  620.                 end
  621.             end
  622.         else
  623.             T:saveToLog("lib.getStorageFromList() ["..item.."] found")
  624.             return savedItems[item] -- list of chests with this item available
  625.         end
  626.         return nil
  627.     end
  628.    
  629.     function lib.checkCapacity(storageList, item, sendAmount, itemsPerSlot)
  630.         -- find a chest/barrel with sufficient capacity from list of storage
  631.         for store = 1, #storageList do
  632.             local inStock, canStore, partMatch = network.checkInventory(storageList[store], item, itemsPerSlot, "")
  633.             if canStore > sendAmount then
  634.                 return storageList[store]
  635.             end
  636.         end
  637.         return ""
  638.     end
  639.    
  640.     amountToSend = amountToSend or 0                        -- 0 = remove all of this item
  641.     local totalSent = 0                                     -- track quantity sent
  642.     local minSend = 0                                       -- minimum amount to send
  643.     if amountToSend > 0 then minSend = amountToSend end     -- update minimum to send
  644.     local _, turtleName, peripheralNames = network.wrapModem(R, storageType)    -- _, name of turtle, list of names of storage
  645.    
  646.     T:saveToLog("network.sendItemToNetworkStorage(R, storageType = '"..storageType.."', itemToSend = '"..itemToSend.."', amountToSend = "..amountToSend..")")
  647.     T:saveToLog("network.wrapModem(R, storageType = "..storageType..", turtleName = "..turtleName..")")
  648.     local savedItems = nil
  649.     if storageType == "barrel" then
  650.         savedItems = barrelItems
  651.     elseif storageType == "chest" then
  652.         savedItems = chestItems
  653.     end
  654.    
  655.     if itemToSend == "all" then -- empty Turtle, so item names not relevant
  656.         T:saveToLog("itemToSend = all")
  657.         repeat
  658.             local item, turtleSlot, slotCount, itemsPerSlot = "", 0, 0, 64
  659.             for slot = 1, 16 do
  660.                 item, slotCount = T:getSlotContains(slot)
  661.                 if slotCount > 0 then
  662.                     turtleSlot = slot
  663.                     T:saveToLog("for slot = 1, 16 do: item = "..item..", slotCount = "..slotCount)
  664.                     itemsPerSlot = utils.getSlotCapacity(slot)  -- most items capacity is 64 per slot
  665.                     T:saveToLog("sending'"..item.."' from slot "..slot..", quantity = "..slotCount)
  666.                     break
  667.                 end
  668.             end
  669.             if turtleSlot > 0 then
  670.                 lib.sendItem(savedItems, peripheralNames, turtleName, turtleSlot, item, slotCount, itemsPerSlot)
  671.             end
  672.         until turtleSlot == 0
  673.         return 0    -- exit function
  674.     else
  675.         local storageToUse  = ""
  676.         repeat  -- until item no longer present in inventory or requested amount has been sent
  677.             local sourceSlot, total, data = T:getItemSlot(itemToSend)   -- which slot and how much of itemToSend is in turtle?
  678.             local slotCount = data.leastCount
  679.             T:saveToLog("T:getItemSlot('"..itemToSend.."' sourceSlot = "..sourceSlot..", total = "..total..")")
  680.             if sourceSlot == 0 then
  681.                 T:saveToLog(itemToSend.." not found in turtle inventory")
  682.                 return 0    -- exit function
  683.             else
  684.                 local itemsPerSlot = utils.getSlotCapacity(sourceSlot)  -- most items capacity is 64 per slot
  685.                 itemToSend = data.leastName                             -- full name of item with lowest itemCount
  686.                 T:saveToLog("network.sendItemToNetworkStorage(R, '"..itemToSend.."', sourceSlot = "..sourceSlot..", slotCount = "..slotCount) --..", data = "..textutils.serialise(data)..")")
  687.                 if sourceSlot > 0 then                                  -- item is present in turtle inventory
  688.                     local newStore = false
  689.                     local sent = 0
  690.                     local storageList = lib.getStorageFromList(savedItems, itemToSend, slotCount, itemsPerSlot)
  691.                     if storageList == nil then
  692.                         T:saveToLog("not found in any storage, finding empty")
  693.                         storageToUse = lib.findEmptyStorage(peripheralNames, itemToSend, itemsPerSlot, total)
  694.                         T:saveToLog("new storage: "..storageToUse.." is available")
  695.                         network.addToStorageList(storageType, itemToSend, storageToUse, true)
  696.                         sent = network.moveItemsFromTurtle(turtleName, storageToUse, sourceSlot, slotCount)
  697.                     else
  698.                         storageToUse  = lib.checkCapacity(storageList, itemToSend, slotCount, itemsPerSlot)
  699.                         if storageToUse == "" then  -- no capacity in known storage list, so start a new one
  700.                             storageToUse = lib.findEmptyStorage(peripheralNames, itemToSend, itemsPerSlot, total)
  701.                             newStore = true
  702.                         end
  703.                         T:saveToLog("sent = network.moveItemsFromTurtle(turtleName = "..turtleName..", storageToUse = "..storageToUse..", sourceSlot = "..sourceSlot..", slotCount = ".. slotCount)
  704.                         sent = network.moveItemsFromTurtle(turtleName, storageToUse, sourceSlot, slotCount)
  705.                     end
  706.                     totalSent = totalSent + sent
  707.                     if minSend > 0 and totalSent >= minSend then
  708.                         return totalSent
  709.                     end
  710.                     if amountToSend > 0 then    -- sending specified amount
  711.                         amountToSend = amountToSend - sent
  712.                     end
  713.                     if newStore then
  714.                         network.addToStorageList(storageType, itemToSend, storageName, true)
  715.                     end
  716.                 end
  717.             end
  718.         until sourceSlot == 0
  719.     end
  720.    
  721.     return totalSent
  722. end
  723.  
  724. function network.sendItemsToCrafter(crafterName, fromInventoryName, fromInventorySlot, quantity, toCrafterSlot)
  725.     --[[
  726.     fromInventoryName:  The name of the inventory to move items from.
  727.     fromInventorySlot:  The slot to move from. must be pre-determined for the item required
  728.     quantity:           The amount to transfer (nil for full stack)
  729.     toCrafterSlot:      The slot to move to. (nil will use any available slot(s))
  730.     ]]
  731.     return peripheral.call(fromInventoryName, "pushItems", crafterName, fromInventorySlot, quantity, toCrafterSlot)
  732. end
  733.  
  734. function network.sendItemsToTurtle(turtleName, fromInventoryName, fromInventorySlot, quantity, toTurtleSlot)
  735.     --network.sendItemsToTurtle(turtleName, storageName, storageSlot, count, toTurtleSlot)
  736.     --[[
  737.     Move quantity of an item from one inventory to another. Turtles MUST use attachedInventory.pushItems()
  738.     eg network.sendItemsToTurtle(turtleName, chestName, chestSlot, chestCount, 16) -- move to slot 16 so must be empty
  739.     fromInventoryName:  The name of the inventory to move items from.
  740.     fromInventorySlot:  The slot to move from. must be pre-determined for the item required
  741.     quantity:           The amount to transfer (nil for full stack)
  742.     toTurtleSlot:       The slot to move to. (nil will use any available slot(s))
  743.     ]]
  744.     return peripheral.call(fromInventoryName, "pushItems", turtleName, fromInventorySlot, quantity, toTurtleSlot)
  745. end
  746.  
  747. function network.transferItem(fromInventoryName, toInventoryName, itemName, quantity, toSlot)
  748.     --[[
  749.     Move a specific number of an item from one inventory to another
  750.     fromInventoryName:  The name of the chest/barrel/dropper to search.
  751.     toInventoryName:    The name of the receiving inventory (chest/barrel/dropper/smelter)
  752.     itemName:           The name of the item to find.
  753.     toSlot:             optional. nil picks any slot
  754.     return:             boolean success Whether or not the item was successfully moved to toSlot (or already existed there)
  755.     ]]
  756.     --T:saveToLog("network.transferItem(from: "..fromInventoryName..", to: "..toInventoryName..", itemName = "..itemName..", quantity = "..tostring(quantity)..", toSlot = "..tostring(toSlot))
  757.     local list = peripheral.call(fromInventoryName, "list")
  758.     local size = peripheral.call(fromInventoryName, "size")
  759.     --T:saveToLog("network.transferItem() size = "..size..", list = \n"..textutils.serialize(list))
  760.     local count, data = network.findItemCountInInventory(list, itemName)    -- either nil or eg {{1, 64},{4, 22}}
  761.     --T:saveToLog("network.transferItem() data = "..textutils.serialize(data, {compact = true}))
  762.     local remaining = quantity          -- eg 22 items needed
  763.  
  764.     if data == nil then -- Item not found
  765.         return quantity -- return amount requested = nothing sent
  766.     end
  767.    
  768.     local fromSlot = 0
  769.     local count = 64
  770.     local available = 0
  771.     for _, v in pairs(data) do  -- eg {1, 64},{2, 64}
  772.         if v[2] < count and v[2] >= quantity then
  773.             fromSlot = v[1]
  774.             count = v[2]
  775.         else
  776.             available = available + v[2]
  777.         end
  778.     end
  779.     if fromSlot > 0 then                        -- now have slot with min required quantity
  780.         T:saveToLog("network.transferItem() from: "..fromInventoryName..", to: "..toInventoryName..", fromSlot: "..fromSlot..", toSlot: "..tostring(toSlot)..", quantity: "..tostring(quantity))
  781.         network.transferItems(fromInventoryName, toInventoryName, fromSlot, toSlot, quantity)
  782.         return 0
  783.     else                                    -- available must be at least 1
  784.         for i = 1, #data do                 -- itreate all slots containg at least 1 item
  785.             fromSlot = data[i][1]           -- eg slot 1
  786.             local send = data[i][2]         -- eg 10 items
  787.             if remaining - send < 0 then    -- eg 22 - 10 = 12
  788.                 send = remaining
  789.                 remaining = 0
  790.             else
  791.                 remaining = remaining - send-- eg remaining = 22 - 10 = 12
  792.             end
  793.             network.transferItems(fromInventoryName, toInventoryName, fromSlot, toSlot, send)
  794.             if remaining <= 0 then          -- all required items transferred
  795.                 return 0
  796.             end
  797.         end
  798.     end
  799.    
  800.     return remaining                        -- return remaining items to be found from a  different inventory
  801. end
  802.  
  803. function network.transferItems(fromInventoryName, toInventoryName, fromSlot, toSlot, quantity)
  804.     --[[
  805.     Move quantity of an item from one inventory to another
  806.     fromInventoryName:  The name of the inventory to move items from.
  807.     toInventoryName:    The name of the inventory to move items into.
  808.     fromSlot:           The slot to move from. must be pre-determined for the item required
  809.     toSlot:             The slot to move to. (nil will use any available slot(s))
  810.     quantity:           The amount to transfer (nil for full stack)
  811.     ]]
  812.     T:saveToLog("network.transferItems(from: "..fromInventoryName..", to: "..toInventoryName..", fromSlot: "..fromSlot..", toSlot: "..tostring(toSlot)..", quantity: "..tostring(quantity)..")")
  813.     return peripheral.call(fromInventoryName, "pushItems", toInventoryName, fromSlot, quantity, toSlot)
  814. end
  815.  
  816. function network.transferItemToTurtle(availableStorage, availableStorageKeys, turtleName, crafterData)
  817.     -- network.transferItemToTurtle(<availableStorage>, data = {{1, 64},{4, 22}}, "crafter_01", <crafterData>)
  818.     -- availableStorage.minecraft:chest_114 = {count = 86, data = {{1, 64},{4, 22}},
  819.     -- availableStorage.minecraft:chest_115 = {count = 1024, data = {{1, 64},{2, 64},{3, 64}, ... }
  820.     -- crafterData = {{2,64}, {4,64}, {6,64}, {8,64}} 64 items in each of 4 slots in the crafter
  821.     -- glitch? in crafter inventory, cannot add sequentially to existing items.
  822.     -- send to turtle slot first, then transfer
  823.     T:saveToLog("network.transferItemToTurtle(availableStorage = "..textutils.serialise(availableStorage, {compact = true})..
  824.                 "\navailableStorageKeys = "..textutils.serialise(availableStorageKeys, {compact = true})..
  825.                 "\n"..turtleName..", crafterData = "..textutils.serialise(crafterData, {compact = true}))
  826.                
  827.     local total = 0
  828.     local numSlots = 0
  829.     local sent = 0
  830.     for _, v in ipairs(crafterData) do                              -- how many items required in total?
  831.         total = total + v[2]                                        -- how many slots does it go in
  832.         numSlots = numSlots + 1
  833.     end
  834.     for _, availableStorageKey in ipairs(availableStorageKeys) do   -- eg {minecraft:chest_114, minecraft:chest_115}
  835.         local storageName = availableStorageKey                     -- eg minecraft:chest_114
  836.         local object = availableStorage[storageName]                -- availableStorage.minecraft:chest_114 = {count = 90, data = {{14,64},{15,26}}
  837.         local storageData = object.data                             -- eg data = {{14,64},{15,26}}
  838.         local storageCount = object.count                           -- eg count = 90
  839.         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
  840.             local toCrafterSlot = crafterSlotData[1]                -- eg slot 2 in turtle
  841.             local amountToSend = crafterSlotData[2]                 -- eg place 22 items in slot 2
  842.             T:saveToLog("storageData = "..textutils.serialise(storageData, {compact = true}))
  843.             T:saveToLog("crafterSlotData = "..textutils.serialise(crafterSlotData, {compact = true}))
  844.             for i = 1, #storageData do                              -- {{14,64},{15,26}}                   
  845.                 local slotData = storageData[i]                     -- {14,64}
  846.                 local availableToSend = slotData[2]                 -- 64
  847.                 local fromStorageSlot = slotData[1]                 -- 14
  848.                 local confirmedSent = 0
  849.                
  850.                 T:saveToLog("i = "..i..", slotData = "..textutils.serialise(slotData, {compact = true}))
  851.                 if availableToSend >= amountToSend then
  852.                     T:saveToLog("availableToSend ("..availableToSend..") >= amountToSend: ("..amountToSend.."), current value of sent = "..sent)
  853.                     T:saveToLog("?confirmedSent = peripheral.call("..storageName..", 'pushItems', "..turtleName..
  854.                                 ", from slot "..fromStorageSlot..", amountToSend = "..
  855.                                 amountToSend..", to turtle slot "..toCrafterSlot)
  856.                     confirmedSent = peripheral.call(storageName, "pushItems", turtleName, fromStorageSlot, amountToSend, toCrafterSlot)
  857.                     sent = sent + confirmedSent
  858.                     T:saveToLog("verified confirmedSent = "..confirmedSent..", sent = "..sent)
  859.                     slotData[2] = slotData[2] - confirmedSent
  860.                     crafterSlotData[2] = 0
  861.                     T:saveToLog("slotData[2] = "..slotData[2]..", crafterSlotData[2] = "..crafterSlotData[2])
  862.                 else
  863.                     T:saveToLog("availableToSend ("..availableToSend..") < amountToSend: ("..amountToSend.."), current value of sent = "..sent)
  864.                     T:saveToLog("?confirmedSent = peripheral.call("..storageName..", 'pushItems', "..turtleName..
  865.                                 ", from slot "..fromStorageSlot..", availableToSend = "..
  866.                                 availableToSend..", to turtle slot "..toCrafterSlot)
  867.                     -- taking items from multiple storage slots requires loading into turtle first
  868.                     confirmedSent = peripheral.call(storageName, "pushItems", turtleName, fromStorageSlot, availableToSend, toCrafterSlot)
  869.                     sent = sent + confirmedSent
  870.                     T:saveToLog("verified confirmedSent = "..confirmedSent..", sent = "..sent)
  871.                     amountToSend = amountToSend - confirmedSent
  872.                     slotData[2] = slotData[2] - confirmedSent
  873.                     crafterSlotData[2] = amountToSend
  874.                     T:saveToLog("slotData[2] = "..slotData[2]..", crafterSlotData[2] = "..crafterSlotData[2])
  875.                 end
  876.                
  877.                 if crafterSlotData[2] == 0 then
  878.                     T:saveToLog("crafterSlotData[2]("..crafterSlotData[2]..") == 0: breaking\n")
  879.                     break   -- already sent correct amount
  880.                 end
  881.             end
  882.         end
  883.         if sent >= total then
  884.             T:saveToLog("sent("..sent..") >= total ("..total.."): breaking\n")
  885.             break
  886.         end
  887.     end
  888.  
  889.     return sent
  890. end
  891.  
  892. function network.pullItems(fromName, fromSlot, quantity, toName, toSlot)
  893.     quantity = quantity or 1
  894.     toSlot = toSlot or nil
  895.     if type(fromName) == "table" then           -- already a wrapped peripheral
  896.         fromName = peripheral.getName(fromName)
  897.     end
  898.     if type(toName) == "string" then
  899.         toName = peripheral.wrap(toName)
  900.     end
  901.     toName.pullItems(fromName, fromSlot, quantity, toSlot)
  902. end
  903.  
  904. function network.updateList(storageType)
  905.     local output = ""
  906.     if storageType == "barrel" then
  907.         output = textutils.serialize(barrelItems)       -- serialise to json ready to write to file
  908.     elseif storageType == "chest" then
  909.         output = textutils.serialize(chestItems)        -- serialise to json ready to write to file
  910.     end
  911.     local fileName = storageType.."Items.lua"           -- barrelList.lua or chestList.lua
  912.     local outputHandle = fs.open(fileName, "w")         -- open file
  913.     outputHandle.writeLine("return")                    -- start file with return
  914.     outputHandle.write(output)                          -- add serialised table
  915.     outputHandle.close()                                -- close file
  916. end
  917.  
  918. function network.wrapModem(R, storageType)
  919.     --[[To move turtle inventory items use the target peripheral:
  920.         local modem = peripheral.wrap("front")      -- wrap modem next to turtle (beware crafting table!)
  921.         local turtleName = modem.getNameLocal()     -- get name of the turtle
  922.         local barrel = peripheral.find("barrel")    -- find barrel name you want to receive goods
  923.         barrel.pushItems(turtleName, 1, 1)          -- push items FROM turtle to barrel  pushItems(toName, fromSlot , limit , toSlot)
  924.         barrel.pullItems(turtleName, fromSlot , limit , toSlot)
  925.     ]]
  926.     local modem = peripheral.find("modem")      -- find modem
  927.     if modem == nil then
  928.         return nil, "Modem not found"
  929.     end
  930.     local turtleName = modem.getNameLocal()     -- get name of the turtle
  931.     local rawStorage = nil                      -- peripheral find can return duplicate values
  932.     if storageType:find("barrel") ~= nil then
  933.         rawStorage = {peripheral.find("minecraft:barrel")}
  934.     elseif storageType:find("chest") ~= nil then
  935.         rawStorage = {peripheral.find("minecraft:chest")}
  936.     end
  937.     local storage = {}
  938.     local names = {}
  939.     for k, value in ipairs(rawStorage) do
  940.         local name = peripheral.getName(value)
  941.         if not utils.tableContains(names, name, true) then  -- use exact match as checking peripherals
  942.             table.insert(storage, value)
  943.             table.insert(names, name)
  944.         end
  945.     end
  946.     return storage, turtleName, names
  947. end
  948.  
  949. function utils.assessFarm(R)
  950.     -- sets R.networkFarm and R.mysticalAgriculture
  951.     menu.colourPrint("Assessing farm properties...", colors.yellow)
  952.     --local storage = true
  953.     for turns = 1, 4 do
  954.         T:turnRight(1)
  955.         local itemAhead = T:getBlockType("forward")
  956.         if itemAhead:find("modem") ~= nil then
  957.             if not R.networkFarm then   -- prevents repeated text display
  958.                 R.networkFarm = true
  959.                 menu.colourPrint("Network storage in use.", colors.magenta)
  960.             end
  961.         end
  962.         if itemAhead:find("mysticalagriculture") ~= nil then
  963.             R.mysticalAgriculture = true
  964.             menu.colourPrint("Mystical Agriculture deployed.", colors.lime)
  965.         end
  966.     end
  967.     sleep(1.5)
  968.    
  969.     return R
  970. end
  971.  
  972. function utils.assessTreeFarm(R)
  973.     local lib = {}
  974.        
  975.     function lib.getSaplingType(R, blockType)
  976.         if blockType == "minecraft:mangrove_log" then
  977.             R.logType = "mangrove"
  978.             R.useBlockType ="mangrove_propagule"
  979.         else
  980.             local parts = T:getNames(blockType)
  981.             if parts[2] == "dark" then
  982.                 R.logType = "dark_oak"
  983.                 parts[2] = "dark_oak"
  984.             else
  985.                 R.logType = parts[2]                -- eg "oak"
  986.             end
  987.            
  988.             R.useBlockType = parts[2].."_sapling"   -- eg "oak_sapling"
  989.         end
  990.         utils.goBack(1)
  991.         turtle.turnLeft()
  992.        
  993.         return R
  994.     end
  995.    
  996.     local blockType = T:getBlockType("down")
  997.     if blockType:find("modem") ~= nil then
  998.         R.networkFarm = true
  999.     else
  1000.         if blockType:find("polished") == nil then
  1001.             R.message = "Not on modem or polished. Check position"
  1002.             return R
  1003.         end
  1004.     end
  1005.     if R.networkFarm then
  1006.         T:forward(1)
  1007.         blockType = T:getBlockType("down")
  1008.         T:saveToLog("T:forward(1): blockType = "..blockType)
  1009.         if blockType:find("log") ~= nil then
  1010.             R = lib.getSaplingType(R, blockType)
  1011.         elseif blockType:find("barrel") ~= nil then
  1012.             T:go("R2F2")
  1013.             blockType = T:getBlockType("down")
  1014.             if blockType:find("log") ~= nil then
  1015.                 R = lib.getSaplingType(R, blockType)
  1016.             end
  1017.         else    -- if in correct position water is 2 blocks below
  1018.             if turtle.down() then
  1019.                 blockType = T:getBlockType("down")
  1020.                 if blockType:find("water") == nil then
  1021.                     R.message = "No modem below. Check position"
  1022.                     return R
  1023.                 else
  1024.                     turtle.up()
  1025.                     utils.goBack(1)
  1026.                     T:go("R1F1")
  1027.                     blockType = T:getBlockType("down")
  1028.                     if blockType:find("log") ~= nil then
  1029.                         R = lib.getSaplingType(R, blockType)
  1030.                     else
  1031.                         R.message = "Unable to determine location. Check position"
  1032.                         return R
  1033.                     end
  1034.                 end
  1035.             else
  1036.                 R.message = "Unable to determine location. Check position"
  1037.                 return R
  1038.             end
  1039.         end
  1040.     end
  1041.    
  1042.     return R    -- sets R.networkFarm and R.useBlockType to sapling type
  1043. end
  1044.  
  1045. function utils.calculateDimensions(R)
  1046.     if R.data == "" then -- not being called from other functions
  1047.         if R.width == 0 then -- user chose auto settings
  1048.             R.length = utils.calculateDistance(R.length) -- still facing forward
  1049.             print("Calculated Length = "..R.length)
  1050.             T:turnRight(1) -- ready to check width
  1051.             R.width = utils.calculateDistance(R.width) -- now facing right
  1052.             print("Calculated width = "..R.width)
  1053.             T:go("R2F"..R.width - 1 .."L1".. "F"..R.length - 1 .."R2") -- back at start. width/length confirmed
  1054.         end
  1055.     end
  1056.     if R.height == 0 then -- use auto settings based on water detection
  1057.         R.height = 64
  1058.     end
  1059.     return R
  1060. end
  1061.  
  1062. function utils.calculateDistance(estDistamcMajorVersionnce)
  1063.     --[[
  1064.     measure length
  1065.     movement stops when either solid block in front or solid below
  1066.     called from utils.calculateDimensions
  1067.     ]]
  1068.     local moves = 1
  1069.     local nonSolidBelow = utils.clearVegetation("down")
  1070.     if estDistance == 0 then
  1071.         while nonSolidBelow and turtle.forward() do -- while able to move and air/water below
  1072.             moves = moves + 1
  1073.             nonSolidBelow = utils.clearVegetation("down")
  1074.         end
  1075.     else
  1076.         while moves < estDistance * 2 do -- loop can only run for max of double estimated distance
  1077.             if turtle.forward() then -- able to move forward
  1078.                 moves = moves + 1
  1079.             else
  1080.                 break
  1081.             end
  1082.             if not utils.clearVegetation("down") then -- solid below
  1083.                 turtle.back()
  1084.                 moves = moves - 1
  1085.                 break
  1086.             end
  1087.         end
  1088.     end
  1089.     return moves
  1090. end
  1091.  
  1092. function utils.checkFarmPosition(R)
  1093.     local discovered = ""
  1094.     local success, storage, detected = false, "", 0
  1095.     local blockType = T:getBlockType("down")
  1096.     T:saveToLog("Checking position "..blockType.. " down")
  1097.     if blockType:find("water") ~= nil then -- over water, ? facing E (crops)
  1098.         -- network: E = __modem_, N = _modem__, W = modem___, S = ___modem
  1099.         -- storage: E = _chestchest_, N = chestchest__, W = chest__chest, S = __chestchest
  1100.         -- storage: E = _barrelbarrel_, N = barrelbarrel__, W = barrel__barrel, S = __barrelbarrel
  1101.         T:saveToLog("? over water = true")
  1102.         for i = 1, 4 do
  1103.             success, storage = utils.isStorage("forward")   -- true/false, chest, barrel, modem / ""
  1104.             local itemAhead = T:getBlockType("forward")
  1105.             if success then
  1106.                 discovered = discovered .. storage
  1107.                 if storage == "modem" then
  1108.                     R.networkFarm = true
  1109.                 end
  1110.             else
  1111.                 discovered = discovered .. "_"
  1112.             end
  1113.             if itemAhead:find("mysticalagriculture") ~= nil then
  1114.                 R.mysticalAgriculture = true
  1115.             end
  1116.             T:turnRight(1)
  1117.         end
  1118.         T:saveToLog("Storage blocks found:"..discovered)
  1119.         if R.networkFarm then
  1120.             if discovered == "modemmodem__" then
  1121.                T:turnLeft(1)
  1122.             elseif discovered == "modem__modem" then
  1123.                T:turnLeft(2)
  1124.             elseif discovered == "__modemmodem" then
  1125.                T:turnRight(1)
  1126.             end
  1127.             R.ready = true
  1128.         else
  1129.             if discovered == "barrelbarrel__" or
  1130.                discovered == "barrelchest__" or
  1131.                discovered == "chestchest__" then
  1132.                T:turnLeft(1)
  1133.             elseif discovered == "barrel__barrel" or discovered == "chest__chest" then
  1134.                T:turnLeft(2)
  1135.             elseif discovered == "__barrelbarrel" or discovered == "__chestchest" then
  1136.                T:turnRight(1)
  1137.             end
  1138.             R.ready = true
  1139.         end
  1140.     end -- else not over water
  1141.        
  1142.     T:saveToLog("utils.checkFarmPosition(R) return R = "..textutils.serialise(R, {compact = true}))
  1143.     return R -- now facing crops, R.ready = true/false, R.networkFarm = true/false
  1144. end
  1145.  
  1146. function utils.checkFuelNeeded(quantity)
  1147.     local fuelNeeded = quantity - turtle.getFuelLevel() -- eg 600
  1148.     if fuelNeeded > 0 then
  1149.         if T:checkInventoryForItem({"minecraft:lava_bucket"}, {1}, false) == nil then  
  1150.             if T:checkInventoryForItem({"coal"}, {math.ceil(fuelNeeded / 60)}, false) == nil then
  1151.                 T:checkInventoryForItem({"planks"}, {math.ceil(fuelNeeded / 15)})
  1152.             end
  1153.         end
  1154.         T:refuel(quantity, true)
  1155.     end
  1156. end
  1157.  
  1158. function utils.clearVegetation(direction)
  1159.     local isAirWaterLava = true -- default true value air/water/lava presumed
  1160.     -- blockType, data
  1161.     local blockType = T:getBlockType(direction)
  1162.     if blockType ~= "" then --not air
  1163.         if T:isVegetation(blockType) then
  1164.             T:dig(direction)
  1165.         elseif blockType:find("water") == nil
  1166.                and blockType:find("lava") == nil
  1167.                and blockType:find("bubble") == nil
  1168.                and blockType:find("ice") == nil then
  1169.             -- NOT water, ice or lava
  1170.             isAirWaterLava = false -- solid block
  1171.         end
  1172.     end
  1173.    
  1174.     return isAirWaterLava --clears any grass or sea plants, returns true if air or water, bubble column or ice
  1175. end
  1176.  
  1177. function utils.convertEssence(R)
  1178.     -- R.subChoice = 1 to 5
  1179.     -- R.size = quantity
  1180.     -- 1 = inferium to prudentium
  1181.     -- 2 = prudentium to tertium
  1182.     -- 3 = Tertium to Imperium
  1183.     -- 4 = Imperium to Supremium
  1184.     -- 5 = Supremium to Insanium
  1185.     local lib = {}
  1186.    
  1187.     function lib.getCrystal()
  1188.         local turtleSlot, turtleCount = 0, 0
  1189.         if utils.isStorage("up") then
  1190.             turtle.select(2)
  1191.             turtle.suckUp()
  1192.         end
  1193.         turtleSlot = T:getItemSlot("crystal")
  1194.         if turtleSlot == 0 then
  1195.             turtleSlot, turtleCount = network.getItemFromNetwork("chest", "infusion_crystal", 1)
  1196.         end
  1197.         if turtleSlot > 0 then
  1198.             turtle.select(turtleSlot)
  1199.             turtle.transferTo(2)
  1200.             return true
  1201.         end
  1202.        
  1203.         return false
  1204.     end
  1205.        
  1206.     function lib.loadCrafter(crafterName)
  1207.         -- up to 64 essences stored in turtle slots 2, 4, 6, 8
  1208.         local slots = {2, 4, 6, 8}
  1209.         for _, slot in ipairs(slots) do     -- drop each slot contents into crafter ? slot 1
  1210.             turtle.select(slot)
  1211.             turtle.drop()
  1212.             peripheral.call(crafterName, "pushItems", crafterName, 1, nil, slot)    -- move to correct place
  1213.         end
  1214.     end
  1215.    
  1216.     function lib.loadTurtle(availableStorage, availableStorageKeys, turtleName, essence, loaded)
  1217.         -- collect essence as above and place in minecraft:crafter slots 2, 4, 6, 8
  1218.         local crafterData = {{2, loaded}, {4, loaded}, {6, loaded}, {8, loaded}} -- eg 64 items in each of 4 slots in the crafter
  1219.         network.transferItemToTurtle(availableStorage, availableStorageKeys, turtleName, crafterData)
  1220.     end
  1221.        
  1222.     function lib.runCrafter(crafterName, count)
  1223.         -- can only craft once, then crystal + output is returned to turtle
  1224.         -- repeat process count times
  1225.         -- drop crystal into crafter, move to slot 5
  1226.         for i = 1, count do
  1227.             local slot = T:getItemSlot("crystal")
  1228.             if slot > 0 then
  1229.                 turtle.select(slot)
  1230.                 turtle.drop()
  1231.                 network.moveItem(crafterName, "crystal", 5)
  1232.                 for i = 1, 3 do
  1233.                     rs.setAnalogOutput("front",15)
  1234.                     sleep(0.1)
  1235.                     rs.setAnalogOutput("front", 0)
  1236.                 end
  1237.             end
  1238.         end
  1239.         while turtle.suck() do end
  1240.     end
  1241.    
  1242.     function lib.storeCrystal()
  1243.         if utils.isStorage("up") then
  1244.             T:dropItem("crystal", "up")
  1245.         else
  1246.             network.sendItemToNetworkStorage(R, "chest", "crystal", 0)  -- empty turtle to storage chests
  1247.         end
  1248.     end
  1249.    
  1250.     function lib.storeOutput()
  1251.         if utils.isStorage("up") then
  1252.             T:dropItem("crystal", "up")
  1253.         end
  1254.         network.sendItemToNetworkStorage(R, "chest", "all", 0)  -- empty turtle to storage chests
  1255.     end
  1256.    
  1257.     function lib.getStorageData(essence)
  1258.         local _, turtleName, storageNames = network.wrapModem(R, "chests")  -- _, turtleName, list of chest names
  1259.         if turtleName == "Modem not found" then return 0, nil, nil, turtleName end
  1260.         local availableStorage = {}
  1261.         local availableStorageKeys = {}
  1262.         local total = 0
  1263.         for _, storageName in pairs(storageNames) do
  1264.             local storageCount, storageData = network.findItemCountInInventory(storageName, essence)    -- eg 86, {{1, 64},{4, 22}}
  1265.             if storageCount > 0 then        -- eg 86: storage has some items
  1266.                 T:saveToLog("storageCount = "..storageCount)
  1267.                 availableStorage[storageName] = {}
  1268.                 availableStorage[storageName].count = storageCount
  1269.                 availableStorage[storageName].data = storageData
  1270.                 table.insert(availableStorageKeys, storageName)
  1271.                 total = total + storageCount
  1272.             end
  1273.         end
  1274.        
  1275.         return total, availableStorage, availableStorageKeys, turtleName
  1276.         --[[
  1277.             availableStorage.minecraft:chest_114 = {count = 86, data = {{1, 64},{4, 22}},
  1278.             availableStorage.minecraft:chest_115 = {count = 1024, data = {{1, 64},{2, 64},{3, 64}, ... }
  1279.         ]]
  1280.     end
  1281.    
  1282.     local essences = {"mysticalagriculture:inferium_essence",
  1283.                     "mysticalagriculture:prudentium_essence",
  1284.                     "mysticalagriculture:tertium_essence",
  1285.                     "mysticalagriculture:imperium_essence",
  1286.                     "mysticalagriculture:supremium_essence",
  1287.                     "mysticalagriculture:insanium_essence"}
  1288.    
  1289.     local message = network.loadStorageLists()
  1290.     if message ~= nil then return {message} end
  1291.     local essence = essences[R.subChoice]
  1292.     local missing = ""
  1293.     local crafter = peripheral.find("minecraft:crafter")
  1294.     local crafterName = ""
  1295.     local completed = 0
  1296.     if R.size > 0 then
  1297.         R.size = math.floor(R.size / 4) * 4
  1298.         if R.size == 0 then R.size = 4 end
  1299.     end
  1300.     if crafter == nil then
  1301.         return {"No crafter found"}
  1302.     else
  1303.         crafterName = peripheral.getName(crafter)
  1304.         T:saveToLog("utils.convertEssence started using "..essence..", quantity = "..R.size )
  1305.         while turtle.suck() do end                      -- empty crafter
  1306.         lib.storeOutput()   -- empty turtle to storage chests, (except crystal if storage above)
  1307.         if not lib.getCrystal() then
  1308.             return {"Failed: missing infusion crystal"}
  1309.         end
  1310.         lib.storeCrystal()
  1311.         local totalInStorage, availableStorage, availableStorageKeys, turtleName = lib.getStorageData(essence)
  1312.         if totalInStorage < 4 then
  1313.             return {"Only "..totalInStorage.." ".. essence:sub(20).." available"}
  1314.         else
  1315.             -- can only transfer full amount direct to crafter
  1316.             -- if a chest has 32 items in 1 slot and 32 in another Slot
  1317.             -- these have to be sent into the turtle first, dropped into crafter slot 1 and moved to correct position
  1318.             totalInStorage = math.floor(totalInStorage / 4) * 4
  1319.             if R.size == 0 then -- process all
  1320.                 T:saveToLog("initial call lib.loadTurtle("..turtleName..", "..totalInStorage.." essence")
  1321.                 local processed = 0
  1322.                 local loaded = 0
  1323.                
  1324.                 while totalInStorage > 0 do     -- use ALL stored essence
  1325.                     if totalInStorage - 256 > 0 then
  1326.                         loaded = 64
  1327.                     else
  1328.                         loaded = totalInStorage / 4
  1329.                     end
  1330.                     lib.loadTurtle(availableStorage, availableStorageKeys, turtleName, essence, loaded)
  1331.                     lib.loadCrafter(crafterName)
  1332.                     lib.getCrystal()
  1333.                     processed = processed + loaded
  1334.                     totalInStorage = totalInStorage - loaded * 4
  1335.                     lib.runCrafter(crafterName, loaded)
  1336.                     lib.storeOutput()
  1337.                 end
  1338.             elseif R.size <= 64 then
  1339.                 local loaded = math.min(R.size, totalInStorage) -- process requested or available if less
  1340.                 if loaded > 0 then
  1341.                     lib.loadTurtle(availableStorage, availableStorageKeys, turtleName, essence, loaded)
  1342.                     lib.loadCrafter(crafterName)
  1343.                     lib.getCrystal()
  1344.                     lib.runCrafter(crafterName, loaded)
  1345.                     lib.storeOutput()
  1346.                 end
  1347.             else    -- amount > 64
  1348.                 local processed = 0
  1349.                 local loaded = 0
  1350.                 while totalInStorage > 0 and processed < R.size do  -- stop if run out of essence, or reached target
  1351.                     if totalInStorage - 256 > 0 and R.size - processed >= 64 then
  1352.                         loaded = 64
  1353.                     elseif totalInStorage - 256 > 0 and R.size - processed > 0 then
  1354.                         loaded = R.size - processed
  1355.                     end
  1356.                     lib.loadTurtle(availableStorage, availableStorageKeys, turtleName, essence, loaded)
  1357.                     lib.loadCrafter(crafterName)
  1358.                     lib.getCrystal()
  1359.                     processed = processed + loaded
  1360.                     totalInStorage = totalInStorage - loaded * 4
  1361.                     lib.runCrafter(crafterName, loaded)
  1362.                     lib.storeOutput()
  1363.                 end
  1364.             end
  1365.             lib.storeOutput()
  1366.         end
  1367.         return {"success"}
  1368.     end
  1369. end
  1370.  
  1371. function utils.craftItem(R)
  1372.     local message = network.loadStorageLists()  -- initialises or creates lists of where an item can be found: GLOBAL LISTS!
  1373.     if message ~= nil then return {message} end
  1374.     T:saveToLog("call network.emptyInventory(R, {'sapling', 'propagule', 'dirt', 'crafting'}, {'all'}, true)")
  1375.     network.emptyInventory(R, {"sapling", "propagule", "dirt", "crafting"}, {"all"}, true)  -- 2 lists, one for barrels, one for chests
  1376.     T:saveToLog("\ncall network.getItemFromNetwork('barrel', 'minecraft:dirt', 169)")
  1377.     --network.getItemFromNetwork(storageType, itemRequired, countRequired, toTurtleSlot, ignoreStock)
  1378.     local turtleSlot, turtleCount = network.getItemFromNetwork("barrel", "minecraft:dirt", 169, nil, false)
  1379.     T:saveToLog("Get from barrel completed: turtleSlot = "..turtleSlot..", turtleCount = "..turtleCount..", ignoreStock = "..tostring(ignoreStock))
  1380.     if turtleCount < 169 then
  1381.         T:saveToLog("\ncall network.getItemFromNetwork('chest', 'minecraft:dirt', ".. 169 - turtleCount..", ignoreStock = "..tostring(ignoreStock)..")")
  1382.         turtleSlot, turtleCount = network.getItemFromNetwork("chest", "minecraft:dirt", 169 - turtleCount, nil, true)
  1383.         if turtleCount < 169 then   -- ask player for saplings
  1384.             T:checkInventoryForItem({"dirt"}, {169 - turtleCount})
  1385.         end
  1386.     end
  1387.     return {""}
  1388. end
  1389.  
  1390. function utils.createPath(R) -- 51
  1391.     --[[places a path in air/lava/water. R can also be of type int]]
  1392.     local length = 0
  1393.     local reduce = false
  1394.     local torchInterval = 8
  1395.     if type(R) == "table" then
  1396.         length = R.length
  1397.         if R.data == "reduce" then
  1398.             reduce = true
  1399.         end
  1400.         torchInterval = R.torchInterval
  1401.     else
  1402.         length = R
  1403.     end
  1404.     local numBlocks = 0
  1405.    
  1406.     if reduce then
  1407.         T:forward(1)
  1408.         local blockType = T:getBlockType("down")
  1409.         local useBlock = blockType
  1410.         while blockType == useBlock do
  1411.             T:go("x2F1")
  1412.             numBlocks = numBlocks + 1
  1413.             blockType = T:getBlockType("down")
  1414.         end
  1415.         utils.goBack(numBlocks + 1)
  1416.     else
  1417.         for i = 1, 2 do
  1418.             T:fillVoid("down", {}, false)
  1419.             T:forward(1)
  1420.             numBlocks = numBlocks + 1
  1421.         end
  1422.         local place = utils.clearVegetation("down")
  1423.         while place do -- while air, water, normal ice, bubble column or lava below
  1424.             if T:fillVoid("down", {}, false) then -- false if out of blocks
  1425.                 T:forward(1)
  1426.                 numBlocks = numBlocks + 1
  1427.                 if numBlocks % torchInterval == 1 or numBlocks == 0 then
  1428.                     if T:getItemSlot("minecraft:torch", -1) > 0 then
  1429.                         T:turnRight(2)
  1430.                         T:place("minecraft:torch", "forward", false)
  1431.                         T:turnRight(2)
  1432.                     end
  1433.                 end
  1434.             else
  1435.                 break
  1436.             end
  1437.             if length > 0 and numBlocks >= length then -- not infinite path (length = 0)
  1438.                 break
  1439.             end
  1440.             place = utils.clearVegetation("down")
  1441.         end
  1442.     end
  1443.     return {numBlocks} -- returned as a table in case called as task 51 back to main()
  1444. end
  1445.  
  1446. function utils.createStorage()
  1447.     -- start in centre, above 1st level
  1448.     T:place("modem", "down")
  1449.     utils.goBack(1)
  1450.    
  1451.     T:place("chest", "down")    -- places with handle at back of turtle
  1452.     T:go("R1F1L1")
  1453.     T:place("chest", "down")    -- 2nd of pair
  1454.     for i = 1, 3 do
  1455.         T:go("F1L1")
  1456.         T:place("chest", "down")
  1457.         T:go("R1F1L1")
  1458.         T:place("chest", "down")
  1459.     end
  1460.    
  1461.     T:go("F1L1F1")  -- end at starting position
  1462.     network.attachModem()
  1463. end
  1464.  
  1465. function utils.createWalledSpace(D)
  1466.     --[[
  1467.     D.width  = #
  1468.     D.length = #
  1469.     D.height = #
  1470.     D.ceiling = false
  1471.     D.floor = false
  1472.     D.vDirection = "U" or "D"
  1473.     D.hDirection = "LR" or "RL"
  1474.     D.goHome = true
  1475.     T:go(path, useTorch, torchInterval, leaveExisting, preferredBlock)
  1476.     ]]
  1477.     local turn  = "R1"
  1478.     local oTurn = "L1"
  1479.     if D.hDirection == "RL" then
  1480.         turn    = "L1"
  1481.         oTurn   = "R1"
  1482.     end
  1483.     local placeF = "C1"
  1484.  
  1485.     local lib = {}
  1486.    
  1487.     function lib.getPlace(D, start, finish)
  1488.         if start then                       -- start of build
  1489.             if D.vDirection == "U" then     -- bottom and going up
  1490.                 if D.floor then
  1491.                     return "C2"
  1492.                 end
  1493.             else                            -- D.vDirection == "D"
  1494.                 if D.ceiling then
  1495.                     return "C0"
  1496.                 end
  1497.             end
  1498.         elseif finish then                  -- end of build
  1499.             if D.vDirection == "U" then     -- bottom and going up
  1500.                 if D.ceiling then
  1501.                     return "C0"
  1502.                 end
  1503.             else
  1504.                 if D.floor then
  1505.                     return "C2"
  1506.                 end
  1507.             end
  1508.         end
  1509.  
  1510.         return ""                           -- start and finish both false
  1511.     end
  1512.    
  1513.     function lib.layer(D, start, finish)
  1514.         local outward = true
  1515.         local place = lib.getPlace(D, start, finish)
  1516.         for width = 1, D.width do
  1517.             for length = 1, D.length do
  1518.                 if start or finish then
  1519.                     T:go(place, false, 0, true)-- place floor / ceiling
  1520.                 end
  1521.                 if width == 1 then
  1522.                     T:go(oTurn..placeF..turn, false, 0, true)       -- face wall and check if block
  1523.                 elseif width == D.width then                        -- checking opposite wall
  1524.                     if outward then                                 -- travelling away from start
  1525.                         T:go(turn..placeF..oTurn, false, 0, true)   -- face wall and check if block
  1526.                     else                                            -- travelling towards start
  1527.                         T:go(oTurn..placeF..turn, false, 0, true)   -- face wall and check if block
  1528.                     end                
  1529.                 end
  1530.                 -- move forward
  1531.                 if length < D.length then
  1532.                     T:forward(1)
  1533.                 end
  1534.             end
  1535.             if width < D.width then
  1536.                 -- change direction
  1537.                 if outward then
  1538.                     T:go(placeF..turn.."F1"..oTurn..place..placeF..turn..turn, false, 0, true)
  1539.                 else
  1540.                     T:go(placeF..oTurn.."F1"..turn..place..placeF..oTurn..oTurn, false, 0, true)
  1541.                 end
  1542.                 outward = not outward
  1543.             else
  1544.                 if outward then
  1545.                     T:go(placeF..oTurn.."F"..D.width -1 ..oTurn.."F"..D.length - 1 .."C1R2", false, 0, true)
  1546.                 else
  1547.                     T:go(placeF..turn.."F"..D.width -1 ..oTurn..placeF..turn..turn, false, 0, true)
  1548.                 end
  1549.             end
  1550.         end
  1551.     end
  1552.    
  1553.     T:go("R2"..placeF.."R2", false, 0, true)            -- place block behind
  1554.     for height = 1, D.height do
  1555.         if height == 1 then
  1556.             lib.layer(D, true, false)                   -- first layer, so start = true, finish = false
  1557.         elseif height == D.height then     
  1558.             lib.layer(D, false, true)                   -- final layer, so start = false, finish = true
  1559.         else
  1560.             lib.layer(D, false, false)                  -- mid build layer, so start = false, finish = false
  1561.         end
  1562.         if height < D.height then                       -- go up or down unless finished build
  1563.             T:go(D.vDirection.."1")
  1564.         end
  1565.     end
  1566.     if D.goHome then-- ends at starting point
  1567.         if D.vDirection == "U" then
  1568.             T:down(D.height - 1)
  1569.         else
  1570.             T:up(D.height - 1)
  1571.         end
  1572.     end
  1573. end
  1574.  
  1575. function utils.createWaterSource(level)
  1576.     if level == nil then
  1577.         level = 0
  1578.     end
  1579.     if level > 0 then
  1580.         T:up(level)
  1581.     elseif level < 0 then
  1582.         T:down(math.abs(level))
  1583.     end
  1584.     -- assume on flat surface, but allow for blocks above
  1585.     T:go("x0C2F1 x0C2F1 x0C2F1 x0C2R1 F1 x0C2F1 x0C2F1 x0C2R1 F1 x0C2F1 x0C2F1 x0C2R1 F1 x0C2F1 x0C2", false, 0, false)
  1586.     T:go("R1F1D1", false, 0, false) --move to corner and drop down
  1587.     T:go("C2F1R1 C2F1R1 C2F1R1 C2F1R1", false, 0, false)
  1588.     T:go("U1")
  1589.     for i = 1, 2 do
  1590.         T:placeWater("down")
  1591.         T:go("F1R1F1R1", false, 0, false)
  1592.     end
  1593.     -- refill water buckets
  1594.     for i = 1, 2 do
  1595.         sleep(0.5)
  1596.         T:placeWater("down")
  1597.     end
  1598.     T:go("R2F1R1F1R1")
  1599.     -- end above lower left of pond (starting point)
  1600.     return {}
  1601. end
  1602.  
  1603. function utils.drainWaterLava(R) -- 84
  1604.     local lib = {}
  1605.        
  1606.     function lib.startCorner()
  1607.         -- assume starting mid 3 high column
  1608.         T:go("D1C1R1C1 U1C1 L1C1 U1C1 R1C1 L1D1 C2C0", false, 0, true)
  1609.         -- finished on same mid 3 high column with wall in front and right completed
  1610.     end
  1611.    
  1612.     function lib.midWall()
  1613.         -- assume starting mid 3 high column
  1614.         T:go("D1R1C1 U1C1 U1C1 L1D1 C2C0", false, 0, true)
  1615.         -- finished on same mid 3 high column with wall on right completed
  1616.     end
  1617.    
  1618.     function lib.endCorner()
  1619.         -- assume starting mid 3 high column
  1620.         T:go("D1R1C1 R1C1 U1C1 L1C1 U1C1 R1C1 L2D1 C2C0", false, 0, true)
  1621.         -- finished on same mid 3 high column with wall behind and right completed
  1622.     end
  1623.    
  1624.     function lib.backTurn(offset)
  1625.         -- assume starting mid 3 high column with completed wall behind
  1626.         T:go("L1F"..offset.."L2C1 R1D1C1 U1C1 U1C1 D1")
  1627.         -- end next row along the width, facing the back, mid point
  1628.         -- wall ahead completed
  1629.     end
  1630.    
  1631.     function lib.frontTurn()
  1632.         -- assume starting mid 3 high column facing back
  1633.         -- next move is to turn toward previous wall and remove
  1634.  
  1635.         T:go("L2D1C1 U1C1 U1C1 D1R1 C2C0 F1R2C1 L1")
  1636.         -- end facing back ready to remove wall
  1637.     end
  1638.    
  1639.     function lib.placeRetainingWall(distance)
  1640.         -- assume starting above 3 block high area facing front (player)
  1641.         --T:down(1)
  1642.         for i = 1, distance do
  1643.             if i == 1 then -- start wall
  1644.                 lib.startCorner()
  1645.             else
  1646.                 if i < distance then -- mid wall
  1647.                     lib.midWall()
  1648.                 else -- end of the wall
  1649.                     lib.endCorner()
  1650.                 end
  1651.             end
  1652.             if i < distance then
  1653.                 if not turtle.back() then
  1654.                     T:go("R2F1R2")
  1655.                 end
  1656.                 T:go("C1", false, 0, true)
  1657.             end
  1658.         end
  1659.         return 1
  1660.     end
  1661.        
  1662.     function lib.placeDam(distance)
  1663.         --T:go(path, useTorch, torchInterval, leaveExisting, preferredBlock)
  1664.         local waterPresent = false
  1665.         for i = 1, distance do
  1666.             if T:isWater("down") then
  1667.                 waterPresent = true
  1668.             end
  1669.             T:go("C0C2")
  1670.             if i < distance then
  1671.                 if not turtle.back() then
  1672.                     T:go("R2F1R2")
  1673.                 end
  1674.                 T:go("C1", false, 0, true)
  1675.             end
  1676.         end
  1677.         return waterPresent
  1678.     end
  1679.    
  1680.     function lib.removeDam(distance)
  1681.         -- facing towards back, dig wall up down and ahead
  1682.         for i = 1, distance do
  1683.             T:go("x0x2")
  1684.             if i < distance then
  1685.                 T:forward(1)
  1686.             end
  1687.         end
  1688.         T:turnRight(2)
  1689.         -- end up back to retaining wall, mid point
  1690.     end
  1691.    
  1692.     if R.data == "" then
  1693.         if turtle.detectDown() then -- on solid surface
  1694.             turtle.forward()
  1695.         end
  1696.     end
  1697.     if deletesWater then --turtle replaces source so use clearSolid()
  1698.         R.silent = true
  1699.         R.down = true
  1700.         R.up = false
  1701.         clearSolid(R)
  1702.     else -- mc 1.12.15+ turtle does NOT replace source blocks
  1703.         -- place first stone along the length of water and measure length
  1704.         R = utils.calculateDimensions(R)
  1705.         local depth = 0
  1706.         for h = 1, R.height do
  1707.             local waterPresent = false -- resets at each level
  1708.             local row = 0
  1709.             T:go("R2D2", false, 0, true)    -- at start, level 2, facing towards the front, move backards
  1710.             --lib.startCorner()
  1711.             row = row + lib.placeRetainingWall(R.length)
  1712.             lib.backTurn(1)
  1713.             if lib.placeDam(R.length) then
  1714.                 waterPresent = true
  1715.             end
  1716.             row = row + 1
  1717.             -- left side wall has been built, 2 rows of walls laid. row = 2
  1718.             while row < R.width - 1 do -- eg width=2, row=2, this will not run. width=5, row=2
  1719.                 lib.frontTurn()
  1720.                 lib.removeDam(R.length)
  1721.                 lib.backTurn(2)
  1722.                 if lib.placeDam(R.length) then
  1723.                     waterPresent = true
  1724.                 end
  1725.                 row = row + 1
  1726.             end
  1727.             -- lay right wall
  1728.             lib.frontTurn()
  1729.             lib.removeDam(R.length)
  1730.             lib.backTurn(2)
  1731.             row = row + lib.placeRetainingWall(R.length)
  1732.             lib.frontTurn()
  1733.             lib.removeDam(R.length) -- faces to front on finish
  1734.             T:go("L1F1R1")
  1735.             lib.removeDam(R.length) -- faces to front on finish
  1736.             depth = depth + 3
  1737.             T:go("L1F"..R.width - 1 .."R1D1")
  1738.             if depth == R.height or not waterPresent then -- still replacing water
  1739.                 T:up(depth - 1) -- when h == R.height will go up
  1740.                 break -- in case water dries up before loop has completed
  1741.             end
  1742.         end
  1743.     end
  1744.     return {}
  1745. end
  1746.  
  1747. function utils.dropSand()
  1748.     while not turtle.detectDown() do -- over water. will be infinite loop if out of sand
  1749.         if not T:place("sand", "down", false) then
  1750.             print("Out of sand. Add more to continue...")
  1751.             sleep(2)
  1752.         end
  1753.     end
  1754.     return true --will only get to this point if turtle.detectDown() = true
  1755. end
  1756.  
  1757. function utils.pause(R)
  1758.     --[[
  1759.     allows 2 turtles to co-operate
  1760.     When they face each other and move together
  1761.     R.side = "R" or "L"
  1762.     ]]
  1763.     local rsIn = rs.getAnalogueInput("front")
  1764.     local rsOut = 1
  1765.     local present = false
  1766.     local confirmed = false
  1767.     local timer = 0
  1768.     local endTimer = 0
  1769.     local finished = false
  1770.     T:go(R.side.."1")
  1771.     rs.setAnalogueOutput("front", 0) -- switch off output
  1772.     local blockType = T:getBlockType("forward")
  1773.     while blockType:find("turtle") == nil do
  1774.         menu.colourWrite("Waiting for other turtle "..endTimer, colors.orange, nil, nil, false, true)
  1775.         blockType = T:getBlockType("forward")
  1776.         sleep(0.1)
  1777.         timer = timer + 0.1
  1778.         if timer >= 1 then
  1779.             timer = 0
  1780.             endTimer = endTimer + 1
  1781.         end
  1782.     end
  1783.     timer = 0
  1784.     endTimer = 0
  1785.    
  1786.     while not finished do
  1787.         if endTimer == 0 then
  1788.             if present then
  1789.                 if confirmed then
  1790.                     menu.colourWrite("Turtle confirmed: input = "..rsIn.." output = "..rsOut, colors.orange, nil, nil, false, true)
  1791.                 else
  1792.                     menu.colourWrite("Other turtle ok: input = "..rsIn.." output = "..rsOut, colors.orange, nil, nil, false, true)
  1793.                 end
  1794.             else
  1795.                 menu.colourWrite("Waiting: input = "..rsIn.." output = "..rsOut, colors.orange, nil, nil, false, true)
  1796.             end
  1797.         end
  1798.         sleep(0.1)
  1799.         timer = timer + 1
  1800.         if endTimer > 0 then
  1801.             endTimer = endTimer + 1
  1802.         end
  1803.         if endTimer >= 10 then -- allows time for other computer to get ready
  1804.             finished = true
  1805.         end
  1806.         rs.setAnalogueOutput("front", 1) -- output 1 as a signal initially
  1807.         if present then
  1808.             rs.setAnalogueOutput("front", rsOut) -- output 1 as a signal initially
  1809.         end
  1810.         rsIn = rs.getAnalogueInput("front")
  1811.         if rsIn == 1 then
  1812.             present = true
  1813.             if not confirmed then
  1814.                 rsOut = 7
  1815.             end
  1816.         elseif rsIn == 7 then
  1817.             present = true
  1818.             confirmed = true
  1819.             rsOut = 15
  1820.         elseif rsIn == 15 or confirmed then
  1821.             menu.colourWrite("endTimer active = "..endTimer, colors.orange, nil, nil, false, true)
  1822.             endTimer = endTimer + 1 -- start endTimer
  1823.         end
  1824.     end
  1825.     T:go(R.side.."3")
  1826.    
  1827.     return rsIn -- 15
  1828. end
  1829.  
  1830. function utils.startWaterFunction(inWater, onWater, maxDescent, goIn)
  1831.     --[[
  1832.         Use with utils.getWaterStatus
  1833.         could be inWater and not onWater if on single layer of water with floor below
  1834.     ]]
  1835.     maxDescent = maxDescent or 1 -- prevents continuous down due to lack of water
  1836.     if goIn == nil then -- turtle should submerge
  1837.         goIn = true
  1838.     end
  1839.     local descent = 0 -- counter for going down
  1840.     if inWater then -- job done
  1841.         return inWater, onWater
  1842.     end
  1843.     if onWater and goIn then --inWater already returned so must be false
  1844.         if turtle.down() then
  1845.             if T:getBlockType("forward"):find("water") ~= nil or T:getBlockType("forward"):find("bubble") ~= nil then
  1846.                 inWater = true
  1847.                 return inWater, onWater
  1848.             end
  1849.         end
  1850.     end
  1851.     if not onWater then
  1852.         T:forward(1)
  1853.     end
  1854.     -- should now be above water, but may be further down so onWater still false
  1855.     while T:getBlockType("down"):find("water") == nil and T:getBlockType("down"):find("bubble") == nil do-- go down until water below
  1856.         if turtle.down() then
  1857.             descent = descent + 1
  1858.             if descent >= maxDescent then -- limit descent in case no water present
  1859.                 onWater = false
  1860.                 return inWater, onWater -- false,false
  1861.             end
  1862.         else
  1863.             onWater = false
  1864.             return inWater, onWater -- false,false
  1865.         end
  1866.     end
  1867.     -- now above water
  1868.     if goIn then
  1869.         turtle.down() -- now inside water block
  1870.         inWater = true
  1871.     end
  1872.     return inWater, onWater
  1873. end
  1874.  
  1875. function utils.fillBucket(direction)
  1876.     if T:getWater(direction)  then
  1877.         sleep(0.3)
  1878.         return true
  1879.     else
  1880.         return false
  1881.     end
  1882. end
  1883.  
  1884. function utils.getEmptyBucketCount()
  1885.     local lastSlot, leastModifier, total, slotData = T:getItemSlot("minecraft:bucket")
  1886.     return total
  1887. end
  1888.  
  1889. function utils.getRoofStats(R)
  1890.     local isWidthOdd = R.width % 2 == 1             -- is the width odd or even?
  1891.     local isLengthOdd = R.length % 2 == 1           -- is the length odd or even?
  1892.     if isWidthOdd then
  1893.         R.height = math.floor(R.width / 2)          -- eg 7 x 5 roof, layers = 5 / 2 = 2
  1894.     else
  1895.         R.height = R.width / 2                      -- eg 17 x 8 roof, layers = 8 / 2 = 4
  1896.     end
  1897.    
  1898.     local width = 2                                 -- assume even width with  2 block roof ridge
  1899.     local length = R.length - R.height - 1          -- assume even width with  2 block roof ridge
  1900.     if isWidthOdd then
  1901.         width = 3                                   -- adjust to allow for single width roof ridge
  1902.     end
  1903.     if isLengthOdd then
  1904.         length = R.length - R.height                -- adjust as 1 layer less
  1905.     end
  1906.    
  1907.     return R, isWidthOdd, isLengthOdd, width, length
  1908. end
  1909.  
  1910. function utils.getSlotCapacity(slot)
  1911.     return turtle.getItemSpace(slot) + turtle.getItemCount(slot)
  1912. end
  1913.  
  1914. function utils.getWater()
  1915.     if deletesWater then
  1916.         T:getWater("down") -- take water from source
  1917.         sleep(0.2)
  1918.         T:getWater("down") -- take water from source
  1919.     else
  1920.         if not turtle.detectDown() then
  1921.             T:go("C2", false, 0, false)
  1922.         end
  1923.         T:getWater("forward") -- take water from source
  1924.         sleep(0.2)
  1925.         T:getWater("forward") -- take water from source
  1926.     end
  1927. end
  1928.  
  1929. function utils.getWaterBucketCount()
  1930.     -- lastSlot, leastModifier, total, slotData  = T:getItemSlot(item)
  1931.     local lastSlot, leastModifier, total, slotData = T:getItemSlot("minecraft:water_bucket")
  1932.     return total
  1933. end
  1934.  
  1935. function utils.getWaterStatus()
  1936.     --[[ Usage:
  1937.     local inWater, onWater = utils.getWaterStatus() -- returns turtle position near water
  1938.     utils.startWaterFunction(onWater, inWater, 2, true) -- move INTO water max 2 blocks down
  1939.     utils.startWaterFunction(onWater, inWater, 5, false) -- move ABOVE water max 5 blocks down
  1940.     ]]
  1941.     local onWater = false
  1942.     local inWater = false
  1943.     for i = 1, 4 do
  1944.         if T:getBlockType("forward"):find("water") ~= nil  or T:getBlockType("forward"):find("bubble") ~= nil then
  1945.             inWater = true
  1946.         end
  1947.         T:turnRight(1)
  1948.     end
  1949.     if T:getBlockType("down"):find("water") ~= nil or T:getBlockType("down"):find("bubble") ~= nil then
  1950.         onWater = true
  1951.     end
  1952.     return inWater, onWater
  1953. end
  1954.  
  1955. function utils.goBack(blocks)
  1956.     blocks = blocks or 1
  1957.     local success = true
  1958.     for i = 1, blocks do
  1959.         if not turtle.back() then
  1960.             success = false
  1961.             T:go("R2F1R2")
  1962.         end
  1963.     end
  1964.    
  1965.     return success
  1966. end
  1967.  
  1968. function utils.getPrettyPrint(promptColour, menuPromptColour)
  1969.     promptColour = promptColour or colors.yellow
  1970.     menuPromptColour = menuPromptColour or colors.white
  1971.    
  1972.     local pp = {}
  1973.     pp.prompt = promptColour
  1974.     pp.itemColours = {}
  1975.     pp.menuPrompt = menuPromptColour
  1976.     pp.allowModifier = true -- can the menu return 'q' or 'h' / 'i'
  1977.    
  1978.     return pp
  1979. end
  1980.  
  1981. function utils.isStorage(direction)
  1982.     local blockType = T:getBlockType(direction)
  1983.     if blockType:find("barrel") ~= nil then
  1984.         return true, "barrel"
  1985.     elseif blockType:find("chest") ~= nil then
  1986.         return true, "chest"
  1987.     elseif blockType:find("modem") ~= nil then
  1988.         return true, "modem"
  1989.     end
  1990.    
  1991.     return false, blockType
  1992. end
  1993.  
  1994. function utils.isTableEmpty(aTable)
  1995.     if next(aTable) == nil then
  1996.         return true
  1997.     end
  1998.     return false
  1999. end
  2000.  
  2001. function utils.move(R, blocks, reverse)
  2002.     if blocks == 0 then return 0 end
  2003.     if reverse == nil then
  2004.         reverse = false
  2005.     end
  2006.     if reverse then
  2007.         --if R.subChoice == 2 then -- reverse direction
  2008.         if R.direction == "down" or R.subChoice == 2 then -- reverse direction
  2009.             T:up(blocks)
  2010.         else
  2011.             T:down(blocks)
  2012.         end
  2013.         return blocks * -1
  2014.     else
  2015.         --if R.subChoice == 1 then
  2016.         if R.direction == "up" or R.subChoice == 1 then
  2017.             T:up(blocks)
  2018.         else
  2019.             T:down(blocks)
  2020.         end
  2021.         return blocks
  2022.     end
  2023. end
  2024.  
  2025. function utils.printR(R)
  2026.     --[[
  2027.         choice = 0,
  2028.         subChoice = 0,
  2029.         size = 0,
  2030.         width = 0,
  2031.         length = 0,
  2032.         height = 0,
  2033.         depth = 0,
  2034.         up = false,
  2035.         down = false,
  2036.         silent = false,
  2037.         data = {},
  2038.         torchInterval = 0,
  2039.         useBlockType = "",
  2040.         auto = false,
  2041.         side = "",
  2042.         direction = "",
  2043.         ready = false,
  2044.         networkFarm = false,
  2045.         mysticalAgriculture = false,
  2046.         logType = "",
  2047.         treeSize = "",
  2048.         message = ""
  2049.     ]]
  2050.     T:clear()
  2051.     print("choice= ".. tostring(R.choice)..", subChoice= ".. tostring(R.subChoice))
  2052.     print("size= ".. tostring(R.size)..", width= ".. tostring(R.width))
  2053.     print("length= ".. tostring(R.length)..", height= ".. tostring(R.height)..", depth= ".. tostring(R.depth))
  2054.     print("silent= "..tostring(R.silent)..", up= ".. tostring(R.up)..", down= "..tostring(R.down))
  2055.     print("torchInterval= ".. tostring(R.torchInterval))
  2056.     print("useBlockType= "..tostring(R.useBlockType))
  2057.     print("auto= ".. tostring(R.auto)..", side= "..tostring(R.side)..", direction= "..tostring(R.direction))
  2058.     print("ready= ".. tostring(R.ready)..", networkFarm= "..tostring(R.networkFarm)..", mysticalAgriculture= "..tostring(R.mysticalAgriculture))
  2059.     print("logType= ".. tostring(R.logType)..", treeSize= "..tostring(R.treeSize)..", message= "..tostring(R.message))
  2060.     if type(R.data) == "table" then
  2061.         io.write("data= ")
  2062.         for k,v in pairs(R.data) do
  2063.             print(k ..": "..tostring(v))
  2064.         end
  2065.     else
  2066.         print("data= ".. tostring(R.data))
  2067.     end
  2068.     io.write("Enter to continue")
  2069.     read()
  2070. end
  2071.  
  2072. function utils.setStorageOptions()
  2073.     local storage = ""
  2074.     local storageBackup = ""
  2075.     if T:getItemSlot("barrel") > 0 then
  2076.         storage = "barrel"
  2077.         storageBackup = "barrel"
  2078.     end
  2079.     if T:getItemSlot("chest") > 0 then
  2080.         if storage == "" then
  2081.             storage = "chest"
  2082.         end
  2083.         storageBackup = "chest"
  2084.     end
  2085.     return storage, storageBackup
  2086. end
  2087.  
  2088. function utils.tableConcat(tbl, sep)
  2089.     local output = ""
  2090.     for i,value in pairs(tbl) do
  2091.         output = output .. tostring(value)
  2092.         if i ~= #tbl then
  2093.             output = output .. sep
  2094.         end
  2095.     end
  2096.  
  2097.     return output
  2098. end
  2099.  
  2100. function utils.tableContains(tableName, value, exactMatch)
  2101.     exactMatch = exactMatch or false
  2102.     for k, v in ipairs(tableName) do
  2103.         if exactMatch then
  2104.             if v == value then
  2105.                 return true
  2106.             end
  2107.         else
  2108.             if v:find(value) ~= nil then
  2109.                 return true
  2110.             end
  2111.         end
  2112.     end
  2113.     return false
  2114. end
  2115.  
  2116. function utils.useSticksAsFuel()
  2117.     local slot = T:getItemSlot("minecraft:stick")
  2118.     if slot > 0 then -- use any sticks to refuel
  2119.         turtle.select(slot)
  2120.         turtle.refuel()
  2121.     end
  2122.     slot = T:getItemSlot("minecraft:mangrove_roots")
  2123.     if slot > 0 then -- use any roots to refuel
  2124.         turtle.select(slot)
  2125.         turtle.refuel()
  2126.     end
  2127. end
  2128.  
  2129. function utils.waitForInput(message)
  2130.     -- debugging. Pause script and asks user to continue
  2131.     -- global dbug flag has to be set first
  2132.     if dbug then
  2133.         if message ~= nil then
  2134.             print(message)
  2135.             T:saveToLog(message, false)
  2136.         end
  2137.         menu.colourPrint("Enter to continue...", colors.lightBlue)
  2138.         --io.write("Press Enter to continue...")
  2139.         return read()
  2140.     end
  2141. end
  2142.  
  2143. function utils.writeTraceTable(description, tbl)
  2144.     local text = ""
  2145.     for key, value in pairs(tbl) do
  2146.         if type(value) == "table" then
  2147.             for k,v in pairs(value) do
  2148.                 text = text.."k = "..k..", v = "..v.."; "
  2149.             end
  2150.         else
  2151.             text = "key = "..key..", value = "..value
  2152.         end
  2153.     end
  2154.     T:saveToLog(description.."\n".. text)
  2155. end
  2156.  
  2157. local pp = utils.getPrettyPrint()
  2158.  
  2159. local function buildWall(R)
  2160.     -- T:go(path, useTorch, torchInterval, leaveExisting, preferredBlock)
  2161.     local lib = {}
  2162.    
  2163.     function lib.singleLayer(R)
  2164.         for l = 1, R.length do
  2165.             if l == R.length and R.data == "house" then
  2166.                 T:turnRight(1)
  2167.             end
  2168.             utils.goBack(1)
  2169.             T:go("C1", false, 0, false, R.useBlockType)
  2170.         end
  2171.     end
  2172.    
  2173.     function lib.doubleLayer(R)
  2174.         for l = 1, R.length do
  2175.             T:go("C2", false, 0, false, R.useBlockType)
  2176.             if l == R.length and R.data == "house" then
  2177.                 T:turnRight(1)
  2178.             end
  2179.             utils.goBack(1)
  2180.             T:go("C1", false, 0, false, R.useBlockType)
  2181.         end
  2182.     end
  2183.    
  2184.     function lib.tripleLayer(R)
  2185.         for l = 1, R.length do
  2186.             T:go("C2C0", false, 0, false, R.useBlockType)
  2187.             if l == R.length and R.data == "house" then
  2188.                 T:turnRight(1)
  2189.             end
  2190.             utils.goBack(1)
  2191.             T:go("C1", false, 0, false, R.useBlockType)
  2192.         end
  2193.     end
  2194.    
  2195.     -- R.width preset to 1
  2196.     local remaining = R.height
  2197.    
  2198.     if R.subChoice == 1 then
  2199.         T:forward(1)            -- face forward, move over first block
  2200.     end
  2201.     if R.height > 1 then
  2202.         T:up(1)                 -- go up 1 block
  2203.     end
  2204.     if R.data ~= "house" then
  2205.         T:turnRight(2)              -- rotate 180
  2206.     end
  2207.     if R.height == 1 then       -- single block: place in front
  2208.         lib.singleLayer(R)
  2209.     elseif R.height == 2 then   -- 2 blocks, go backwards, place below and ahead
  2210.         lib.doubleLayer(R)
  2211.     else
  2212.         while remaining >= 3 do
  2213.             lib.tripleLayer(R)
  2214.             remaining = remaining - 3
  2215.             if remaining == 1 then
  2216.                 T:go("U2F1R2")
  2217.             elseif remaining > 1 then
  2218.                 T:go("U3F1R2")
  2219.             end
  2220.         end
  2221.         if remaining == 1 then
  2222.             lib.singleLayer(R)
  2223.         elseif remaining == 2 then
  2224.             lib.doubleLayer(R)
  2225.         end
  2226.     end
  2227.     if not R.silent then
  2228.         while turtle.down() do end
  2229.     end
  2230.    
  2231.     return {}
  2232. end
  2233.  
  2234. local function buildStructure(R)
  2235.     local lib = {}
  2236.    
  2237.     function lib.goDown(R)
  2238.         --T:go("L1F1 R1F1 L1")  -- now on ground floor
  2239.         if R.height > 1 then
  2240.             T:down(1)
  2241.         end
  2242.     end
  2243.     local buildHeight = R.height
  2244.     local height = R.height
  2245.     local width = R.width
  2246.     local length = R.length
  2247.    
  2248.     if R.subChoice == 1 then
  2249.         T:forward(1)            -- face forward, move over first block
  2250.     end
  2251.     if R.height > 3 then
  2252.         R.height = 3
  2253.     end
  2254.     T:turnRight(2)
  2255.     R.data = "house"
  2256.     R.silent = true -- prevent return to ground after buildWall(R)
  2257.     R.subChoice = 2 -- do not move forward when building walls
  2258.     while height > 0 do
  2259.         buildWall(R)
  2260.        
  2261.         lib.goDown(R)
  2262.         R.length = width - 1
  2263.         buildWall(R)
  2264.         lib.goDown(R)
  2265.        
  2266.         R.length = length - 1
  2267.         buildWall(R)
  2268.         lib.goDown(R)
  2269.        
  2270.         R.length = width - 2
  2271.         buildWall(R)
  2272.         height = height - R.height  -- 1, 2 or 3
  2273.         if height > 0 then
  2274.             T:go("U2 R1F1 L1F1")
  2275.             R.height = height
  2276.             if height > 3 then
  2277.                 R.height = 3
  2278.             end
  2279.             R.length = length
  2280.         end
  2281.     end
  2282.    
  2283.     T:go("U2F2")
  2284.     while turtle.down() do end
  2285.     T:go("R1F1R1")
  2286.    
  2287.     return {}
  2288. end
  2289.  
  2290. local function buildGableRoof(R)
  2291.     --[[
  2292.     stairs placement:
  2293.        _|   up
  2294.      
  2295.        T L  forward
  2296.        _
  2297.         |   down
  2298.  
  2299.     ]]
  2300.     local lib = {}
  2301.    
  2302.     function lib.placeRoof(R, outward)
  2303.         for i = 1, R.length + 2 do
  2304.             if R.useBlockType:find("stairs") ~= nil then
  2305.                 T:place("stairs", "up")
  2306.             else
  2307.                 T:go("C0", false, 0, false, R.useBlockType)
  2308.             end
  2309.             if i < R.length + 2 then
  2310.                 if outward then
  2311.                     T:go("L1F1R1")
  2312.                 else
  2313.                     T:go("R1F1L1")
  2314.                 end
  2315.             end
  2316.         end
  2317.     end
  2318.    
  2319.     function lib.placeGable(R, outward)
  2320.         local width = R.width
  2321.         for h = 1, R.height do
  2322.             for w = 1, width do
  2323.                 T:go("C1")
  2324.                 if w < width then
  2325.                     if outward then
  2326.                         T:go("L1F1R1")
  2327.                     else
  2328.                         T:go("R1F1L1")
  2329.                     end
  2330.                 end
  2331.             end
  2332.             if h < R.height then
  2333.                 if outward then
  2334.                     T:go("R1F1L1U1")
  2335.                 else
  2336.                     T:go("L1F1R1U1")
  2337.                 end
  2338.             end
  2339.             width = width - 2
  2340.             outward = not outward
  2341.         end
  2342.         return outward
  2343.     end
  2344.    
  2345.     local outward = true
  2346.     -- go to centre of end wall if odd no, or half width if even
  2347.     R.height = math.floor(R.width / 2)
  2348.     local isOdd = R.width % 2 == 1
  2349.    
  2350.     utils.goBack(1)
  2351.     T:go("R1F"..R.height - 1 .."U"..R.height - 1)   -- top of roof, under top layer
  2352.     for h = 1, R.height + 1 do                      -- place tiles on left side of roof
  2353.         lib.placeRoof(R, outward)
  2354.         if h < R.height + 1 then
  2355.             utils.goBack(1)
  2356.             T:down(1)
  2357.             outward = not outward
  2358.         end
  2359.     end
  2360.     if isOdd then
  2361.         T:go("F"..R.height + 2 .."R2U"..R.height)
  2362.     else
  2363.         T:go("F"..R.height + 1 .."R2U"..R.height)
  2364.     end
  2365.     for h = 1, R.height + 1 do                      -- place tiles on right side of roof
  2366.         lib.placeRoof(R, outward)
  2367.         if h < R.height + 1 then
  2368.             utils.goBack(1)
  2369.             T:down(1)
  2370.             outward = not outward
  2371.         end
  2372.     end
  2373.     -- gable ends
  2374.     if outward then
  2375.         T:go("F1R1U1")
  2376.     else
  2377.         T:go("F1L1U1")
  2378.     end
  2379.     outward = lib.placeGable(R, outward)
  2380.     T:go("F2R2 C1R2F"..R.length - 1 .."D"..R.height - 1)
  2381.     if outward then
  2382.         T:go("R1F"..R.height - 1 .."R1")
  2383.     else
  2384.         T:go("L1F"..R.height - 1 .."L1")
  2385.     end
  2386.     outward = not outward
  2387.     outward = lib.placeGable(R, outward)
  2388.     if isOdd then
  2389.         if outward then
  2390.             T:go("L1F1R1U2F1")
  2391.         else
  2392.             T:go("R1F1L1U2F1")
  2393.         end
  2394.         for i = 1, R.length do
  2395.             T:go("C2F1", false, 0, false)
  2396.         end
  2397.         for i = 1, R.length + 2 do
  2398.             utils.goBack(1)
  2399.             T:place("slab", "forward")
  2400.         end
  2401.     end
  2402.     while turtle.down() do end
  2403.    
  2404.     return {}
  2405. end
  2406.  
  2407. local function buildPitchedRoof(R)
  2408.     --[[
  2409.     stairs placement:
  2410.        _|   up
  2411.      
  2412.        T L  forward
  2413.        _
  2414.         |   down
  2415.  
  2416.     ]]
  2417.     local lib = {}
  2418.        
  2419.     function lib.placeRoofSection(length)
  2420.         -- starts facing centre of building
  2421.         for i = 1, length do
  2422.             if i < length then
  2423.                 if R.useBlockType:find("stairs") ~= nil then
  2424.                     T:place("stairs", "up")
  2425.                 else
  2426.                     T:go("C0", false, 0, false, R.useBlockType)
  2427.                 end
  2428.                 T:go("L1F1R1")
  2429.             end
  2430.         end
  2431.         -- ends facing centre of building
  2432.     end
  2433.    
  2434.     function lib.placeRoof(R, width, length)
  2435.         lib.placeRoofSection(length)
  2436.         T:go("R1")
  2437.         lib.placeRoofSection(width)
  2438.         T:go("R1")
  2439.         lib.placeRoofSection(length)
  2440.         T:go("R1")
  2441.         lib.placeRoofSection(width)
  2442.     end
  2443.    
  2444.     function lib.placeSlabs(length)
  2445.         -- add slabs at top
  2446.         T:go("U2F1L1")
  2447.         if length > 1 then
  2448.             T:forward(length - 3)
  2449.             for i = 1, length - 3 do
  2450.                 T:place("slab", "forward")
  2451.                 utils.goBack(1)
  2452.             end
  2453.             T:place("slab", "forward")
  2454.         else
  2455.             T:place("slab", "forward")
  2456.         end
  2457.         T:go("D2R1")
  2458.         utils.goBack(1)
  2459.     end
  2460.     --[[
  2461.     Turtle MUST be placed on left corner of shortest dimension
  2462.    
  2463.     ****   or T******
  2464.     ****      *******
  2465.     ****      *******
  2466.     ****
  2467.     T***
  2468.     shortest dimension is R.width
  2469.     if width is odd, ignore top layer as is only 1 block wide
  2470.     ]]
  2471.     local isWidthOdd, isLengthOdd, width, length = false, false, 0, 0
  2472.     R, isWidthOdd, isLengthOdd, width, length = utils.getRoofStats(R)
  2473.     T:go("F"..R.height - 1 .."R1F"..R.height - 1 .."U"..R.height - 1)       -- top of roof, under top layer
  2474.     if isWidthOdd then
  2475.         lib.placeSlabs(length)
  2476.     end
  2477.     for h = 1, R.height + 1 do                      -- place tiles on left side of roof
  2478.         lib.placeRoof(R, width, length)
  2479.         length = length + 2                         -- increase dimensions
  2480.         width = width + 2
  2481.         if h < R.height + 1 then
  2482.             utils.goBack(1)
  2483.             T:go("D1R1")
  2484.             utils.goBack(1)
  2485.         end
  2486.     end
  2487.     while turtle.down() do end
  2488.    
  2489.     return {}
  2490. end
  2491.  
  2492. local function clearAndReplantTrees() -- 25
  2493.     --[[ clear all trees in a rectangle area defined by walls, fences or non-dirt blocks
  2494.     replant with same type of sapling. If original tree 2 blocks wide, replant 4 if possible. ]]
  2495.    
  2496.     local lib = {}
  2497.    
  2498.     function lib.getSaplingFromLogType(log)
  2499.         --[[ get type of sapling to plant from log type ]]
  2500.         if log:find("oak") ~= nil then
  2501.             return "minecraft:oak_sapling"
  2502.         elseif log:find("spruce") ~= nil then
  2503.             return "minecraft:spruce_sapling"
  2504.         elseif log:find("birch") ~= nil then
  2505.             return "minecraft:birch_sapling"
  2506.         elseif log:find("jungle") ~= nil then
  2507.             return "minecraft:jungle_sapling"
  2508.         elseif log:find("acacia") ~= nil then
  2509.             return "minecraft:acacia_sapling"
  2510.         elseif log:find("dark_oak") ~= nil then
  2511.             return "minecraft:dark_oak_sapling"
  2512.         end
  2513.         return "sapling"
  2514.     end
  2515.    
  2516.     function lib.plantSapling(sapling, double)
  2517.         --[[ plant sapling(s) ]]
  2518.         if sapling == "" or sapling == nil then sapling = "sapling" end
  2519.         T:up(1)
  2520.         lib.suck()
  2521.         if double then  -- check if enough saplings
  2522.             --slotData.lastSlot, slotData.leastModifier, total, slotData = T:getItemSlot(sapling, -1)
  2523.             local a, b, total, _ = T:getItemSlot(sapling, -1)
  2524.             if total >= 4 then
  2525.                 for i = 1, 4 do
  2526.                     T:place(sapling, "down")
  2527.                     T:go("F1R1")
  2528.                 end
  2529.                 T:forward(1)        -- above pre-planted sapling
  2530.             else
  2531.                 if not T:place(sapling, "down") then
  2532.                     T:place("sapling", "down")
  2533.                 end
  2534.             end
  2535.         else
  2536.             if not T:place(sapling, "down") then
  2537.                 T:place("sapling", "down")
  2538.             end
  2539.         end
  2540.         turtle.select(1)
  2541.     end
  2542.        
  2543.     function lib.suck()
  2544.         --[[ Collect saplings, sticks and apples ]]
  2545.         turtle.select(1)
  2546.         turtle.suck()
  2547.         turtle.suckUp()
  2548.         turtle.suckDown()
  2549.     end
  2550.    
  2551.     function lib.turn(direction)
  2552.         --[[ change direction and return new value for direction ]]
  2553.         if direction == "r" then
  2554.             T:turnRight(1)
  2555.             direction = "l"
  2556.         else
  2557.             T:turnLeft(1)
  2558.             direction = "r"
  2559.         end
  2560.         return direction    -- will only change direction variable if return value is used
  2561.     end
  2562.    
  2563.     function lib.emptyInventory(blockTypeD)
  2564.         --[[ Empty all except 32 of each sapling and 1 chest ]]
  2565.         if blockTypeD == nil then
  2566.             blockTypeD = T:getBlockType("down")
  2567.         end
  2568.         if blockTypeD:find("chest") ~= nil or blockTypeD:find("barrel") ~= nil then
  2569.             -- empty logs, apples, sticks and all but 1 stack of each sapling type
  2570.             T:emptyInventorySelection("down", {"chest", "oak_sapling", "birch_sapling", "spruce_sapling", "acacia_sapling", "jungle_sapling","dark_oak_sapling"},{1, 32, 32, 32, 32, 32, 32})
  2571.             return true
  2572.         else
  2573.             return false
  2574.         end
  2575.     end
  2576.    
  2577.     function lib.moveDown(blockTypeD)
  2578.         --[[ move down until hit ground. Break leaves and continue ]]
  2579.         if blockTypeD == nil then
  2580.             blockTypeD = T:getBlockType("down")
  2581.         end
  2582.         while blockTypeD == "" or blockTypeD:find("leaves") ~= nil do   -- move down, breaking leavse
  2583.             T:down(1)
  2584.             lib.suck()
  2585.             blockTypeD = T:getBlockType("down")
  2586.         end
  2587.         return blockTypeD
  2588.     end
  2589.    
  2590.     function lib.moveForward()
  2591.         --[[ Move forward 1 block only, go down to ground while air or leaves below ]]
  2592.         local blockTypeF = T:getBlockType("forward")
  2593.         local blockTypeD = T:getBlockType("down")
  2594.         if blockTypeF == "" or blockTypeF:find("leaves") ~= nil then    -- air or leaves ahead
  2595.             T:forward(1)                                                -- move forward, breaking leaves
  2596.             T:dig("up")                                                 -- remove leaves / low branches above to allow space for player
  2597.             lib.suck()
  2598.             blockTypeD = lib.moveDown()
  2599.             if not lib.emptyInventory(blockTypeD) then                  -- check if above a corner chest / barrel
  2600.                 if lib.isBorder(blockTypeD) then                        -- not above chest so check if above border
  2601.                     return false, blockTypeD                            -- above a border block so stop
  2602.                 end
  2603.             end
  2604.             blockTypeF = T:getBlockType("forward")
  2605.             return true, blockTypeF                                     -- moved ok, could be air or block in front
  2606.         end
  2607.         return false, blockTypeF                                        -- did not move, obstacle in front NOT leaves or air
  2608.     end
  2609.    
  2610.     function lib.moveUp(blockTypeF)
  2611.         --[[ Move up until air in front (dig leaves / harvest tree) ]]
  2612.         if blockTypeF == nil then
  2613.             blockTypeF = T:getBlockType("forward")
  2614.         end
  2615.         while blockTypeF:find("dirt") ~= nil or
  2616.               blockTypeF:find("grass_block") ~= nil or
  2617.               T:isVegetation(blockTypeF) do -- go up while dirt, grass-block or any vegetation in front
  2618.             T:up(1)
  2619.             blockTypeF = T:getBlockType("forward")
  2620.             if blockTypeF:find("log") ~= nil then
  2621.                 lib.harvestTree(blockTypeF)
  2622.                 return T:getBlockType("forward")
  2623.             elseif blockTypeF:find("leaves") ~= nil then
  2624.                 T:dig("forward")
  2625.                 return ""
  2626.             end
  2627.         end
  2628.         return blockTypeF   -- should be "" (air) or any border block
  2629.     end
  2630.    
  2631.     function lib.harvestTree(blockTypeF)
  2632.         --[[ Fell tree, returns true if double size ]]
  2633.         -- clsTurtle.harvestTree(extend, craftChest, direction)
  2634.         local saplingType = lib.getSaplingFromLogType(blockTypeF)
  2635.         local double = T:harvestTree(false, false, "forward")   -- assume single tree, will auto-discover
  2636.         lib.plantSapling(saplingType, double)
  2637.     end
  2638.    
  2639.     function lib.safeMove()
  2640.         --[[ move forward until border reached. loop breaks at that point ]]
  2641.         local blockTypeF = ""
  2642.         local success = true
  2643.         while success do
  2644.             success, blockTypeF = lib.moveForward()             -- move forward 1 block, return block type ahead
  2645.             if not success then                                 -- did not move forwards, block in the way: either log, dirt/grass, border block or vegetation
  2646.                 if blockTypeF:find("log") then                  -- tree found
  2647.                     lib.harvestTree(blockTypeF)
  2648.                     success = true                              -- block (log) removed, try again
  2649.                 else
  2650.                     success = not lib.isBorder(blockTypeF)      -- Is at border?: if is at border success = false so loop stops
  2651.                     if success then                             -- Not at border. Dirt/grass vegetation in front
  2652.                         blockTypeF = lib.moveUp(blockTypeF)     -- move up until leaves/log/air
  2653.                         success = not lib.isBorder(blockTypeF)  -- Is at border?: if is at border success = false so loop stops
  2654.                         if success then                         -- keep moving forward
  2655.                             if blockTypeF:find("log") then      -- tree found
  2656.                                 lib.harvestTree(blockTypeF)
  2657.                             end
  2658.                                                                 -- else blockTypeF is air/leaves  border has been checked
  2659.                         end
  2660.                     end
  2661.                 end
  2662.             end                                                 -- else success = true, 1 block moved so continue
  2663.         end
  2664.     end
  2665.    
  2666.     function lib.isBorder(blockType)
  2667.         --[[ Is the block log, dirt, grass_block, vegetation: non-border, or other:border]]
  2668.         if blockType == nil then                    -- not passed as parameter
  2669.             blockType = T:getBlockType("forward")
  2670.         end
  2671.         if blockType == "" then                     -- air ahead: not border
  2672.             return false, ""
  2673.         else                                        -- could be border or other
  2674.             if blockType:find("dirt") ~= nil or blockType:find("grass_block") ~= nil or blockType:find("log") ~= nil then -- either dirt, grass block or log
  2675.                 return false, blockType             -- dirt, grass, log: not border
  2676.             end
  2677.             if T:isVegetation(blockType) then       -- vegetation found: not border
  2678.                 return false, blockType
  2679.             end
  2680.         end
  2681.         return true, blockType                      -- dirt, grass_block, log and vegetation eliminated:must be border
  2682.     end
  2683.    
  2684.     function lib.inPosition()
  2685.         --[[ check if in lower left corner ]]
  2686.         local inPosition = true         -- assume correct
  2687.         if not turtle.detectDown() then -- hanging in mid-air
  2688.             return false
  2689.         end
  2690.         T:turnLeft(1)
  2691.         if lib.isBorder() then
  2692.             -- so far so good
  2693.             T:turnLeft(1)
  2694.             if not lib.isBorder() then  -- not in correct place
  2695.                 inPosition = false
  2696.             end
  2697.             T:turnRight(2)              -- return to original position
  2698.         else
  2699.             inPosition = false
  2700.             T:turnRight(1)              -- return to original position
  2701.         end
  2702.         return inPosition
  2703.     end
  2704.    
  2705.     function lib.findBorder()
  2706.         --[[ assume started after reset. if log above harvest tree else return to ground. Find starting corner]]
  2707.         local blockType = T:getBlockType("up")                  -- dig any logs above, return to ground
  2708.         local log = "sapling"
  2709.         if blockType:find("log") ~= nil then                    -- originally felling a tree so complete it
  2710.             log = lib.getSaplingFromLogType(blockType)
  2711.             local double = T:harvestTree(false, false, "up")    -- assume single tree, will auto-discover
  2712.             lib.plantSapling(log, double)
  2713.         else                                                    -- no log above so go downm
  2714.             blockType = lib.moveDown()                          -- return to ground (or vegetation)
  2715.         end
  2716.         lib.safeMove()                                          -- move forward until border reached
  2717.         T:turnRight(1)
  2718.         lib.safeMove()                                          -- move forward until second border reached
  2719.         T:turnRight(1)                                          -- should now be in correct position
  2720.         lib.emptyInventory()                                    -- empty inventory if above a chest
  2721.     end
  2722.    
  2723.     local direction = "r"
  2724.     local blockTypeF = ""
  2725.     local success = false
  2726.     if not lib.inPosition() then
  2727.         lib.findBorder()
  2728.     end
  2729.     local secondBorderFound = false
  2730.     while not secondBorderFound do
  2731.         lib.safeMove()                                                      -- moves forward until reaches border forward or below
  2732.         lib.turn(direction)                                                 -- turn r or l. direction is not changed
  2733.         success, blockTypeF = lib.isBorder()                                -- no blockType passed as parameter so will return current block in new forward direction
  2734.         if success then
  2735.             secondBorderFound = true                                        -- game over
  2736.         elseif blockTypeF:find("log") ~= nil then                           -- tree in front
  2737.             lib.harvestTree(blockTypeF)
  2738.         elseif blockTypeF == "" or blockTypeF:find("leaves") ~= nil then    -- air or leaves in front
  2739.             T:forward(1)                                                    -- move forward 1 block
  2740.             lib.moveDown()                                                  -- go down if required
  2741.         elseif  blockTypeF:find("dirt") ~= nil or
  2742.                 blockTypeF:find("grass_block") ~= nil or
  2743.                 T:isVegetation(blockTypeF) then                             -- dirt, grass_block or vegetation in front
  2744.             blockTypeF = lib.moveUp(blockTypeF)                             -- move up until air or border ahead.
  2745.             if lib.isBorder(blockTypeF) then                                -- border ahead
  2746.                 secondBorderFound = true
  2747.             else                                                            -- air ahead                                   
  2748.                 T:forward(1)                                                -- move forward 1 block
  2749.             end
  2750.         end
  2751.         direction = lib.turn(direction)                                     -- turn r or l. direction is changed to opposite
  2752.     end
  2753.     lib.moveDown()                                                          -- return to ground level
  2754.     lib.emptyInventory()
  2755.     return {}
  2756. end
  2757.  
  2758. local function clearArea(R) -- 71, called from 22
  2759.     local evenWidth = false
  2760.     local evenHeight = false
  2761.     local loopWidth
  2762.     -- go(path, useTorch, torchInterval, leaveExisting)
  2763.     if R.width % 2 == 0 then
  2764.         evenWidth = true
  2765.         loopWidth = R.width / 2
  2766.     else
  2767.         loopWidth = math.ceil(R.width / 2)
  2768.     end
  2769.     if R.length % 2 == 0 then
  2770.         evenHeight = true
  2771.     end
  2772.     turtle.select(1)
  2773.     -- clear an area between 2 x 4 and 32 x 32
  2774.     -- if R.width is even no, then complete the up/down run
  2775.     -- if R.width odd no then finish at top of up run and reverse
  2776.     -- should be on flat ground, check voids below, harvest trees
  2777.     for x = 1, loopWidth do
  2778.         -- Clear first column (up)
  2779.         for y = 1, R.length do
  2780.             if R.useBlockType == "dirt" then
  2781.                 if not turtle.detectDown() then
  2782.                     T:place("minecraft:dirt", "down", true)
  2783.                 else --if not water, dirt, grass , stone then replace with dirt
  2784.                     blockType = T:getBlockType("down")
  2785.                     if blockType ~= "" then
  2786.                         if blockType ~= "minecraft:dirt" and blockType ~= "minecraft:grass_block" then
  2787.                             turtle.digDown()
  2788.                             T:place("minecraft:dirt", "down", true)
  2789.                         end
  2790.                     end
  2791.                 end
  2792.             end
  2793.             if y < R.length then
  2794.                 T:go("F1+1", false,0,false)
  2795.             end
  2796.         end
  2797.         -- clear second column (down)
  2798.         if x < loopWidth or (x == loopWidth and evenWidth) then -- go down if on R.width 2,4,6,8 etc
  2799.             T:go("R1F1+1R1", false,0,false)
  2800.             for y = 1, R.length do
  2801.                 if R.useBlockType == "dirt" then
  2802.                     if not turtle.detectDown() then
  2803.                         T:place("minecraft:dirt", "down", true)
  2804.                     else
  2805.                         blockType = T:getBlockType("down")
  2806.                         if blockType ~= "" then
  2807.                             if blockType ~= "minecraft:dirt" and blockType ~= "minecraft:grass_block" then
  2808.                                 turtle.digDown()
  2809.                                 T:place("minecraft:dirt", "down", true)
  2810.                             end
  2811.                         end
  2812.                     end
  2813.                 end
  2814.                 if y < R.length then
  2815.                     T:go("F1+1", false, 0, false)
  2816.                 end
  2817.             end
  2818.             if x < loopWidth then
  2819.                 T:go("L1F1+1L1", false,0,false)
  2820.             else
  2821.                 T:turnRight(1)
  2822.                 T:forward(R.width - 1)
  2823.                 T:turnRight(1)
  2824.             end
  2825.         else -- equals R.width but is 1,3,5,7 etc
  2826.             T:turnLeft(2) --turn round 180
  2827.             T:forward(R.length - 1)
  2828.             T:turnRight(1)
  2829.             T:forward(R.width - 1)
  2830.             T:turnRight(1)
  2831.         end
  2832.     end
  2833.     return {}
  2834. end
  2835.  
  2836. local function clearRectangle(R) -- 72
  2837.     --local function clearRectangle(width, length, up, down)
  2838.     -- height = 0: one level, 1 = +up, 2 = +down, 3 = +up/down
  2839.     local lib = {}
  2840.     function lib.UpDown(length)
  2841.         for l = 1, length do
  2842.             --T:go("x0x2F1x0x2")
  2843.             turtle.digDown()                -- dig below
  2844.             while turtle.digUp() do end     -- dig up including gravity blocks
  2845.             while not turtle.forward() do   -- if no block in front, moves forward
  2846.                 turtle.dig()                -- block in front, so dig it
  2847.             end
  2848.             turtle.digDown()                -- now moved forward so dig down again
  2849.             while turtle.digUp() do end     -- dig up again including gravity blocks
  2850.         end
  2851.     end
  2852.    
  2853.     function lib.Up(length)
  2854.         for l = 1, length do
  2855.             --T:go("x0F1x0")
  2856.             while turtle.digUp() do end
  2857.             while not turtle.forward() do   -- if no block in front, moves forward
  2858.                 turtle.dig()                -- block in front, so dig it
  2859.             end
  2860.             while turtle.digUp() do end
  2861.         end
  2862.     end
  2863.    
  2864.     function lib.Down(length)
  2865.         for l = 1, length do
  2866.             --T:go("x2F1x2")
  2867.             turtle.digDown()
  2868.             while not turtle.forward() do   -- if no block in front, moves forward
  2869.                 turtle.dig()                -- block in front, so dig it
  2870.             end
  2871.             turtle.digDown()
  2872.         end
  2873.     end
  2874.    
  2875.     function lib.Forward(length)
  2876.         T:forward(length)
  2877.     end
  2878.    
  2879.     -- could be 1 wide x xx R.length (trench) R.up and return
  2880.     -- could be 2+ x 2+
  2881.     -- even no of runs return after last run
  2882.     -- odd no of runs forward, back, forward, reverse and return
  2883.     turtle.select(1)
  2884.     if R.width == 1 then                    -- single block trench ahead only
  2885.         if R.up and R.down then             -- single block wide trench dig R.up and R.down = 3 blocks deep
  2886.             lib.UpDown(R.length - 1)
  2887.         elseif R.up then                    -- single block wide trench dig R.up = 2 blocks deep
  2888.             lib.Up(R.length - 1)
  2889.         elseif R.down then                  -- single block wide trench dig R.down = 2 blocks deep
  2890.             lib.Down(R.length - 1)
  2891.         else                                -- single block wide = 1 block deep
  2892.             lib.Forward(R.length - 1)
  2893.         end
  2894.         T:turnRight(2)                      -- turn at the top of the run
  2895.         T:forward(R.length - 1)             -- return to start
  2896.         T:turnRight(2)                      -- turn round to original position
  2897.     else                                    -- R.width 2 or more blocks
  2898.         local iterations = 0                -- R.width = 2, 4, 6, 8 etc
  2899.         if R.width % 2 == 1 then            -- R.width = 3, 5, 7, 9 eg R.width 7
  2900.             iterations = (R.width - 1) / 2  -- iterations 1, 2, 3, 4 for widths 3, 5, 7, 9
  2901.         else
  2902.             iterations = R.width / 2        -- iterations 1, 2, 3, 4 for widths 2, 4, 6, 8
  2903.         end
  2904.         for i = 1, iterations do            -- eg 3 blocks wide, iterations = 1
  2905.             if R.up and R.down then                         -- dig R.up and R.down
  2906.                 lib.UpDown(R.length - 1)
  2907.                 T:go("x0x2R1F1x0x2R1x0x2")              -- turn round
  2908.                 lib.UpDown(R.length - 1)
  2909.             elseif R.up then                                -- dig R.up
  2910.                 lib.Up(R.length - 1)
  2911.                 T:go("x0R1F1x0R1x0")
  2912.                 lib.Up(R.length - 1)
  2913.             elseif R.down then                          -- dig R.down
  2914.                 lib.Down(R.length - 1)
  2915.                 T:go("x2R1F1x2R1x2")
  2916.                 lib.Down(R.length - 1)
  2917.             else                                        -- no digging R.up or R.down
  2918.                 lib.Forward(R.length - 1)
  2919.                 T:go("R1F1R1")
  2920.                 lib.Forward(R.length - 1)
  2921.             end
  2922.             -- if 1 less than end, reposition for next run
  2923.             if i < iterations then
  2924.                 T:go("L1F1L1", false, 0, false)
  2925.             end
  2926.         end
  2927.         if R.width % 2 == 1 then  -- additional run and return to base needed
  2928.             T:go("L1F1L1", false, 0, false)
  2929.             if R.up and R.down then
  2930.                 lib.UpDown(R.length - 1)
  2931.             elseif R.up then
  2932.                 lib.Up(R.length - 1)
  2933.             elseif R.down then
  2934.                 lib.Down(R.length - 1)
  2935.             else
  2936.                 lib.Forward(R.length - 1)
  2937.             end
  2938.             T:turnRight(2)
  2939.             T:forward(R.length - 1)
  2940.         end
  2941.         T:go("R1F"..R.width - 1 .."R1", false, 0, false)
  2942.     end
  2943.     return {}
  2944. end
  2945.  
  2946. local function clearPerimeter(R) -- 74
  2947.     local lib = {}
  2948.     function lib.UpDown(length)
  2949.         for l = 1, length do
  2950.             T:go("x0x2F1x0x2")
  2951.         end
  2952.     end
  2953.    
  2954.     function lib.Up(length)
  2955.         for l = 1, length do
  2956.             T:go("x0F1x0")
  2957.         end
  2958.     end
  2959.    
  2960.     function lib.Down(length)
  2961.         for l = 1, length do
  2962.             T:go("x2F1x2")
  2963.         end
  2964.     end
  2965.    
  2966.     function lib.Forward(length)
  2967.         T:forward(length)
  2968.     end
  2969.    
  2970.     if R.up and R.down then
  2971.         for i = 1, 2 do
  2972.             lib.UpDown(R.length - 1)
  2973.             T:turnRight(1)
  2974.             lib.UpDown(R.width - 1)
  2975.             T:turnRight(1)
  2976.         end
  2977.     elseif R.up then
  2978.         for i = 1, 2 do
  2979.             lib.Up(R.length - 1)
  2980.             T:turnRight(1)
  2981.             lib.Up(R.width - 1)
  2982.             T:turnRight(1)
  2983.         end
  2984.     elseif R.down then
  2985.         for i = 1, 2 do
  2986.             lib.Down(R.length - 1)
  2987.             T:turnRight(1)
  2988.             lib.Down(R.width - 1)
  2989.             T:turnRight(1)
  2990.         end
  2991.     else
  2992.         for i = 1, 2 do
  2993.             lib.Forward(R.length - 1)
  2994.             T:turnRight(1)
  2995.             lib.Forward(R.width - 1)
  2996.             T:turnRight(1)
  2997.         end
  2998.     end
  2999.     return {}
  3000. end
  3001.  
  3002. local function clearBuilding(R) -- 75, 43
  3003.     --[[
  3004.     menu item: Clear hollow structure
  3005.     Clear the outer shell of a building, leaving inside untouched. Optional floor/ceiling removal
  3006.     if outside active area will have already moved in
  3007.     R.direction = "up" or "down" as direction of travel
  3008.     R.up = true / false to dig up as well
  3009.     R.down = true / false to dig down as well
  3010.     R.data.ceiling = true / false to remove ceiling
  3011.     R.data.floor = true / false to remove floor
  3012.     examples use a 5 x 5 x 7 cube
  3013.     ]]
  3014. --utils.printR(R)                                   -- debug values in R
  3015.     local height = 1                                -- current level of turtle
  3016.     local cleared = false
  3017.     R.silent = true
  3018.     R.up = false
  3019.     R.down = false
  3020.     -- check if floor/ceiling is to be cleared
  3021.     if R.direction == "up" then                     -- from bottom to top
  3022.         if R.data.floor then
  3023.             clearRectangle(R)
  3024.             cleared = true
  3025.         end
  3026.     else                                            -- from top to bottom
  3027.         if R.data.ceiling then
  3028.             clearRectangle(R)
  3029.             cleared = true
  3030.         end
  3031.     end
  3032.    
  3033.     if R.height == 1 and not cleared then           -- only 1 layer, perimeter only eg R.height = 7
  3034.         clearPerimeter(R)
  3035.     elseif R.height == 2 and not cleared then       -- only 2 layers, perimeter only
  3036.         if R.direction == "up" then                 -- from bottom to top
  3037.             R.up = true
  3038.         else
  3039.             R.down = true
  3040.         end
  3041.         clearPerimeter(R)
  3042.     elseif R.height >= 3 then                       -- 3 or more levels, floor/ceiling already assessed eg R.height = 7
  3043.         R.up = true
  3044.         R.down = true
  3045.         height = height + utils.move(R, 1)          -- move up/down 1 block for first layer eg height: 1 + 1 = 2
  3046.         -- height h + 3 R.height   loop
  3047.         --  2       5       5       end
  3048.         --  2       5       7       +
  3049.         --  5       8       7       end
  3050.         repeat         
  3051.             clearPerimeter(R)                       -- runs at least once, removes 3 layers: 1,2,3
  3052.             local move = 3
  3053.             --if height + 3 < R.height then         -- range with min 3(0), 4(1), 5(2), 6(3), 7(4), 8(5), 9(6) etc         
  3054.                 --height = height + utils.move(R, move) -- move up/down 3 blocks eg height 5, 8, 11, 14
  3055.             --else
  3056.                 --move = R.height - height - 1
  3057.                 --height = height + utils.move(R, move) -- move up/down 1-2 blocks eg height = 2 + 3 = 5
  3058.             --end
  3059.            
  3060.             if height + 3 >= R.height then          -- range with min 3(0), 4(1), 5(2), 6(3), 7(4), 8(5), 9(6) etc         
  3061.                 move = R.height - height
  3062.             end
  3063.             height = height + utils.move(R, move)   -- move up/down 1-2 blocks eg height = 2 + 3 = 5
  3064.         until height > R.height - 3             -- min 3 levels eg height = 2, R.height - 3 --> 7 - 3 = 4, 4 - 3 = 1
  3065.         R.up = false
  3066.         R.down = false 
  3067.         if height == R.height then                      -- already on top/bottom layer eg height = 5, R.height = 5
  3068.             if R.direction == "up" then                     -- reverse digging direction. Up = move to top and dig down
  3069.                 R.down = true
  3070.             else                                        -- reverse digging direction. Up = move to floor and dig up
  3071.                 R.up = true
  3072.             end
  3073.             clearPerimeter(R)
  3074.         else
  3075.             if R.direction == "up" then                     -- reverse digging direction. Up = move to top and dig down
  3076.                 R.down = true
  3077.             else                                        -- reverse digging direction. Up = move to floor and dig up
  3078.                 R.up = true
  3079.             end
  3080.  
  3081.             clearPerimeter(R)
  3082.         end
  3083.     end
  3084.    
  3085.     R.up = false
  3086.     R.down = false
  3087.     if R.direction == "up" then
  3088.         if R.data.ceiling then      -- from bottom to top
  3089.             clearRectangle(R)
  3090.         end
  3091.     elseif R.direction == "down" then
  3092.         if R.data.floor then    -- from top to bottom
  3093.             clearRectangle(R)
  3094.         end
  3095.     end
  3096.    
  3097.     if height > 1 then
  3098.         utils.move(R, height - 1, true) -- reverse direction
  3099.     end
  3100.    
  3101.     return {}
  3102. end
  3103.  
  3104. local function clearMineshaft(R, equippedRight, equippedLeft, inInventory) -- 19
  3105.     local lib = {}
  3106.    
  3107.     function lib.checkCobweb(directions, inInventory)
  3108.         assert(type(directions) == "table", "directions must be a table eg {'up','down'}")
  3109.         for _, direction in ipairs(directions) do
  3110.             local blockType = T:getBlockType(direction)
  3111.             if inInventory == "minecraft:diamond_sword" then -- using a sword
  3112.                 local side = "left"
  3113.                
  3114.                 if blockType:find("cobweb") ~= nil then
  3115.                     --clsTurtle.equip(self, side, useItem, useDamage)
  3116.                     if equippedRight == "minecraft:diamond_pickaxe" then
  3117.                         side = "right"
  3118.                     end
  3119.                     T:equip(side, "minecraft:diamond_sword")
  3120.                     T:dig(direction)
  3121.                     T:equip(side, "minecraft:diamond_pickaxe")
  3122.                 elseif blockType ~= "" then
  3123.                     T:dig(direction)
  3124.                 end
  3125.             else
  3126.                 T:dig(direction)
  3127.             end
  3128.         end
  3129.     end
  3130.  
  3131.     -- check position by rotating until facing away from wall
  3132.     length = 0
  3133.     torch = 0
  3134.     turns = 0
  3135.     while not turtle.detect() do
  3136.         T:turnRight(1)
  3137.         turns = turns + 1
  3138.         if turns > 4 then
  3139.             return {"I am not facing a wall. Unable to continue"}
  3140.         end
  3141.     end
  3142.     T:turnRight(2)
  3143.     -- move forward until obstructed, digging up/down. place torches
  3144.     while not turtle.detect() do
  3145.         lib.checkCobweb({"up","down"}, inInventory) -- dig cobweb or any other block up/down
  3146.         length = length + 1
  3147.         torch = torch + 1
  3148.         if torch == R.torchInterval then
  3149.             torch = 0
  3150.             T:place("minecraft:torch", "down", false) ---(self, blockType, damageNo, direction, leaveExisting, signText)
  3151.         end
  3152.         lib.checkCobweb({"forward"}, inInventory) -- dig cobweb or any other block in front
  3153.         T:forward(1)
  3154.     end
  3155.     -- turn right, forward, right, return to start with up/down dig
  3156.     T:go("R1")
  3157.     lib.checkCobweb({"up","down"}, inInventory) -- dig cobweb or any other block up/down
  3158.     T:go("F1R1")
  3159.     for i = 1, length, 1 do
  3160.         lib.checkCobweb({"up", "down", "forward"}, inInventory) -- dig cobweb or any other block up/down/forward
  3161.         T:forward(1)
  3162.     end
  3163.     -- move to other wall and repeat.
  3164.     T:go("R1")
  3165.     lib.checkCobweb({"up","down"}, inInventory) -- dig cobweb or any other block up/down
  3166.     T:go("F1")
  3167.     lib.checkCobweb({"up","down"}, inInventory) -- dig cobweb or any other block up/down
  3168.     T:go("F1R1")
  3169.     lib.checkCobweb({"up","down"}, inInventory) -- dig cobweb or any other block up/down
  3170.     for i = 1, length, 1 do
  3171.         lib.checkCobweb({"up", "down", "forward"}, inInventory) -- dig cobweb or any other block up/down/forward
  3172.         T:forward(1)
  3173.     end
  3174.     lib.checkCobweb({"up", "down"}, inInventory) -- dig cobweb or any other block up/down/forward
  3175.    
  3176.     return {}
  3177. end
  3178.  
  3179. local function clearMonumentLayer(R) -- 85
  3180.     R.up = true
  3181.     R.down = true
  3182.     R.silent = true
  3183.     if R.subChoice == 0 then
  3184.         R.up = false
  3185.         R.down = false
  3186.     end
  3187.     -- send turtle down until it hits bottom
  3188.     -- then clear rectangle of given size
  3189.     -- start above water, usually on cobble scaffold above monument
  3190.     if T:detect("down") then -- in case not over wall
  3191.         T:forward(1)
  3192.     end
  3193.     local height = 1
  3194.     -- go down until solid block detected
  3195.     while utils.clearVegetation("down") do
  3196.         T:down(1)
  3197.         height = height + 1
  3198.     end
  3199.     T:down(1)
  3200.     height = height + 1
  3201.     clearRectangle(R)
  3202.     T:up(height - 1)
  3203.    
  3204.     return {}
  3205. end
  3206.  
  3207. local function clearWaterPlants(R) -- 87
  3208.     local lib = {}
  3209.    
  3210.     function lib.getLength()
  3211.         local length = 1
  3212.         print("Checking water length")
  3213.         while utils.clearVegetation("forward") do
  3214.             T:forward(1)
  3215.             length = length + 1
  3216.         end
  3217.         for i = 1, length do
  3218.             turtle.back()
  3219.         end
  3220.         return length
  3221.     end
  3222.    
  3223.     function lib.clearDown(D)
  3224.         while utils.clearVegetation("down") do --clears any grass or sea plants, returns true if air or water, bubble column or ice
  3225.             T:down(1)
  3226.             D.depth = D.depth + 1
  3227.         end
  3228.         -- if slab at bottom, replace with solid block
  3229.         D.blockType = T:getBlockType("down")
  3230.         if D.blockType:find("slab") ~= nil then
  3231.             T:go("C2")
  3232.         end
  3233.         if D.depth > D.maxDepth then
  3234.             D.maxDepth = D.depth
  3235.         end
  3236.        
  3237.         return D
  3238.     end
  3239.    
  3240.     function lib.clearLength(D)
  3241.         local moves = 0
  3242.         local blockHeight = 1
  3243.         D.floorLength = 0 -- reset
  3244.         while moves < D.length - 1 do
  3245.             if utils.clearVegetation("forward") then
  3246.                 T:forward(1)
  3247.                 moves = moves + 1
  3248.                 if turtle.detectUp() then -- could be on monument going under arch, or faulty monument floor
  3249.                     if not D.monumentArchesClear then
  3250.                         D = lib.checkArches(D)
  3251.                     end
  3252.                 end
  3253.                 local temp = D.depth
  3254.                 D = lib.clearDown(D) -- go down if in water/air
  3255.                 if blockHeight == 4 and D.depth - temp == 3 then -- could be just gone over 3 block high column in ocean monument
  3256.                     if not D.monumentTopClear then
  3257.                         D = lib.checkTop(D)
  3258.                     end
  3259.                     blockHeight = 1
  3260.                 end
  3261.                 if D.useBlockType == "prismarine" then
  3262.                     if D.blockType:find("prismarine") == nil then
  3263.                         -- on monument floor so restrict forward movement to 8 blocks
  3264.                         D.floorLength = D.floorLength + 1
  3265.                         if D.floorLength == 8 then
  3266.                             D.blockType = T:getBlockType("forward")
  3267.                             while D.blockType:find("prismarine") == nil do
  3268.                                 T:up(1)
  3269.                                 D.blockType = T:getBlockType("forward")
  3270.                             end
  3271.                             D.floorLength = 0
  3272.                         end
  3273.                     end
  3274.                 end
  3275.                 if moves >= D.length - 1 then
  3276.                     D.width = D.width + 1   -- another length completed so increase width
  3277.                     return D
  3278.                 end
  3279.             else -- block in front
  3280.                 blockHeight = 1
  3281.                 local waterAbove = utils.clearVegetation("up")
  3282.                 local waterAhead = utils.clearVegetation("forward") -- true if air/water in front
  3283.                 while not waterAhead do     -- solid block in front
  3284.                     if waterAbove then      -- move up
  3285.                         T:up(1)
  3286.                         D.depth = D.depth - 1
  3287.                         blockHeight = blockHeight + 1
  3288.                         if D.depth < 1 then
  3289.                             D.width = D.width + 1   -- another length completed so increase width
  3290.                             return D
  3291.                         end
  3292.                     else                    -- block above so go back
  3293.                         while not waterAbove do
  3294.                             utils.goBack()  --  returns true if no object behind, but moves anyway
  3295.                             waterAbove = utils.clearVegetation("up")
  3296.                             moves = moves - 1
  3297.                             if moves == 0 then
  3298.                                 T:up(1)
  3299.                                 D.depth = D.depth - 1
  3300.                                 waterAbove = utils.clearVegetation("up")
  3301.                                 while not waterAbove do
  3302.                                     T:up(1)
  3303.                                     D.depth = D.depth - 1
  3304.                                 end
  3305.                             end
  3306.                         end
  3307.                         -- go up 1 to prevent loop
  3308.                         T:up(1)
  3309.                         D.depth = D.depth - 1
  3310.                     end
  3311.                     waterAbove = utils.clearVegetation("up")
  3312.                     waterAhead = utils.clearVegetation("forward")
  3313.                 end
  3314.             end
  3315.         end
  3316.         D.width = D.width + 1   -- another length completed so increase width
  3317.         D.maxDepth = D.maxDepth + 1 -- +1 to allow for starting pos in top layer
  3318.        
  3319.         return D
  3320.     end
  3321.    
  3322.     function lib.checkArches(D)
  3323.         -- gone under a block so could be monument arch (6)
  3324.         if T:getBlockType("up"):find("prismarine") ~= nil then -- confirm on monument, not previously cleared
  3325.             local direction = ""    -- initialise direction
  3326.             T:go("B1U2F1R1")
  3327.             if T:getBlockType("forward"):find("prismarine") ~= nil then -- correct guess
  3328.                 direction = "R"
  3329.             else    -- wrong direction. turn round and check other side
  3330.                 T:go("R2")
  3331.                 if T:getBlockType("forward"):find("prismarine") ~= nil then
  3332.                     direction = "L"
  3333.                 end
  3334.             end
  3335.             local path = "U1F1 U1F3 D1F1 D1R2 U2F5 D2"
  3336.             --for i = 1, 6 do
  3337.             T:go(path) -- clears arch top 1 and returns
  3338.             T:go(direction.."1F6"..direction.."1")
  3339.             T:go(path) -- clears arch top 2 and returns
  3340.             T:go(direction.."1F6"..direction.."1")
  3341.             T:go(path) -- clears arch top 3 and returns
  3342.             T:go(direction.."1F9"..direction.."1")
  3343.             T:go(path) -- clears arch top 4 and returns
  3344.             T:go(direction.."1F6"..direction.."1")
  3345.             T:go(path) -- clears arch top 5 and returns
  3346.             T:go(direction.."1F6"..direction.."1")
  3347.             T:go(path) -- clears arch top 6 and returns
  3348.            
  3349.             T:go(direction.."3F34"..direction.."2D2F1")
  3350.             D.monumentArchesClear =  true
  3351.         end
  3352.         return D
  3353.     end
  3354.    
  3355.     function lib.checkTop(D)
  3356.         -- gone over 3 block column so could be at the top of ocean monument
  3357.         if T:getBlockType("down"):find("prismarine") ~= nil then -- confirm on monument, not previously cleared
  3358.             local direction = ""    -- initialise direction
  3359.             T:go("U3R1")            -- up to top of column and try right side
  3360.             if T:getBlockType("forward"):find("prismarine") ~= nil then -- correct guess
  3361.                 direction = "L"
  3362.             else    -- wrong direction. turn round and check other side
  3363.                 T:go("R2")
  3364.                 if T:getBlockType("forward"):find("prismarine") ~= nil then
  3365.                     direction = "R"
  3366.                 end
  3367.             end
  3368.             if direction ~= "" then -- facing single block on layer 2
  3369.                 T:go("U1F6"..direction.."1F5".. direction.."1F5"..direction.."1F5"..direction.."1") -- clear 4 single blocks
  3370.                 T:go("F1U1"..direction.."1F1"..direction.."3") -- on top of monument
  3371.                 T:go("F3".. direction.."1F3"..direction.."1F3"..direction.."1F3"..direction.."3") -- clear top, face return
  3372.                 T:go("F2D5"..direction.."3B1")
  3373.             end
  3374.             D.monumentTopClear = true
  3375.         end
  3376.         return D
  3377.     end
  3378.    
  3379.     function lib.findBlockTypeEnd(D)
  3380.         D.blockType = "" -- reset
  3381.         repeat
  3382.             if utils.clearVegetation("forward") then
  3383.                 T:forward(1)
  3384.                 D = lib.clearDown(D) -- go down if in water/air, D.blockType is updated with floor type
  3385.             else -- block in front
  3386.                 local waterAbove = utils.clearVegetation("up")
  3387.                 local waterAhead = utils.clearVegetation("forward") -- true if air/water in front
  3388.                 while not waterAhead do     -- solid block in front
  3389.                     if waterAbove then      -- move up
  3390.                         T:up(1)
  3391.                     else                    -- block above so go back
  3392.                         while not waterAbove do
  3393.                             utils.goBack()  --  returns true if no object behind, but moves anyway
  3394.                             waterAbove = utils.clearVegetation("up")
  3395.                         end
  3396.                     end
  3397.                     waterAbove = utils.clearVegetation("up")
  3398.                     waterAhead = utils.clearVegetation("forward")
  3399.                 end
  3400.             end
  3401.         until D.blockType:find(D.useBlockType) == nil
  3402.         -- now above a non-prismarine block, facing away from monument
  3403.         T:turnRight(2) -- facing monument
  3404.         D.blockType = T:getBlockType("forward")
  3405.         while D.blockType:find(D.useBlockType) ~= nil do
  3406.             T:up(1)
  3407.             D.blockType = T:getBlockType("forward")
  3408.         end
  3409.         T:go("F1L1")
  3410.         D.blockType = T:getBlockType("down")
  3411.         while D.blockType:find(D.useBlockType) ~= nil do
  3412.             T:forward(1)
  3413.             D.blockType = T:getBlockType("down")
  3414.         end
  3415.         turtle.back()
  3416.         -- should now be at end of monument base
  3417.     end
  3418.    
  3419.     function lib.turn(D)
  3420.         local direction = "R"
  3421.         if not D.facingForward then
  3422.             direction = "L"
  3423.         end
  3424.         T:go(direction.. 1)
  3425.         if utils.clearVegetation("forward") then
  3426.             T:forward(1)
  3427.             --D.depth = D.depth + lib.clearDown(depth)
  3428.             D = lib.clearDown(D)
  3429.         else
  3430.             while not utils.clearVegetation("forward") do
  3431.                 T:up(1)
  3432.                 D.depth = D.depth - 1
  3433.             end
  3434.             T:forward(1)
  3435.         end
  3436.         T:go(direction.. 1)
  3437.         D.facingForward = not D.facingForward
  3438.        
  3439.         return D
  3440.     end
  3441.  
  3442.     local D = {}
  3443.     D.depth = 0
  3444.     D.maxDepth = 0
  3445.     D.blockType = ""
  3446.     D.useBlockType = R.useBlockType -- "" or "prismarine"
  3447.     D.length = 0    -- copy of R.length used in lib.clearLength
  3448.     D.width = 0     -- increased every iteration
  3449.     D.facingForward = true
  3450.     D.monumentTopClear = false
  3451.     D.monumentArchesClear = false
  3452.     D.inWater = false
  3453.     D.onWater = false
  3454.     D.inWater, D.onWater = utils.getWaterStatus()
  3455.     D.floorLength = 0 -- used on monument floor
  3456.    
  3457.     if R.data == "clearWaterPlants" then -- NOT monument corner discovery
  3458.         D.inWater, D.onWater = utils.startWaterFunction(D.onWater, D.inWater, 2, true) -- move into water
  3459.         if R.length == 0 then
  3460.             R.length = lib.getLength()
  3461.         end
  3462.         if R.width == 0 then
  3463.             T:turnRight(1)
  3464.             R.width = lib.getLength()
  3465.             T:turnLeft(1)
  3466.         end
  3467.     end
  3468.     D.length = R.length
  3469.     D = lib.clearDown(D) -- go down to floor, set depth, maxDepth, blockType
  3470.     if R.data == "clearWaterPlants" then -- NOT monument corner discovery
  3471.         if R.width == 1 then
  3472.             print("Single row clearing")
  3473.             D = lib.clearLength(D) --D.width also increased
  3474.         else
  3475.             while D.width < R.width do -- D.width starts at 0
  3476.                 -- now on floor, move along sea/river bed following contour
  3477.                 D = lib.clearLength(D) --D.width also increased
  3478.                 -- now turn and repeat
  3479.                 if D.width < R.width then  
  3480.                     D = lib.turn(D)
  3481.                     if D.depth <= 0 then
  3482.                         break
  3483.                     end
  3484.                 end
  3485.             end
  3486.         end
  3487.         -- finished so return to surface
  3488.         T:up(1) -- up 1 to check for water below
  3489.         while T:getBlockType("down"):find("water") ~= nil do
  3490.             T:up(1)
  3491.         end
  3492.         T:down(2) -- return to surface
  3493.         --[[while utils.clearVegetation("forward") do
  3494.             T:forward(1)
  3495.         end]]
  3496.     elseif R.data == "oceanMonumentColumns" then -- monument corner discovery
  3497.         -- this function used to find edge of monument base
  3498.         if D.blockType:find(D.useBlockType) ~= nil then
  3499.             lib.findBlockTypeEnd(D)
  3500.             return {""}
  3501.         else
  3502.             T:up(D.depth)
  3503.             return {"Prismarine not found on ocean floor"}
  3504.         end
  3505.     end
  3506.     if R.silent then
  3507.         return {D.maxDepth, R.length}
  3508.     else
  3509.         return {""}
  3510.     end
  3511. end
  3512.  
  3513. local function clearMountainSide(R) -- 78
  3514.     --[[
  3515.     First row              outward  l  s  f  mid    (length, start, finish, midPoint)
  3516.  
  3517.           >|*|*|           true                     lib.clearRow() moves: 7-5=2, midPoint starts at 2
  3518.           +|*|*|
  3519.           +|*|*|
  3520.           ^|*|*|        <  false    7  5  6  2      lib.clearRow() moves: 9-2=7, midPoint starts at -4 -1 = -5
  3521.            |*|*|*|      +
  3522.          |*|*|*|*|*|    +
  3523.       >  |*|*|*|*|*|    ^  true     9  2  6  -4     lib.getStartingLength(). Ends 1 block after block finish
  3524.       ^|*|*|*|*|*|*|*|*|                            starts here, moves up 1
  3525.         1 2 3 4 5 6 7 8                             block index
  3526.          
  3527.     Second row                outward l  s  f  mid
  3528.                  <            false
  3529.                  +
  3530.           |*|*|  +
  3531.        >|*|*|*|*|^            true
  3532.        +|*|*|*|*|
  3533.        +|*|*|*|*|
  3534.        ^|*|*|*|*|          <  false   6  5  6  2
  3535.         |*|*|*|*|*|*|      +
  3536.       |*|*|*|*|*|*|*|*|    +
  3537.       |*|*|*|>|*|*|*|*|    ^  true    7+5   5 -7    lib.getHalf(R); goBack(7)
  3538.     |*|*|*|*|*|*|*|*|*|*|*|                           starts at midPoint of previous row eg block 4
  3539.      0 1 2 3 4 5 6 7 8 9 10
  3540.     ]]
  3541.    
  3542.     -- variables declared before lib for them to be used within lib as is then in scope
  3543.     local turn = "R"
  3544.     local oTurn = "L"
  3545.     if R.subChoice == 1 then    -- remove left side
  3546.         turn = "L"
  3547.         oTurn = "R"
  3548.     end
  3549.     local outward = true        -- direction flag
  3550.    
  3551.     local lib = {}
  3552.    
  3553.     function lib.isAnyAbove(above)
  3554.         T:saveToLog("lib.isAnyAbove: "..utils.tableConcat(above, ", "), false)
  3555.         for k,v in ipairs(above) do
  3556.             if v then
  3557.                 T:saveToLog("lib.isAnyAbove Found: "..k , false)
  3558.                 return true
  3559.             end
  3560.         end
  3561.         return false
  3562.     end
  3563.        
  3564.     function lib.clearLevel(R, above)   -- eg 9, 2, 6, -4 from lib.getStartingLength OR 7, 5, 6, 3 from previous
  3565.         -- clearLevel always follows either lib.getStartingLength or a previous lib.clearLevel
  3566.         -- midPoint should be adjusted as turtle moves to reflect current row length
  3567.         if #above == 0 then return above, 1 end
  3568.         --local index = 0
  3569.         local minMoves = math.floor(#above / 2)
  3570.         local up, forward, down = lib.getDetect()
  3571.         if outward then                                     -- follow table indexes
  3572.             for x = 1, minMoves do                          -- clear first half
  3573.                 above[x] = up                               -- reset this with new value
  3574.                 T:go("x0x2F1")                              -- clear and move forward
  3575.                 up, forward, down = lib.getDetect()
  3576.                 --index = index + 1
  3577.             end
  3578.             for x = minMoves + 1, #above do                 -- check remaing half and clear
  3579.                 T:go("x0x2")                                -- clear above / below
  3580.                 if above[x] then                            -- is a block recorded as present?(now below)
  3581.                     above[x] = up                           -- reset this with new value
  3582.                     T:forward(1)                            -- move forward
  3583.                     up, forward, down = lib.getDetect()
  3584.                     --index = index + 1
  3585.                 else
  3586.                     break
  3587.                 end
  3588.             end
  3589.         else                                                -- iterate table in reverse
  3590.             --index = #above
  3591.             for x = #above, minMoves, -1 do                 -- clear first half
  3592.                 above[x] = up                               -- reset this with new value
  3593.                 T:go("x0x2F1")                              -- clear and move forward
  3594.                 up, forward, down = lib.getDetect()
  3595.                 --index = index - 1
  3596.             end
  3597.             for x = minMoves - 1, 1, -1 do                  -- check remaing half and clear
  3598.                 T:go("x0x2")                                -- clear up / down
  3599.                 if above[x] then                            -- is a block recorded as present?(now below)
  3600.                     above[x] = up                           -- reset this with new value
  3601.                     T:forward(1)                            -- move forward
  3602.                     up, forward, down = lib.getDetect()
  3603.                     --index = index - 1
  3604.                 else
  3605.                     break
  3606.                 end
  3607.             end
  3608.         end
  3609.         T:go("x0x2 F1R2 x0x2 F1")                           -- face opposite direction, delete blocks above and below
  3610.         outward = not outward                               -- switch direction flag
  3611.         return above                                        -- eg {false, true, true, true, false}
  3612.     end
  3613.    
  3614.     function lib.getDetect()
  3615.         return turtle.detectUp(),  turtle.detect(),  turtle.detectDown()
  3616.     end
  3617.    
  3618.     function lib.getStartingLength(R)
  3619.         --[[
  3620.             length of column by excavating blocks above, ahead or below
  3621.             Rotate 180 at end of run ready to return
  3622.             already 1 block above ground
  3623.         ]]
  3624.         local above = {}                                    -- empty table of boolean values
  3625.         local length = 0                                    -- used as counter                         
  3626.         T:forward(1)                                        -- start check 1 block ahead
  3627.         local up, forward, down = lib.getDetect()           -- check if anything around current block
  3628.         if up or forward or down then                       -- block found nearby: continue
  3629.             while up or forward or down do                  -- while blocks ahead / up / down move forward
  3630.                 table.insert(above, up)
  3631.                 T:go("x0x2F1")
  3632.                 up, forward, down = lib.getDetect()         -- check if anything around current block
  3633.                 length = length + 1
  3634.                 if length >= R.length then                  -- check if going out of range
  3635.                     T:saveToLog("lib.getStartingLength(R) : outward = "..tostring(outward).." MaxLength "..maxLength.." reached", false)
  3636.                     break
  3637.                 end
  3638.             end
  3639.             T:go("R2x0x2")                                  -- Rotate 180 and clear blocks above/below
  3640.         else                                                -- no blocks nearby: exit
  3641.             T:go("R2F1")                                    -- return to start position rotated 180
  3642.         end
  3643.         outward = not outward
  3644.  
  3645.         return above                                        -- above = {false, true, true, true, true, false, false}
  3646.     end
  3647.    
  3648.     function lib.firstRow(R)
  3649.         local height = 1                                    -- starts at ground level, but forced up 1
  3650.         T:up(1)
  3651.         local above = lib.getStartingLength(R)              -- clear the ground level and 1 above eg 9, 2, 5, 4
  3652.         if T:saveToLog("startLength: "..#above, true) then
  3653.             T:saveToLog(utils.tableConcat(above, ", "), false)
  3654.             utils.waitForInput()
  3655.         end                                                 -- at end of first row as already turned 180, outward set to false in getStartingLength
  3656.         while lib.isAnyAbove(above) do
  3657.             T:go("U3")  -- go up 3
  3658.             height = height + 3
  3659.             above = lib.clearLevel(R, above)    -- returns start and finish of blocks above, rotates 180
  3660.             if T:saveToLog("checking level: "..height, true) then
  3661.                 T:saveToLog(utils.tableConcat(above, ", "), false)
  3662.                 utils.waitForInput()
  3663.             end
  3664.         end                                                 -- first row all levels completed.
  3665.         T:down(height)                                      -- now on ground + 1, facing last column cleared.
  3666.        
  3667.         return above
  3668.     end
  3669.        
  3670.     function lib.deepCopy(tbl)
  3671.         local copy = {}
  3672.         for key, value in ipairs(tbl) do
  3673.             table.insert(copy, value)
  3674.         end
  3675.         return copy
  3676.     end
  3677.    
  3678.     function lib.getHalf(R, above)
  3679.         -- already 1 block above ground
  3680.         local maxLength = R.length
  3681.         local temp = {}
  3682.         local retValue = {}
  3683.         if #above > 0 then                              -- not empty table, so must be second half
  3684.             temp = lib.deepCopy(above)                  -- copy existing table
  3685.             above = {}                                  -- initialise above
  3686.         end
  3687.         local up, forward, down = lib.getDetect()
  3688.        
  3689.         while up or forward or down do                  -- while blocks ahead / up / down move forward
  3690.             T:go("x0x2F1")
  3691.             table.insert(above, up)
  3692.             up, forward, down = lib.getDetect()         -- check if anything around current block
  3693.            
  3694.             if #above >= math.floor(maxLength / 2) then -- check if going out of range
  3695.                 T:saveToLog("lib.getHalf(R) : outward = "..tostring(outward).." MaxLength "..maxLength.." reached", false)
  3696.                 T:go("x0x2")
  3697.                 break
  3698.             end
  3699.         end
  3700.         T:turnRight(2)                                  -- ready for next half or return
  3701.         outward = not outward
  3702.         if #temp > 0 then                               -- completing a second half measurement
  3703.             for i = #above, 1, -1 do
  3704.                 table.insert(retValue, above[i])        -- combine 2 tables into 1
  3705.             end
  3706.             for i = 1, #temp do
  3707.                 table.insert(retValue, temp[i])
  3708.             end
  3709.         else
  3710.             retValue = above
  3711.         end
  3712.         return retValue
  3713.     end
  3714.    
  3715.     function lib.nextRow(R)
  3716.         local height = 1
  3717.         T:saveToLog("lib.nextRow(R)", false)
  3718.         T:up(1)
  3719.         local pattern = turn.."1F1"..turn.."1"
  3720.         if not outward then
  3721.             pattern = oTurn.."1F1"..oTurn.."1"
  3722.         end
  3723.         T:go(pattern)
  3724.         T:saveToLog("    T:go("..pattern..")", false)
  3725.         outward = not outward -- reverse direction flag
  3726.         -- now in next vertical row
  3727.         local above = lib.getHalf(R, {})
  3728.         local index = 0
  3729.         if T:saveToLog("\t  first half Length: "..#above.." Enter", true) then
  3730.             T:saveToLog(utils.tableConcat(above, ", "), false)
  3731.             utils.waitForInput()
  3732.         end
  3733.         lib.returnToMidPoint(#above)                    -- return to starting point
  3734.         T:forward(1)       
  3735.         above = lib.getHalf(R, above)                   -- returns length - 1 eg 5, 4
  3736.         if T:saveToLog("\t  total length: "..#above.." Enter", true) then
  3737.             T:saveToLog(utils.tableConcat(above, ", "), false)
  3738.             utils.waitForInput()
  3739.         end
  3740.        
  3741.         while lib.isAnyAbove(above) do
  3742.             T:go("U3")              -- go up 3
  3743.             height = height + 3     -- increment height
  3744.             T:saveToLog("\tClear height loop: height = "..height, false)
  3745.             above = lib.clearLevel(R, above)    -- returns start and finish of blocks above
  3746.         end
  3747.         T:down(height)  -- now on ground + 1
  3748.         lib.returnToMidPoint(above)
  3749.     end
  3750.    
  3751.     function lib.returnToMidPoint(above)
  3752.         --[[ value can be integer or table]]
  3753.         if type(above) == "table" then
  3754.             T:saveToLog("lib.returnToMidPoint("..#above..")", false)
  3755.             if #above > 0 then
  3756.                 local midPoint = math.floor(#above / 2)
  3757.                 if #above % 2 == 1 and  not outward then -- length is odd no
  3758.                     midPoint = math.ceil(#above / 2)
  3759.                     T:saveToLog("    midPoint adjusted "..midPoint..")", false)
  3760.                 end
  3761.                 T:saveToLog("    T:forward("..midPoint..")", false)
  3762.                 T:forward(midPoint)
  3763.             end
  3764.         else
  3765.             T:saveToLog("lib.returnToMidPoint("..above..")", false)
  3766.             if above > 0 then
  3767.                 T:saveToLog("    T:forward("..above..")", false)
  3768.                 T:forward(above)
  3769.             end
  3770.         end
  3771.         -- now back at starting point
  3772.     end
  3773.    
  3774.     -- Start here
  3775.     -- if "tk log d.." typed instead of "tk" will start logfile and display comments. read() will be activated for debugging
  3776.     T:saveToLog("Starting function clearMountainSide", false)
  3777.     local above = lib.firstRow(R)           -- outward depends on height eg 1-2 = false, 3-5 = true, 6-8 = false
  3778.     lib.returnToMidPoint(above)             -- return to mid first row of blocks
  3779.     for row = 1, R.width -1 do
  3780.         lib.nextRow(R)
  3781.     end
  3782.    
  3783.     return {}
  3784. end
  3785.  
  3786. local function clearSandWall(R) -- 81
  3787.     --dig down while on top of sand/red_sand/soul_sand
  3788.     local lib = {}
  3789.    
  3790.     function lib.checkAbove(height)
  3791.         if turtle.detectUp() then -- moved under a ledge
  3792.             T:go("B1U1")
  3793.             height = height - 1
  3794.         end
  3795.         return height
  3796.     end
  3797.    
  3798.     function lib.moveDown(height)
  3799.         blockType = T:getBlockType("down")
  3800.         while blockType:find("sand") ~= nil do
  3801.             T:down(1)
  3802.             height = height + 1
  3803.             blockType = T:getBlockType("down")
  3804.         end
  3805.         return height
  3806.     end
  3807.    
  3808.     function lib.moveForward(length)
  3809.         lib.digForward()
  3810.         T:forward(1)
  3811.         length = length + 1
  3812.         local blockType = T:getBlockType("forward")
  3813.         return length, blockType
  3814.     end
  3815.    
  3816.     function lib.digForward()
  3817.         while T:dig("forward") do
  3818.             while T:suck("forward") do end
  3819.             while T:suck("up") do end
  3820.         end
  3821.     end
  3822.    
  3823.     local moves  = 0
  3824.     local height = 0
  3825.     local length = 0
  3826.     local search = 0
  3827.     local reverse = false
  3828.     local blockType = T:getBlockType("down")
  3829.     if R.length == 0 then
  3830.         R.length = 64
  3831.     end
  3832.    
  3833.     print("Checking for sand below")
  3834.     while blockType:find("sand") == nil do --move forward until sand detected or 3 moves
  3835.         T:forward(1)
  3836.         search = search + 1
  3837.         blockType = T:getBlockType("down")
  3838.         if search > 3 then
  3839.             T:go("B"..search)
  3840.             return {"Unable to locate sand"}
  3841.         end
  3842.     end
  3843.     -- must be sand below
  3844.     height = lib.moveDown(height)   -- go down if sand below
  3845.     -- repeat until height == 0
  3846.     repeat -- starts at bottom of sand wall
  3847.         blockType = T:getBlockType("forward")
  3848.         if blockType:find("sand") ~= nil then -- sand in front
  3849.             length, blockType = lib.moveForward(length) -- move forward 1 and dig sand
  3850.             if blockType == "" or  blockType:find("sand") ~= nil then -- sand or nothing in front
  3851.                 height = lib.moveDown(height)   -- go down if sand below
  3852.             end
  3853.         else -- solid block, air or water, not sand so move up
  3854.             if turtle.detect() then -- block in front
  3855.                 blockType = T:getBlockType("down")
  3856.                 if blockType:find("sand") ~= nil then -- sand below
  3857.                     T:dig("down")
  3858.                 end
  3859.                 T:up(1)
  3860.                 height = height - 1
  3861.             else -- air/water in front so move forward
  3862.                 if length < 60 then -- in case missing wall and in open ocean
  3863.                     length, blockType = lib.moveForward(length) -- move forward 1 and dig sand
  3864.                     height = lib.checkAbove(height)
  3865.                 else -- already > monument length of 56
  3866.                     T:up(1)
  3867.                     height = height - 1
  3868.                 end
  3869.             end
  3870.         end
  3871.     until height == 0 or length == R.length
  3872.     blockType = T:getBlockType("down")
  3873.     if blockType:find("sand") ~= nil then -- sand below
  3874.         T:dig("down")
  3875.     end
  3876.     if height > 0 then -- finished as length ran out
  3877.         T:up(height)
  3878.     end
  3879.     -- stay at end of cleared wall unless user chose to return
  3880.     if R.data == "return" then
  3881.         T:go("R2F"..length.."R2")
  3882.     end
  3883.    
  3884.     return {}
  3885. end
  3886.  
  3887. local function clearSolid(R) -- 76
  3888.     --[[ direction = R.direction "up" or "down" ]]
  3889.     local height = 1
  3890.     local remaining = R.height
  3891.     local lib = {}
  3892.    
  3893.     function lib.singleLayer(R)
  3894.         R.up = false
  3895.         R.down = false
  3896.         clearRectangle(R)
  3897.     end
  3898.    
  3899.     function lib.doubleLayer(R)
  3900.         R.up = false
  3901.         R.down = false
  3902.         if R.direction == "up" then
  3903.             R.up = true
  3904.         else
  3905.             R.down = true
  3906.         end
  3907.         clearRectangle(R)
  3908.     end
  3909.        
  3910.     function lib.tripleLayer(R)
  3911.         -- turtle in centre layer
  3912.         R.up = true
  3913.         R.down = true
  3914.         clearRectangle(R)
  3915.     end
  3916.  
  3917.     R.silent = true
  3918.     if R.height < 3 then                            --1-3 layers only
  3919.         if R.height == 1 then                       --one layer only
  3920.             lib.singleLayer(R)
  3921.         elseif R.height == 2 then                   --2 layers only current + dig up/down
  3922.             lib.doubleLayer(R)
  3923.         end
  3924.     else -- 3 or more levels
  3925.         height = height + utils.move(R, 1)          -- move up/down 1 block for first layer
  3926.         while remaining >= 3 do                     -- min 3 levels
  3927.             lib.tripleLayer(R)
  3928.             remaining = remaining - 3
  3929.             if remaining == 0 then                  -- all finished
  3930.                 break
  3931.             elseif remaining == 1 then
  3932.                 height = height + utils.move(R, 2)  -- move up/down 2 blocks
  3933.                 lib.singleLayer(R)
  3934.             elseif remaining == 2 then
  3935.                 height = height + utils.move(R, 2)  -- move up/down 2 blocks
  3936.                 lib.doubleLayer(R)
  3937.             else
  3938.                 height = height + utils.move(R, 3)  -- move up/down 3 blocks
  3939.                 if remaining == 3 then
  3940.                     finish = true
  3941.                 end
  3942.             end
  3943.         end
  3944.     end
  3945.    
  3946.     if height > 1 then
  3947.         utils.move(R, height - 1, true) -- reverse direction
  3948.     end
  3949.    
  3950.     return {}
  3951. end
  3952.  
  3953. local function clearSandCube(R) -- 81
  3954.     R.data = ""
  3955.     for w = 1, R.width do
  3956.         clearSandWall(R)
  3957.         if w < R.width then
  3958.             if w % 2 == 1 then
  3959.                 T:go("R1F1R1")
  3960.             else
  3961.                 T:go("L1F1L1")
  3962.             end
  3963.         end
  3964.     end
  3965.    
  3966.     return {}
  3967. end
  3968.  
  3969. local function clearWall(R) -- 73
  3970.     local lib = {}
  3971.    
  3972.     function lib.move(direction, blocks, reverse)
  3973.         --[[ Move up or down by blocks count ]]
  3974.         if reverse == nil then
  3975.             reverse = false
  3976.         end
  3977.         if reverse then
  3978.             if direction == "down" then -- reverse direction
  3979.                 T:up(blocks)
  3980.             else
  3981.                 T:down(blocks)
  3982.             end
  3983.         else
  3984.             if direction == "up" then
  3985.                 T:up(blocks)
  3986.             else
  3987.                 T:down(blocks)
  3988.             end
  3989.         end
  3990.         return blocks
  3991.     end
  3992.    
  3993.     function lib.singleLayer(length)
  3994.         T:go("F"..length - 1)
  3995.     end
  3996.    
  3997.     function lib.doubleLayer(modifier, length)
  3998.         for i = 1, length do
  3999.             if i < length then
  4000.                 T:go("x"..modifier.."F1")
  4001.             else
  4002.                 T:go("x"..modifier)
  4003.             end
  4004.         end
  4005.     end
  4006.    
  4007.     function lib.tripleLayer(direction, length)
  4008.         for i = 1, length do
  4009.             if i < length then
  4010.                 T:go("x0x2F1")
  4011.             else
  4012.                 T:go("x0x2")
  4013.             end
  4014.         end
  4015.     end
  4016.    
  4017.    
  4018.     -- R.width preset to 1
  4019.     -- R.subChoice = 1 up / 2 down
  4020.     if R.height < 3 then
  4021.         R.silent = true
  4022.     end
  4023.     -- dig along and up/down for specified R.length
  4024.     local modifier = "0"
  4025.     local direction = "U"
  4026.     local outbound = true
  4027.     local height = 0
  4028.     if R.subChoice == 2 then
  4029.          modifier = "2"
  4030.          direction = "D"
  4031.     end
  4032.     if R.height == 1 then               -- single block so dig and return
  4033.         lib.singleLayer(R.length)
  4034.     elseif R.height == 2 then
  4035.         lib.doubleLayer(modifier, R.length)
  4036.     else                                -- 4 blocks or more. start with bulk 3 blocks
  4037.         local remaining = R.height
  4038.         T:go(direction.."1")            -- up 1 or down 1
  4039.         height = 1
  4040.         while remaining >= 3 do
  4041.             lib.tripleLayer(direction, R.length)
  4042.             remaining = remaining - 3
  4043.            
  4044.             if remaining == 0 then      -- no more, return home, already in position
  4045.                
  4046.             elseif remaining == 1 or remaining == 2 then
  4047.                 T:go(direction.."2")
  4048.                 height = height + 2
  4049.             else
  4050.                 T:go(direction.."3")
  4051.                 height = height + 3
  4052.                 if remaining >= 3 then -- another iteration
  4053.                     T:go("R2")
  4054.                     outbound = not outbound
  4055.                 end
  4056.             end
  4057.         end
  4058.         -- 0, 1 or 2 layers left
  4059.         if remaining > 0 then
  4060.             T:go("R2")
  4061.             outbound = not outbound
  4062.             if remaining == 1 then
  4063.                 lib.singleLayer(R.length)
  4064.             elseif remaining == 2 then
  4065.                 lib.doubleLayer(modifier, R.length)
  4066.             end
  4067.         end
  4068.     end
  4069.     if outbound then
  4070.         T:go("R2F"..R.length)
  4071.     else
  4072.         T:forward(1)
  4073.     end
  4074.     direction = "D" -- reverse direction
  4075.     if R.subChoice == 2 then
  4076.          direction = "U"
  4077.     end
  4078.     if height > 0 then
  4079.         T:go(direction..height.."R2")
  4080.     else
  4081.         T:go("R2")
  4082.     end
  4083.     return {}
  4084. end
  4085.  
  4086. local function convertWater(R) -- 88
  4087.     --[[
  4088.     if dry need enough buckets to place along (width + length - 1) / 2
  4089.     use 12 buckets
  4090.     start on floor + 1
  4091.     place slab down and water up along 2 edges. stay on this level
  4092.     return round same 2 edges removing slabs and and placing them 1 above
  4093.     placeUp water onto slabs on both edges
  4094.     repeat recover slabs, place 1 above , placeUp water
  4095.    
  4096.     for sloping water, place full area with slabs
  4097.     place sloping water on top of slabs
  4098.     remove slabs
  4099.    
  4100.     ]]
  4101.     local lib = {}
  4102.    
  4103.     function lib.checkStartPosition()
  4104.         --[[
  4105.         0 T             -- T=turtle, W=wall, S=source, F=flowing
  4106.         1 W|S|F|F|F|F|F -- sloping flowing water
  4107.         2 W|F|F|F|F|F|F -- blocks removed after placing flowing water above
  4108.         3 W|S|S|S|S|S|S -- original sources
  4109.         4 W|?|?|?|?|?|? -- may be sources
  4110.         ]]
  4111.         -- need to be on floor or R.height if specified
  4112.         local depth = 0
  4113.         local blockType = T:getBlockType("down")
  4114.         local isWaterUp, isSourceUp = T:isWater("up")
  4115.         local isWaterForward, isSourceForward = T:isWater("forward")
  4116.         local isWaterDown, isSourceDown = T:isWater("down")
  4117.         print("Block below is "..blockType)
  4118.         print("Water above is "..tostring(isWaterUp))
  4119.         print("Water forward is "..tostring(isWaterForward))
  4120.         print("Water below is "..tostring(isWaterDown))
  4121.         if blockType:find("water") == nil then -- on at least level 0
  4122.             print("Moving forward in 2 seconds...")
  4123.             sleep(2)
  4124.             T:forward(1)
  4125.             blockType = T:getBlockType("down")
  4126.             if blockType:find("water") ~= nil then
  4127.                 print("Water found. Going down to floor")
  4128.                 depth = -1
  4129.             else
  4130.                 T:down(1)
  4131.                 blockType = T:getBlockType("down")
  4132.                 if blockType:find("water") ~= nil then
  4133.                     depth = -2
  4134.                 else
  4135.                     return 0, "Not close to water. Aborting..."
  4136.                 end
  4137.             end
  4138.         end
  4139.         while turtle.down() do
  4140.             depth = depth + 1
  4141.         end
  4142.         local emptyBuckets = utils.getEmptyBucketCount()
  4143.         for i = depth, 0, -1 do
  4144.             if emptyBuckets > 0 then
  4145.                 lib.fillBuckets()
  4146.                 emptyBuckets = utils.getEmptyBucketCount()
  4147.             end
  4148.             turtle.up()
  4149.         end
  4150.        
  4151.         return depth, ""
  4152.     end
  4153.    
  4154.     function lib.fillBuckets()
  4155.         local emptyBuckets = utils.getEmptyBucketCount()
  4156.         local direction = "forward"-- start with forward
  4157.         local isWater, isSource, isIce = T:isWater(direction)
  4158.         if emptyBuckets > 0 then
  4159.             if not isSource then
  4160.                 direction = "down"
  4161.                 isWater, isSource, isIce = T:isWater(direction)
  4162.                 if not isSource then
  4163.                     direction = "up"
  4164.                     isWater, isSource, isIce = T:isWater(direction)
  4165.                     if not isSource then
  4166.                         direction = ""
  4167.                     end
  4168.                 end
  4169.             end
  4170.             if direction == "" then
  4171.                 print("Unable to locate water source")
  4172.             else
  4173.                 for i = 1, emptyBuckets do
  4174.                     if utils.fillBucket(direction) then
  4175.                         print("Bucket filled "..direction)
  4176.                         sleep(0.3)
  4177.                     else
  4178.                         print("Unable to fill bucket ".. i .." / "..emptyBuckets)
  4179.                     end
  4180.                 end
  4181.             end
  4182.         end
  4183.         return utils.getWaterBucketCount()
  4184.     end
  4185.    
  4186.     function lib.placeSlabs(length)
  4187.         for i = 1, length do
  4188.             T:place("slab", "down", false)
  4189.             if i < length then
  4190.                 T:forward(1)
  4191.             end
  4192.         end
  4193.     end
  4194.    
  4195.     function lib.placeSources(length, place)
  4196.         local moves = 1
  4197.         local waterBuckets = utils.getWaterBucketCount()
  4198.         -- place sources alternate positions + start and finish
  4199.         while moves < length do
  4200.             if place then
  4201.                 if T:placeWater("up") then
  4202.                     print("Placed source up")
  4203.                     waterBuckets = waterBuckets - 1
  4204.                 end
  4205.             end
  4206.             place = not place
  4207.             if moves < length then
  4208.                 T:forward(1)
  4209.                 moves = moves + 1
  4210.             end
  4211.             if waterBuckets == 0 then
  4212.                 T:down(1) -- break the slab below
  4213.                 waterBuckets = lib.fillBuckets()
  4214.                 T:up(1)
  4215.                 T:place("slab", "down", false)
  4216.             end
  4217.         end
  4218.         if T:placeWater("up") then -- end of length
  4219.             print("Placed final source up")
  4220.         end
  4221.         return place
  4222.     end
  4223.    
  4224.     function lib.moveSlabs(length)
  4225.         for i = 1, length do
  4226.             T:dig("down")
  4227.             T:up(1)
  4228.             T:place("slab", "down", true)
  4229.             if i < length then
  4230.                 T:forward(1)
  4231.                 T:down(1)
  4232.             end
  4233.         end
  4234.     end
  4235.    
  4236.     function lib.recoverSlabs(length)
  4237.         for i = 1, length do
  4238.             T:dig("down")
  4239.             if i < length then
  4240.                 T:forward(1)
  4241.             end
  4242.         end
  4243.     end
  4244.    
  4245.     local depth, message = lib.checkStartPosition()
  4246.     if message ~= "" then
  4247.         return {message}
  4248.     end
  4249.     local maxDepth = R.height
  4250.     local buckets = utils.getWaterBucketCount()
  4251.     R = utils.calculateDimensions(R) -- if R.width or R.length == 0
  4252.     T:down(depth)
  4253.     lib.placeSlabs(R.length)
  4254.     T:go("R1")
  4255.     lib.placeSlabs(R.width)
  4256.     T:go("R2")
  4257.    
  4258.     while depth > 0 do
  4259.         local place = true
  4260.         lib.fillBuckets()
  4261.         place = lib.placeSources(R.width, place)
  4262.         T:go("L1")
  4263.         place = lib.placeSources(R.length, place)
  4264.         lib.fillBuckets()
  4265.         T:go("R2")
  4266.         lib.moveSlabs(R.length) -- dig slab from below, move up and replace below
  4267.         T:go("R1F1D1")
  4268.         lib.moveSlabs(R.width - 1)
  4269.         T:go("R2") -- now moved up 1 layer
  4270.         depth = depth - 1
  4271.         if depth == 0 then
  4272.             place = lib.placeSources(R.width, true)
  4273.             T:go("L1")
  4274.             place = lib.placeSources(R.length, place)
  4275.             T:go("R2")
  4276.             lib.recoverSlabs(R.length)
  4277.             T:go("R1")
  4278.             lib.recoverSlabs(R.width)
  4279.         end
  4280.     end
  4281.    
  4282.     return {}
  4283. end
  4284.  
  4285. local function createBoatLift(R) -- 59 state:0=new, size:1=extend, side:0=left, 1=right
  4286.     -- build stepped lift with fencing gates and soul sand
  4287.     local lib = {}
  4288.    
  4289.     function lib.getWater(backToWater, downToWater)
  4290.         if backToWater > 0 then
  4291.             utils.goBack(backToWater)
  4292.         end
  4293.         if downToWater > 0 then
  4294.             T:down(downToWater)
  4295.         end
  4296.         T:getWater("down") -- take water from source
  4297.         sleep(0.2)
  4298.         T:getWater("down") -- take water from source
  4299.         if downToWater > 0 then
  4300.             T:up(downToWater)
  4301.         end
  4302.         if backToWater > 0 then
  4303.             T:forward(backToWater)
  4304.         end
  4305.     end
  4306.    
  4307.     --T:place(blockType, direction, leaveExisting, signText)
  4308.    
  4309.     local backToWater = 0
  4310.     local downToWater = 0
  4311.    
  4312.     T:go("R1F1L1")                                      -- over canal facing forward
  4313.     for h = 1, R.height do
  4314.         lib.getWater(backToWater, downToWater)          -- check water supplies, return to starting position
  4315.         T:go("L1C1 R1D1 L1C1 R1", false, 0, false)      -- place towpath, forward, down, place towpath, face forward
  4316.         T:place("soul", "down", false)              -- place soulsand down
  4317.         T:place("soul", "forward", false)           -- place soulsand forward
  4318.         T:go("R1F1C1L1", false, 0, false)               -- place right towpath face forward
  4319.         T:place("soul", "down", false)              -- place soulsand down
  4320.         T:place("soul", "forward", false)           -- place soulsand forward
  4321.         T:go("U1 R1C1 L1")                              -- place towpath, face forward
  4322.         T:placeWater("down")                            -- place water down
  4323.         utils.goBack(1)
  4324.         T:place("gate", "forward", false)           -- place fence gate
  4325.         T:go("R1C1 U1C1 D1 L2F1 C1R1 F1 L1C1R1")        -- over left soul sand
  4326.         T:placeWater("down")                            -- place water down
  4327.         utils.goBack(1)
  4328.         T:place("gate", "forward", false)           -- place fence gate
  4329.         T:go("U1 L1C1 R1F1 L1C1 R1x1")                  -- facing forward first unit complete
  4330.         T:go("R1F1 L1x1 R1C1")
  4331.         utils.goBack(1)
  4332.         T:go("L1F1")
  4333.         if backToWater == 0 then
  4334.             backToWater = 1
  4335.         end
  4336.         backToWater = backToWater + 1
  4337.         downToWater = downToWater + 1
  4338.     end
  4339.    
  4340.     -- now finish the canal
  4341.     lib.getWater(backToWater, downToWater)
  4342.     T:go("D1 L1C1 U1C1")                    -- build left towpath, facing towpath, above water level
  4343.     T:go("R1F1 L1C1 D1C1")                  -- move forward, build towpath, facing towpath ground level
  4344.     T:go("R1C1 R1F1 L1C1 R1C1 U1C1")        -- build right towpath, facing towpath, above water level
  4345.     T:go("R1F1 L1C1 D1C1 U1")               -- build right towpath next to gate, facing towpath, above water level
  4346.     T:placeWater("down")
  4347.     utils.goBack(1)
  4348.     T:go("L1F1")
  4349.     T:placeWater("down")  
  4350.    
  4351.     return {}
  4352. end
  4353.  
  4354. local function createBorehole(R)
  4355.     --[[go down to bedrock and return. Chart all blocks dug/ passed through]]
  4356.     local diary = {}
  4357.     local lib = {}
  4358.     local depth = R.height  -- eg 63 start position
  4359.     local moves = 0
  4360.     --R.height = current level
  4361.    
  4362.     function lib.addBlock(depth, blockType, diary)
  4363.         if blockType == "" then
  4364.             blockType = "air"
  4365.         end
  4366.         table.insert(diary, blockType)
  4367.        
  4368.        
  4369.         --[[if blockType ~= "" then
  4370.             local add = true
  4371.             for k,v in pairs(diary) do
  4372.                 if blockType == v then
  4373.                     add = false
  4374.                     break
  4375.                 end
  4376.             end
  4377.             if add then
  4378.                 diary[depth] = blockType
  4379.             end
  4380.         end]]
  4381.        
  4382.         return diary
  4383.     end
  4384.    
  4385.     function lib.processItem(item)
  4386.         if item:find("minecraft") ~= nil then
  4387.             return item:sub(11)
  4388.         end
  4389.         return item
  4390.     end
  4391.    
  4392.     function lib.writeReport(R, diary)
  4393.         local numLevels = #diary                        -- eg 125 levels
  4394.         local levelsPerCol = math.ceil(numLevels / 4)   -- eg 31.25 -> 32
  4395.         local lines = {}
  4396.         for l = 1, levelsPerCol do                      -- add 32 empty strings
  4397.             table.insert(lines, "")
  4398.         end
  4399.         local lineNo = 1
  4400.         for k, v in ipairs(diary) do
  4401.             local level = R.height - k                  -- eg 63 range 63 to -59
  4402.             local lev = "      "
  4403.             local item = lib.processItem(v)
  4404.             if level < -9 then
  4405.                 lev = tostring(level).."   "            -- "-10   " to "-59   "
  4406.             elseif level < 0 then              
  4407.                 lev = "-0"..math.abs(level).."   "      -- "-09   " to "-01   "
  4408.             elseif level < 10 then
  4409.                 lev = " 0"..level.."   "                -- " 01   " to " 09   "
  4410.             elseif level < 100 then
  4411.                 lev = " "..level.."   "                 -- " 10   " to " 99   "
  4412.             else
  4413.                 lev = " "..level.."  "                  -- " 100  " to " 319  "
  4414.             end
  4415.             local output = lev..item                    -- eg "-10   grass_block"
  4416.             if #output > 20 then                        -- eg "-10   some_long_block_name"  
  4417.                 output = output:sub(1, 20)              -- eg "-10   some_long_block_"
  4418.             else
  4419.                 output = menu.padRight(output, 20, " ") -- eg "-10   grass_block     "
  4420.             end
  4421.             lines[lineNo] = lines[lineNo]..output       -- add new entry to this line
  4422.             lineNo = lineNo + 1                         -- increase line no
  4423.             if lineNo > levelsPerCol then               -- past last line number
  4424.                 lineNo = 1                              -- reset to 1
  4425.             end
  4426.         end
  4427.        
  4428.         local fileName = "borehole"..os.getComputerID()..".txt"
  4429.         local handle = fs.open(fileName, "w")       --create file eg "borehole0.txt"
  4430.         handle.writeLine("Level Block         Level Block         Level Block         Level Block")
  4431.         for k,v in ipairs(lines) do
  4432.             handle.writeLine(v)
  4433.         end
  4434.        
  4435.         handle.close()
  4436.        
  4437.         return fileName
  4438.     end
  4439.    
  4440.     local blockType = T:getBlockType("down")
  4441.     while T:down(1) do
  4442.         depth = depth - 1
  4443.         moves = moves + 1
  4444.         if depth == R.depth then
  4445.             break
  4446.         end
  4447.         diary = lib.addBlock(depth, blockType, diary)
  4448.         blockType = T:getBlockType("down")
  4449.     end
  4450.     local fileName = lib.writeReport(R, diary)
  4451.     T:up(moves)
  4452.    
  4453.     return {"File '"..fileName.."' written"}
  4454. end
  4455.  
  4456. local function createBubbleLift(R) -- 15
  4457.     local lib = {}
  4458.    
  4459.     function lib.addLayer()
  4460.         T:go("F2 L1C1 R1C1 R1C1 L1", false, 0, true)
  4461.         turtle.back()
  4462.         T:dig("up") -- clear block above so completed lift can be found
  4463.         T:placeWater("forward")
  4464.         turtle.back()
  4465.         T:go("C1x0")       
  4466.     end
  4467.    
  4468.     function lib.addSign()
  4469.         turtle.back()
  4470.         T:placeWater("forward")
  4471.         T:go("L1C1B1")
  4472.         T:place("sign", "forward")
  4473.     end
  4474.    
  4475.     function lib.buildLift(toHeight)
  4476.         local built = lib.goToWater()       -- returns lift blocks already placed, total height of drop from starting point
  4477.         local toBuild = toHeight - built    -- no of blocks remaining to increase lift size
  4478.         local water = 0
  4479.         while toBuild > 0 do                -- at least 1 block height remaining
  4480.             water = lib.fillBuckets(toBuild, false) -- no of water buckets onboard (could be more than required)
  4481.             if water > toBuild then         -- more water than required
  4482.                 water = toBuild             -- reduce to correct amount
  4483.             end
  4484.             T:up(built)                     -- climb to top of existing lift
  4485.             if T:getBlockType("forward"):find("sign") ~= nil then
  4486.                 T:up(1)
  4487.             end
  4488.             while water > 0 and toBuild > 0 do
  4489.                 lib.addLayer()
  4490.                 water = water - 1
  4491.                 T:up(1)
  4492.                 toBuild = toBuild - 1
  4493.             end
  4494.             -- may still be some height to complete, but needs refill
  4495.             if toBuild > 0 then
  4496.                 built = lib.goToWater() --return to source
  4497.                 toBuild = toHeight - built
  4498.             end
  4499.         end
  4500.     end
  4501.    
  4502.     function lib.cleanUp(fromHeight)
  4503.         local plug = false
  4504.         T:turnRight(2)
  4505.         for i = 1, fromHeight do
  4506.             plug = false
  4507.             if turtle.detect() then
  4508.                 plug = true
  4509.             end
  4510.             turtle.down()
  4511.             if plug then
  4512.                 T:place("stone", "up")
  4513.             end
  4514.         end
  4515.     end
  4516.        
  4517.     function lib.fillBuckets(withSort)
  4518.         local emptyBuckets = T:getItemCount("minecraft:bucket")
  4519.         for i = 1, emptyBuckets do
  4520.             if T:getWater("down") then
  4521.                 sleep(0.5)
  4522.             end
  4523.         end
  4524.        
  4525.         return T:getItemCount("minecraft:water_bucket")
  4526.     end
  4527.    
  4528.     function lib.getEmptySlots()
  4529.         local empty = 0
  4530.         for i = 1, 16 do
  4531.             if turtle.getItemCount(i) == 0 then
  4532.                 empty = empty + 1
  4533.             end
  4534.         end
  4535.         return empty
  4536.     end
  4537.    
  4538.     function lib.goToWater()
  4539.         local built = 0 -- measures completed lift height
  4540.         while turtle.down() do -- takes turtle to bottom of water source
  4541.             if turtle.detect() then
  4542.                 built = built + 1
  4543.             end
  4544.         end
  4545.         T:up(1) -- above watersource ready to fill buckets
  4546.         -- height = height - 1
  4547.         -- built = built - 1 not required as next block is water source: not detected
  4548.         return built -- , height
  4549.     end
  4550.    
  4551.     function lib.stackBuckets(withSort)
  4552.         if withSort == nil then withSort = false end
  4553.         local data = {}
  4554.         local bucketSlot = 0
  4555.         local emptySlots = 0
  4556.         local water = 0
  4557.         if withSort then
  4558.             T:sortInventory()
  4559.         end
  4560.         for i = 1, 16 do
  4561.             -- find first empty bucket
  4562.             if turtle.getItemCount(i) > 0 then
  4563.                 data = turtle.getItemDetail(i)
  4564.                 if data.name == "minecraft:bucket" then
  4565.                     if bucketSlot == 0 then
  4566.                         bucketSlot = i
  4567.                     else
  4568.                         turtle.select(i)
  4569.                         turtle.transferTo(bucketSlot)
  4570.                     end
  4571.                 elseif data.name == "minecraft:water_bucket" then
  4572.                     water = water + 1
  4573.                 end
  4574.             else
  4575.                 emptySlots = emptySlots + 1
  4576.             end
  4577.         end
  4578.         return emptySlots, water
  4579.     end
  4580.    
  4581.     T:go("C1R1")                                                -- place block next to ladder support block, turn right to check ladder
  4582.     local blockType = T:getBlockType("forward")                 -- Is there a ladder to the right?
  4583.     if blockType:find("ladder") == nil then
  4584.         T:go("C1L2")                                            -- Place block as ladder not present
  4585.     else                                                        -- ladder present
  4586.         T:go("L2")                                              -- ready to check other side
  4587.     end
  4588.     blockType = T:getBlockType("forward")
  4589.     if blockType:find("ladder") == nil then
  4590.         T:go("C1R1")
  4591.     else
  4592.         T:go("R1")
  4593.     end
  4594.     blockType = T:getBlockType("down")
  4595.     local dirt = "minecraft:dirt"
  4596.     if T:getItemSlot("minecraft:soul_sand") > 0 then
  4597.         dirt = "minecraft:soul_sand"
  4598.     end
  4599.     if blockType:find("bedrock") ~= nil then
  4600.         utils.goBack(1)
  4601.         T:place(dirt, "forward", false)                     -- placed at end of potential water source, next to ladder
  4602.     else
  4603.         T:place(dirt, "down", false)                        -- placed in ground, next to ladder
  4604.         utils.goBack(1)
  4605.         T:placeWater("forward")                             -- place extra water source
  4606.     end
  4607.     T:go("R1C1 R2C1 L1F1C2 R1C1 L2C1 R1F1C2 R1C1 L1C1 L1C1 L1F1", false, 0, true)
  4608.     -- ready for water sources to be placed
  4609.     T:placeWater("forward")                                 -- place front water source
  4610.     T:turnRight(2)                                          -- facing backward
  4611.     T:placeWater("forward")                                 -- place back water source
  4612.     T:go("R2U1")                                            -- facing forward, U1, above centre of water source
  4613.  
  4614.     --lib.fillBuckets(R.height, true)                           -- fill as many buckets as required or until inventory full, sort inventory as well
  4615.     lib.fillBuckets(true)                           -- fill as many buckets as required or until inventory full, sort inventory as well
  4616.     local nextToLadder = false
  4617.     --T:go("F2R1")
  4618.     T:go("x0F1 x0F1C1 R1")                                  -- move forward 2 taking out blocks above, plug behind soul sand
  4619.     if T:getBlockType("forward"):find("ladder") == nil then -- no ladder here
  4620.         T:turnLeft(2)
  4621.         if T:getBlockType("forward"):find("ladder") ~= nil then -- ladder here
  4622.             nextToLadder = true
  4623.         end
  4624.         T:turnRight(1)
  4625.     else
  4626.         nextToLadder = true
  4627.         T:turnLeft(1)
  4628.     end
  4629.     -- now above soul sand, facing forward
  4630.     if nextToLadder then                                -- if nextToLadder, no need for signs
  4631.         utils.goBack(2)                                 -- return to source centre
  4632.     else
  4633.         T:go("L1C1 R1C1 R1C1 L1", false, 0, true)       -- prepare layer 1
  4634.         lib.addSign()
  4635.         T:go("U1F1 R1F1 L1C1 R1C1 R1C1 L1", false, 0, true) -- prepare layer 2
  4636.         lib.addSign()
  4637.         T:go("L1F1 R1F1R1", false, 0, true)             -- above source, level 2
  4638.     end
  4639.     -- ready to build lift
  4640.     lib.buildLift(R.height - 1)
  4641.     lib.cleanUp(R.height - 1)
  4642.    
  4643.     return {"Bubble lift created", "Check correct operation", "Check exit before using" }
  4644. end
  4645.  
  4646. local function createBubbleTrap(R) -- 34
  4647.     --[[
  4648.     Replace floor of existing volume of water with soulsand
  4649.     Used in river biome for squid farm. Area should be enclosed
  4650.     with solid walls surface to floor.
  4651.     Usually 7 blocks wide
  4652.     ]]
  4653.     local lib = {}
  4654.    
  4655.     function lib.clearDown(depth)
  4656.         while utils.clearVegetation("down") do
  4657.             T:down(1)
  4658.             depth = depth + 1
  4659.         end
  4660.         T:place("soul_sand", "down", false)
  4661.         return depth
  4662.     end
  4663.    
  4664.     function lib.clearLength(length, depth)
  4665.         local moves = 0
  4666.         while moves < length - 1 do
  4667.             if utils.clearVegetation("forward") then
  4668.                 T:forward(1)
  4669.                 moves = moves + 1
  4670.                 depth = lib.clearDown(depth) -- go down if in water/air
  4671.                 if moves >= length - 1 then
  4672.                     return depth
  4673.                 end
  4674.             else -- block in front
  4675.                 local waterAhead = utils.clearVegetation("forward") -- true if air/water in front
  4676.                 while not waterAhead do     -- solid block in front
  4677.                     T:up(1)
  4678.                     depth = depth - 1
  4679.                     if depth < 1 then
  4680.                         return 0
  4681.                     end
  4682.                     waterAhead = utils.clearVegetation("forward")
  4683.                 end
  4684.             end
  4685.         end
  4686.  
  4687.         return depth
  4688.     end
  4689.    
  4690.     function lib.turn(facingForward, depth)
  4691.         local direction = "R"
  4692.         if not facingForward then
  4693.             direction = "L"
  4694.         end
  4695.         T:go(direction.. 1)
  4696.         if utils.clearVegetation("forward") then
  4697.             T:forward(1)
  4698.             depth = depth + lib.clearDown(depth)
  4699.         else
  4700.             while not utils.clearVegetation("forward") do
  4701.                 T:up(1)
  4702.                 depth = depth - 1
  4703.             end
  4704.         end
  4705.         T:go(direction.. 1)
  4706.        
  4707.         return depth
  4708.     end
  4709.    
  4710.     local depth = 0
  4711.     local length = 0
  4712.     local width = 0
  4713.     local outbound = true
  4714.     local facingForward = true
  4715.     local inWater, onWater = utils.getWaterStatus()
  4716.     inWater, onWater = utils.startWaterFunction(onWater, inWater, 2 ,true) -- move into water, max descent 2
  4717.     if R.length == 0 then
  4718.         R.length = lib.getLength()
  4719.     end
  4720.     if R.width == 0 then
  4721.         T:turnRight(1)
  4722.         R.width = lib.getLength()
  4723.         T:turnLeft(1)
  4724.     end
  4725.     -- go down to floor
  4726.     depth = lib.clearDown(depth)
  4727.     while width < R.width do
  4728.         -- now on floor, move along sea/river bed following contour
  4729.         depth = lib.clearLength(R.length, depth)
  4730.         width = width + 1
  4731.         -- now turn and repeat
  4732.         if width < R.width then
  4733.             depth = lib.turn(facingForward, depth)
  4734.             facingForward = not facingForward
  4735.         end
  4736.     end
  4737.     -- finished so return to surface
  4738.     T:up(1) -- up 1 to check for water below
  4739.     while T:isWater("down") do
  4740.         T:up(1)
  4741.     end
  4742.    
  4743.     return {}
  4744. end
  4745.  
  4746. local function createCorridor(R) -- 52
  4747.     --[[create a corridoor 2 blocks high, with floor and ceiling guaranteed
  4748.     T:go(path, useTorch, torchInterval, leaveExisting, preferredBlock)]]
  4749.     local lib = {}
  4750.    
  4751.     function lib.continue(R, currentSteps, totalSteps)
  4752.         if not R.silent then
  4753.             if currentSteps >= 64 and R.length == 0 then
  4754.                 -- request permission to continue if infinite
  4755.                 T:clear()
  4756.                 print("Completed "..totalSteps..". Ready for 64 more")
  4757.                 print("Do you want to continue? (y/n)")
  4758.                 response = read()
  4759.                 if response:lower() ~= "y" then
  4760.                     return true, 0
  4761.                 else
  4762.                     return false, currentSteps
  4763.                 end
  4764.             end
  4765.         end
  4766.         return true, currentSteps
  4767.     end
  4768.    
  4769.     function lib.seal()
  4770.         local blockType = T:getBlockType("forward")
  4771.         if blockType:find("water") ~= nil then
  4772.             T:place("stone", "forward", false)
  4773.             return "water"  -- water found
  4774.         elseif blockType:find("lava") ~= nil then
  4775.             T:place("stone", "forward", false)
  4776.             return "lava"   -- lava found
  4777.         end
  4778.         return ""   -- no water or lava
  4779.     end
  4780.    
  4781.     function lib.checkSeal(R)
  4782.         local fluidType = ""
  4783.         if R.data == "seal" then        -- check for lava/water at the sides
  4784.             T:turnRight(1)
  4785.             fluidType = lib.seal()      -- could be "", "water", "lava"
  4786.             T:turnLeft(2)
  4787.             local blockType = lib.seal()
  4788.             if fluidType == "" then     -- no water / lava so far
  4789.                 fluidType = blockType   -- could be "", "water", "lava"
  4790.             end
  4791.             T:turnRight(1)
  4792.         end
  4793.         return fluidType                -- could be "", "water", "lava"
  4794.     end
  4795.    
  4796.     function lib.placeTorch(R, torchSpaces, totalSteps)
  4797.         if R.torchInterval > 0 then -- torches onboard
  4798.             if torchSpaces == R.torchInterval then -- time to place another torch
  4799.                 if totalSteps < R.length then -- not at end of run
  4800.                     if T:getItemSlot("minecraft:torch") > 0 then
  4801.                         T:place("minecraft:torch", "down")
  4802.                     end
  4803.                     torchSpaces = 1
  4804.                 end
  4805.             end
  4806.         end
  4807.         return torchSpaces -- original value or 1
  4808.     end
  4809.    
  4810.     local currentSteps = 0                  -- counter for infinite length. pause every 64 blocks
  4811.     local totalSteps = 0                    -- counter for all steps so far
  4812.     local torchSpaces = R.torchInterval     -- if torches present, counter to place with 8 blocks between
  4813.     local fluidType = ""
  4814.     local damLength = 0
  4815.     local damStarted = false
  4816.     local doContinue = true
  4817.     if T:getItemSlot("minecraft:torch") == 0 then
  4818.         R.torchInterval = 0 -- set to default 9 in getTask()
  4819.     end
  4820.     for steps = 1, R.length do
  4821.         -- starts on floor of tunnel
  4822.         doContinue, currentSteps = lib.continue(R, currentSteps, totalSteps) -- continue tunnelling?
  4823.         if not doContinue then
  4824.             break
  4825.         end
  4826.         T:go("C2U1C0", false, 0, true)      -- place floor, up 1, place ceiling
  4827.         fluidType = lib.checkSeal(R)        -- if R.data == "seal", check for water/lava at ceiling level
  4828.         if fluidType == "" then -- either R.data ~= "seal" or no fluid found
  4829.             torchSpaces = lib.placeTorch(R, torchSpaces, totalSteps) -- original value or 1 if torch placed
  4830.             T:go("F1D1")
  4831.         elseif fluidType == "water" then
  4832.             T:go("F1R2 C1D1 C1L2", false, 0, true)
  4833.             damStarted = true
  4834.             damLength = damLength + 1
  4835.         else    --lava
  4836.             T:go("F1D1")
  4837.         end
  4838.         blockType = lib.checkSeal(R)
  4839.         if blockType ~= "" then
  4840.             fluidType = blockType
  4841.         end
  4842.         currentSteps = currentSteps + 1
  4843.         totalSteps = totalSteps + 1
  4844.         torchSpaces = torchSpaces + 1
  4845.         if damStarted and fluidType == "" then -- was in water, but no more
  4846.             T:go("R2 F"..damLength + 1 .."U1L2F"..damLength + 1 .."D1")
  4847.             damStarted = false
  4848.         end
  4849.     end
  4850.     if fluidType ~= "" then -- water or lava found while tunnelling
  4851.         T:go("U1C0", false, 0, true)
  4852.         lib.checkSeal(R)
  4853.         T:go("C1", false, 0, true)
  4854.         T:down(1)
  4855.     end
  4856.     return {}
  4857. end
  4858.  
  4859. local function createDragonTrap() -- 49
  4860.     local lib = {}
  4861.    
  4862.     function lib.attack()
  4863.         local totalHitsF = 0
  4864.         local totalHitsU = 0
  4865.         local totalHitsD = 0
  4866.         while true do
  4867.             local hitF = false
  4868.             local hitU = false
  4869.             local hitD = false
  4870.             if turtle.attackUp() then
  4871.                 hitU = true
  4872.                 totalHitsU = totalHitsU + 1
  4873.             end
  4874.             if turtle.attackDown() then
  4875.                 hitD = true
  4876.                 totalHitsD = totalHitsD + 1
  4877.             end
  4878.             if turtle.attack() then
  4879.                 hitF = true
  4880.                 totalHitsF = totalHitsF + 1
  4881.             end
  4882.             if hitF or hitU or hitD then
  4883.                 print("hits forward: "..totalHitsF..", up: "..totalHitsU..", down: "..totalHitsD)
  4884.             end
  4885.         end
  4886.     end
  4887.     -- build up 145 blocks with ladders
  4888.     T:clear()
  4889.     menu.colourPrint("Press Enter to start 1 minute delay\n", colors.red)
  4890.     menu.colourPrint("Run to island centre across the bridge\n", colors.orange)
  4891.     menu.colourPrint("You have already made the bridge?...", colors.lime)
  4892.     read()
  4893.     for t = 60, 1, -1 do
  4894.         sleep(1)
  4895.         T:clear()
  4896.         io.write("Starting in "..t.. " seconds ")
  4897.     end
  4898.     for i = 1, 145 do
  4899.         T:go("U1C2")
  4900.         turtle.back()
  4901.         T:place("minecraft:ladder", "down")
  4902.         turtle.forward()
  4903.     end
  4904.     T:go("R2F1C1 L1C1 L2C1 R1")
  4905.     for i = 1, 100 do
  4906.         T:go("F1C2U1C0D1")
  4907.     end
  4908.     T:forward(1)
  4909.     T:place("minecraft:obsidian", "down")
  4910.     T:go("R2F1x2R2")
  4911.     T:placeWater("forward")
  4912.     T:go("R2F6R2")
  4913.     lib.attack()
  4914.    
  4915.     return {}
  4916. end
  4917.    
  4918. local function createEnderTower(stage) -- 66
  4919.     --[[ lower base = stage 1, upper base = 2, tower = 3 ]]
  4920.     local lib = {}
  4921.     --[[ go(path, useTorch, torchInterval, leaveExisting, preferredBlock) ]]
  4922.     function lib.getEmptySlots()
  4923.         local empty = 0
  4924.         for i = 1, 16 do
  4925.             if turtle.getItemCount(i) == 0 then
  4926.                 empty = empty + 1
  4927.             end
  4928.         end
  4929.         return empty
  4930.     end
  4931.  
  4932.     function lib.getStone(direction, stacks)
  4933.         --[[ get block user wants to use ]]
  4934.         local suck = turtle.suck   
  4935.         if direction == "down" then
  4936.             suck = turtle.suckDown
  4937.         end
  4938.         if T:getBlockType(direction) == "minecraft:chest" then
  4939.             T:sortInventory()
  4940.             local slot = T:getFirstEmptySlot() --find spare slot
  4941.             if slot > 0 then --empty slot found
  4942.                 turtle.select(1)
  4943.                 if stacks == 0 then
  4944.                     while suck() do end
  4945.                 else
  4946.                     for i = 1, stacks do -- get # stacks of stone from chest
  4947.                         suck()
  4948.                     end
  4949.                 end
  4950.                 if T:getSlotContains(slot) == "" then
  4951.                     return T:getMostItem()              -- empty chest
  4952.                 else
  4953.                     return T:getSlotContains(slot)      -- use this as default building block
  4954.                 end
  4955.             else
  4956.                 return T:getMostItem()              -- full inventory
  4957.             end
  4958.         else
  4959.             return T:getMostItem()              -- no chest
  4960.         end
  4961.     end
  4962.    
  4963.     function lib.stackBuckets()
  4964.         local data = {}
  4965.         local bucketSlot = 0
  4966.         local emptySlots = 0
  4967.         local water = 0
  4968.         T:sortInventory()
  4969.         for i = 1, 16 do
  4970.             -- find first empty bucket
  4971.             if turtle.getItemCount(i) > 0 then
  4972.                 data = turtle.getItemDetail(i)
  4973.                 if data.name == "minecraft:bucket" then
  4974.                     if bucketSlot == 0 then
  4975.                         bucketSlot = i
  4976.                     else
  4977.                         turtle.select(i)
  4978.                         turtle.transferTo(bucketSlot)
  4979.                     end
  4980.                 elseif data.name == "minecraft:water_bucket" then
  4981.                     water = water + 1
  4982.                 end
  4983.             else
  4984.                 emptySlots = emptySlots + 1
  4985.             end
  4986.         end
  4987.         return emptySlots, water
  4988.     end
  4989.    
  4990.     function lib.countWaterBuckets()
  4991.         local data = {}
  4992.         local buckets = 0
  4993.         for i = 1, 16 do
  4994.             data = turtle.getItemDetail(i)
  4995.             if data.name == "minecraft:water_bucket" then
  4996.                 buckets = buckets + 1
  4997.             end
  4998.         end
  4999.         return buckets
  5000.     end
  5001.    
  5002.     function lib.baseRun(preferredBlock, count, turn)
  5003.         for i = 1, count do
  5004.             T:go("C2F1", false, 0, false, preferredBlock)
  5005.         end
  5006.         T:go("C2"..turn, false, 0, false, preferredBlock)
  5007.     end
  5008.    
  5009.     function lib.outsideRun(preferredBlock)
  5010.         T:place("fence", "down", false)
  5011.         T:forward(1)
  5012.         T:place(preferredBlock, "down", false)
  5013.         T:forward(1)
  5014.         T:place(preferredBlock, "down", false)
  5015.         T:forward(2)
  5016.         T:place(preferredBlock, "down", false)
  5017.     end
  5018.    
  5019.     function lib.signRun(preferredBlock ,message)
  5020.         T:place(preferredBlock, "down", false)
  5021.         T:forward(4)
  5022.         T:place(preferredBlock, "down", false)
  5023.         turtle.back()
  5024.         turtle.back()
  5025.         T:down(1)
  5026.         T:place("sign", "forward", false, message)
  5027.         T:go("U1F2")
  5028.     end
  5029.    
  5030.     function lib.goToWater(height)
  5031.         local built = 0 -- measures completed lift height
  5032.         while turtle.down() do -- takes turtle to bottom of water source
  5033.             height = height + 1
  5034.             if turtle.detect() then
  5035.                 built = built + 1
  5036.             end
  5037.         end
  5038.         T:up(1) -- above watersource assuming it is 1-1.5 blocks deep
  5039.         height = height - 1
  5040.         -- built = built - 1 not required as next block is water source: not detected
  5041.         return built, height
  5042.     end
  5043.    
  5044.     function lib.fillBuckets(toBuild)
  5045.         local emptySlots, water = lib.stackBuckets() -- gets no of empty slots + no of water buckets
  5046.         if water < toBuild then -- no of water buckets onboard less than required quantity
  5047.             for i = 1, toBuild do -- fill required no of buckets up to max space in inventory
  5048.                 emptySlots = lib.getEmptySlots()
  5049.                 if emptySlots == 0 then -- inventory full
  5050.                     break
  5051.                 else
  5052.                     if T:getWater("down") then
  5053.                         water = water + 1
  5054.                         sleep(0.5)
  5055.                     end
  5056.                 end
  5057.             end
  5058.         end
  5059.        
  5060.         return water
  5061.     end
  5062.    
  5063.     function lib.buildLift(preferredBlock)
  5064.         local built = 0 -- measures completed lift height
  5065.         local height = 0 -- measures total height from starting position
  5066.         built, height = lib.goToWater(height) -- returns lift blocks already placed, total height of drop from starting point
  5067.         local toBuild = height - built -- no of blocks to increase lift size
  5068.         while toBuild > 0 do -- at least 1 block height remaining
  5069.             local water = lib.fillBuckets(toBuild) -- no of water buckets onboard (could be more than required)
  5070.             if water > toBuild then
  5071.                 water = toBuild
  5072.             end
  5073.             while turtle.detect() do -- climb to top of existing lift
  5074.                 turtle.up()
  5075.                 height = height - 1
  5076.             end
  5077.             T:forward(1)
  5078.             for i = 1, water do -- build lift by no of water buckets
  5079.                 if T:placeWater("forward")  then
  5080.                     T:up(1)
  5081.                     height = height - 1
  5082.                     toBuild = toBuild - 1
  5083.                     T:place(preferredBlock, "down", false)
  5084.                 end
  5085.             end
  5086.             turtle.back()
  5087.             -- may still be some height to complete, but needs refill
  5088.             if toBuild > 0 then
  5089.                 lib.goToWater(0) --return to source
  5090.                 lib.fillBuckets(toBuild)
  5091.             end
  5092.         end
  5093.         if height > 0 then -- if any remaining distance
  5094.             T:up(height)
  5095.         end
  5096.        
  5097.     end
  5098.    
  5099.     function lib.buildSection(preferredBlock, solid)
  5100.         -- builds a section without any blocks in the centre
  5101.         -- second layer of each section end walls have fence posts
  5102.         T:go("F1C2 F2C2 F1R1", false, 0, false, preferredBlock) -- first side solid row
  5103.         if solid then -- first layer of each section
  5104.             T:go("F1C2 F1R1", false, 0, false, preferredBlock) -- top side solid row
  5105.         else
  5106.             T:go("F1") -- top side solid row
  5107.             if not T:place("fence", "down", false) then-- first side
  5108.                 T:place(preferredBlock, "down", false)
  5109.             end
  5110.             T:go("F1R1") -- top side solid row
  5111.         end
  5112.         T:go("F1C2 F2C2 F1R1", false, 0, false, preferredBlock) -- far side solid row
  5113.         T:go("F1C2 F1R1U1", false, 0, false, preferredBlock) -- bottom side solid row
  5114.     end
  5115.     --[[
  5116.         clsTurtle methods:
  5117.         clsTurtle.place(self, blockType, damageNo, direction, leaveExisting)
  5118.         clsTurtle.go(self, path, useTorch, torchInterval, leaveExisting, preferredBlock)
  5119.     ]]
  5120.     -- remove 1 stack stone from chest
  5121.     local preferredBlock = lib.getStone("down", 1) -- use this as default building block
  5122.     if stage == 1 then
  5123.         -- build base floor
  5124.         --T:go("R2F2R1F3R1", false, 0, false, preferredBlock)
  5125.         T:go("R2F1C2R1F1C2F1C2F1C2R1", false, 0, false, preferredBlock)
  5126.         for i = 1, 2 do
  5127.             lib.baseRun(preferredBlock, 8, "R1F1R1")
  5128.             lib.baseRun(preferredBlock, 8, "L1F1L1")
  5129.             lib.baseRun(preferredBlock, 8, "R1F4R1")
  5130.         end
  5131.         -- move back to centre, build water source, with soul sand at base of first source
  5132.         --T:go("R1F3L1C2F1C2F2D1", false, 0, false, preferredBlock) --just behind chest, 1 below ground level
  5133.         T:go("R1F3L1F2C2F1D1", false, 0, false, preferredBlock) --1 block behind chest, 1 below ground level
  5134.         T:place("minecraft:soul_sand", "down", false) -- over block 1 of water source
  5135.         T:go("F1C2F1C2", false, 0, false, preferredBlock) -- over block 2 of water source
  5136.         T:go("F1C2U1C2", false, 0, false, preferredBlock) -- over block 4 of water source
  5137.         T:go("F1C2F1C2R2F5R2", false, 0, false, preferredBlock) -- over block 1 of water source
  5138.         T:placeWater("down")
  5139.         T:forward(2) -- over block 3 of water source
  5140.         T:placeWater("down")
  5141.         turtle.back() -- over block 2 of water source
  5142.         T:getWater("down")
  5143.         T:go("F2D1R2C2") -- over block 4 of water source
  5144.         T:go("U1", false, 0, false, preferredBlock)
  5145.         T:placeWater("down")
  5146.         T:forward(4)
  5147.         lib.stackBuckets() -- put all buckets in same slot
  5148.         T:dropItem("minecraft:dirt", "up", 0) -- drop dirt up:  clsTurtle.dropItem(self, item, direction, keepAmount)
  5149.         preferredBlock = lib.getStone("down", 6)
  5150.         T:go("R1F2R1U1") -- move to start position
  5151.         for i = 1, 2 do
  5152.             -- build first level of tower: 2 x outside run, 2 x sign run
  5153.             lib.outsideRun(preferredBlock)
  5154.             if i == 1 then -- place door
  5155.                 T:go("L1F1L1F1L1D1")
  5156.                 T:place("door", "forward", false)
  5157.                 T:go("U1L1F1R1F1L1")
  5158.             end
  5159.             T:go("R1F1R1")
  5160.             lib.signRun(preferredBlock, "Pint size\nzombies\nProhibited")
  5161.             T:go("L1F1L1C2", false, 0, false, preferredBlock)
  5162.             T:forward(4) -- miss out centre block
  5163.             T:place(preferredBlock, "down", false)
  5164.             T:go("R1F1R1")
  5165.             lib.signRun(preferredBlock, "Pint size\nzombies\nProhibited")
  5166.             T:go("L1F1L1")
  5167.             lib.outsideRun(preferredBlock)
  5168.             if i == 1 then -- layer 1
  5169.                 T:go("R1F1R1F1R1D1") -- place door
  5170.                 T:place("door", "forward", false)
  5171.                 T:go("U1 R1F1 L1F5 L1U1 F2D1  F2R2 U1") -- go over door
  5172.             else -- layer 2
  5173.                 T:go("L1F5L1F6R2U1") -- over corner of lower platform
  5174.             end
  5175.         end
  5176.         for i = 1, 2 do -- build both sides of platform, leave centre missing
  5177.             lib.baseRun(preferredBlock, 8, "R1F1R1")
  5178.             lib.baseRun(preferredBlock, 8, "L1F1L1")
  5179.             lib.baseRun(preferredBlock, 8, "R1F4R1")
  5180.         end
  5181.         T:go("R1F3L1C2F1C2F1C2F4C2F1C2F1C2", false, 0, false, preferredBlock) --fill in centre row
  5182.         --T:go("R2F6R1F1R1U1") -- go to start of tower base
  5183.         T:go("R2F7R2D3") -- go to start on top of chest
  5184.         T:sortInventory()
  5185.     elseif stage == 2 then
  5186.         -- start on top of chest, should have sufficient stone in inventory
  5187.         T:go("U3L1F1R1F1U1") -- go to start of tower base
  5188.         for i = 1, 7 do -- build 14 block high tower
  5189.             lib.buildSection(preferredBlock, false)
  5190.             lib.buildSection(preferredBlock, true)
  5191.         end
  5192.         T:go("R2F4R1F4R1", false, 0, false, preferredBlock) -- build upper platform (154 blocks remaining)
  5193.         for i = 1, 2 do -- build both sides of upper platform, leave centre missing
  5194.             lib.baseRun(preferredBlock, 12, "R1F1R1")
  5195.             lib.baseRun(preferredBlock, 12, "L1F1L1")
  5196.             lib.baseRun(preferredBlock, 12, "R1F1R1")
  5197.             lib.baseRun(preferredBlock, 12, "L1F1L1")
  5198.             lib.baseRun(preferredBlock, 12, "R1F6R1")
  5199.         end
  5200.         T:go("R1F5 L1C2 F1C2 F1C2 F1C2 F1C2 F4C2 F1C2 F1C2 F1C2 F1C2 ", false, 0, false, preferredBlock) --fill in centre row
  5201.         T:go("R2F5") -- return to drop area
  5202.         lib.buildLift(preferredBlock) -- build bubble lift
  5203.         T:go("F3R1F1R1U1") -- go to start of tower base
  5204.         T:go("C2F4 C2R1F1R1", false, 0, false, preferredBlock)      -- left side layer 21
  5205.         T:go("F2C2 F2C2 L1F1L1", false, 0, false, preferredBlock)   -- centre layer 21
  5206.         T:go("C2F4 C2R2U1", false, 0, false, preferredBlock)        -- right side layer 21
  5207.         T:go("C2F4 C2R1F1R1", false, 0, false, preferredBlock)      -- right side layer 22
  5208.         T:place("fence", "down", false)                         -- fence centre of bottom side layer 22
  5209.         T:go("F2C2 F2L1F1L1", false, 0, false, preferredBlock)      -- centre layer 22
  5210.         T:go("C2F4 C2R2F2L1F1R2D2", false, 0, false, preferredBlock) --ready to place ladder
  5211.         T:place("ladder", "forward", false)
  5212.         T:up(1)
  5213.         T:place("ladder", "forward", false)
  5214.         --T:go("U2R1F4R1F1R1") -- ready to make upper part of tower base
  5215.         T:go("U2R1F4R1F1R1") -- ready to make upper part of tower base
  5216.         for i = 1, 2 do -- build both sides of platform, leave centre missing
  5217.             lib.baseRun(preferredBlock, 8, "R1F1R1")
  5218.             lib.baseRun(preferredBlock, 8, "L1F1L1")
  5219.             lib.baseRun(preferredBlock, 8, "R1F4R1")
  5220.         end
  5221.         T:go("R1F3 L1C2 F1C2 F1C2 F1", false, 0, false, preferredBlock) --fill in centre row
  5222.         T:place("minecraft:soul_sand", "down", false)
  5223.         T:go("F1C2 F2C2 F1C2 F1C2", false, 0, false, preferredBlock)
  5224.         T:go("R2F6R1F1R1U1") -- go to start of tower base
  5225.         -- build 2 levels, finish signs and ladders
  5226.         T:go("C2F2 R1D2 U1", false, 0, false, preferredBlock)
  5227.         T:place("ladder", "down", false)
  5228.         T:turnRight(1)
  5229.         T:place("sign", "forward", false, "UP\n^\n|\n|")
  5230.         T:go("U1R2F2C2 R1F2C2 R1", false, 0, false, preferredBlock) --top right corner
  5231.         T:go("F4C2B2D1", false, 0, false, preferredBlock)
  5232.         T:place("sign", "forward", false, "UP\n^\n|\n|")
  5233.         T:go("U1F2R1F1C2F1R1U1", false, 0, false, preferredBlock) --ready for second level
  5234.         T:go("C2F2 R2D1", false, 0, false, preferredBlock)
  5235.         T:place("sign", "forward", false, "UP\n^\n|\n|")
  5236.         T:go("U1R2F2C2R1", false, 0, false, preferredBlock) --top left corner
  5237.         T:go("F1R1C2F4C2", false, 0, false, preferredBlock) --mid bottom row
  5238.         T:go("L1F1L1C2", false, 0, false, preferredBlock) -- bottom right corner
  5239.         T:go("F2R2D1", false, 0, false, preferredBlock)
  5240.         T:place("sign", "forward", false, "UP\n^\n|\n|")
  5241.         T:go("U1R2F2C2", false, 0, false, preferredBlock) -- top right corner
  5242.         -- return to chest
  5243.         T:go("L1F1L1 F5D23R2", false, 0, false, preferredBlock) -- return to chest
  5244.         T:sortInventory()
  5245.     elseif stage == 3 then
  5246.         --[[ move to top of structure
  5247.         | 4 |
  5248.         |3 5|
  5249.         | X |
  5250.         |2 6|
  5251.         | 1 |
  5252.         ]]
  5253.         local towerHeight = 128 -- even no only suggest 128
  5254.         while turtle.detect() do
  5255.             turtle.up()
  5256.         end
  5257.         T:go("F1U1", false, 0, false, preferredBlock) -- return to finish tower
  5258.         for i = 1, towerHeight do -- 1
  5259.             T:go("C2U1", false, 0, false, preferredBlock)
  5260.         end
  5261.         T:go("F1L1F1R1D2")
  5262.         while turtle.down() do -- 2
  5263.             T:fillVoid("up", {preferredBlock})
  5264.         end
  5265.         T:go("F1R2C1R2F1D1", false, 0, false, preferredBlock)
  5266.         for i = 1, towerHeight / 2 do -- 3
  5267.             T:go("U2C2", false, 0, false, preferredBlock)
  5268.         end
  5269.         T:go("U1F1R1F1R1D1", false, 0, false, preferredBlock) -- back of tower facing front
  5270.         local deviate = false
  5271.         while turtle.down() do -- 4
  5272.             T:place("fence", "up", false)
  5273.             if turtle.down() then
  5274.                 T:fillVoid("up", {preferredBlock})
  5275.             else
  5276.                 T:go("F1R2C1R1F1R1D1", false, 0, false, preferredBlock)
  5277.                 deviate = true
  5278.                 break
  5279.             end
  5280.         end
  5281.         if not deviate then
  5282.             T:go("F1L1F1R1D1", false, 0, false, preferredBlock)
  5283.         end
  5284.         for i = 1, towerHeight / 2 do -- 5
  5285.             T:go("U2C2", false, 0, false, preferredBlock)
  5286.         end
  5287.         T:go("F2R2", false, 0, false, preferredBlock) -- facing back of tower
  5288.         while turtle.down() do -- 6
  5289.             T:fillVoid("up", {preferredBlock}) --layer 129
  5290.         end
  5291.         T:go("F1L2C1U"..towerHeight)
  5292.         T:go("F4R1F3R1U1", false, 0, false, preferredBlock)
  5293.         -- add small platform at the top
  5294.         lib.baseRun(preferredBlock, 8, "R1F1R1")
  5295.         lib.baseRun(preferredBlock, 8, "L1F3L1")
  5296.         lib.baseRun(preferredBlock, 8, "L1F1L1")
  5297.         lib.baseRun(preferredBlock, 8, "R1F1R1")
  5298.         T:go("C2 F1C2 F1C2 F4C2 F1C2 F1C2 R2F3", false, 0, false, preferredBlock) --fill in centre row
  5299.         lib.buildLift(preferredBlock) -- build bubble lift
  5300.     end
  5301.     return {}
  5302. end
  5303.  
  5304. local function createFarmNetworkStorage(R, withStorage, removeLegacy)
  5305.     removeLegacy = removeLegacy or false
  5306.     -- new or converted farm will have:
  5307.     -- 2 modems, 1 barrel per plot
  5308.     -- primary plot and storage needs 1 modem, 1 barrel, 8 chests
  5309.     local lib = {}
  5310.    
  5311.     function lib.createBasement(R, up, down, width, length)
  5312.         -- start facing lower left
  5313.         R.up = up
  5314.         R.down = down
  5315.         R.width = width
  5316.         R.length = length
  5317.         clearRectangle(R)   -- dig 10 x 10 x 2 area, return to starting position
  5318.         -- add network cable, modems and chests
  5319.     end
  5320.    
  5321.     function lib.placeNetwork(count, pattern)
  5322.         for i = 1, count do
  5323.             T:place("computercraft:cable", "up", true)
  5324.             if i < count then
  5325.                 T:go(pattern)
  5326.             end
  5327.         end
  5328.     end
  5329.    
  5330.     -- called when starting at lower left side of plot, facing crops
  5331.     if removeLegacy then
  5332.         T:go("L1F1")
  5333.         while turtle.suckDown() do end
  5334.         --T:dig("down", false)  -- do not bypass chests
  5335.         turtle.digDown()
  5336.         T:place("dirt", "down")
  5337.         T:go("R2F1")
  5338.         while turtle.suck() do end
  5339.         T:go("F1L1 F1U1 R2")    -- remove barrel/wall or double chest. face tree
  5340.         T:place("stone", "down")
  5341.         T:forward(1)
  5342.         T:place("modem", "down")
  5343.         -- could be tree/sapling in front
  5344.         T:forward(1)
  5345.         if T:getBlockType("down") == "minecraft:dirt" then
  5346.             T:place("barrel", "down")
  5347.             T:go("U1x0")
  5348.             T:place("dirt", "up")
  5349.             T:go("B1U2")
  5350.             if T:getBlockType("forward"):find("log") ~= nil then
  5351.                 T:place("sapling", "forward")
  5352.             end
  5353.             T:go("D3F1 R1F1")
  5354.         else
  5355.             T:place("barrel", "down")
  5356.             T:go("R1F1")
  5357.         end
  5358.         while turtle.suckDown() do end
  5359.         T:place("modem", "down")
  5360.         network.attachModem()
  5361.         T:go("F1x2 R2C2 F1L1 F1D1")
  5362.     end
  5363.     T:go("L1D3") -- move below crop field, face N
  5364.     lib.createBasement(R, true, true, 10, 10)   -- ends facing N below water source
  5365.     T:go("U1 F9R1 F1R2 C1R2 F8R1 F1R2 C1R2 F8R1 F1R2 C1R2 F8") -- facing W below water source
  5366.     T:go("F1 R2C1 R1 F1L1 x0x2")    -- move to corner, face along front edge
  5367.     for c = 1, 4 do
  5368.         if c == 1 then
  5369.             lib.placeNetwork(12, "F1x0x2")
  5370.         else
  5371.             lib.placeNetwork(11, "F1x0x2")
  5372.         end
  5373.         if c < 4 then
  5374.             T:go("L1F1 x0x2 L1C1R1")
  5375.         end
  5376.     end
  5377.     -- now in bottom left facing S
  5378.     T:go("L2F1R1")  -- under modem/netwok cable facing in
  5379.     for i = 1, 3 do
  5380.         T:go("L1C1 R1C1 R1C1 L1D1")
  5381.         T:place("computercraft:cable", "up", true)
  5382.     end
  5383.     T:go("C1L1 C1L1 C1L1 C1L1 D1L1")-- ready for clearing rectangle up/down
  5384.     T:place("computercraft:cable", "up", true)
  5385.     T:down(1)
  5386.     T:place("computercraft:cable", "up", true)
  5387.     lib.createBasement(R, true, true, 11, 11)
  5388.     T:turnRight(1)
  5389.     lib.placeNetwork(11, "F1")
  5390.     T:go("R2F4 R1")
  5391.     lib.placeNetwork(11, "F1")
  5392.     T:go("L1F5 L1F9 R1D1 C1")   -- ready to make ladder column
  5393.     for i = 1, 5 do
  5394.         if i < 5 then
  5395.             T:go("U1C1")
  5396.         else
  5397.             T:up(1)
  5398.         end
  5399.         T:place("ladder", "down")
  5400.     end
  5401.     if withStorage then
  5402.         T:go("R1F4 R1F3 D1x2")
  5403.         T:place("computercraft:cable", "down", true) --ready to build chest storage area
  5404.         T:up(1)
  5405.         utils.createStorage()
  5406.         T:go("U1R1 F5R1 F5U1 C2U1 R2")  -- end on plot starting position, facing crops
  5407.     else
  5408.         T:go("U1F2 L1F1 U1C2 U1L1") -- end on plot starting position, facing crops
  5409.     end
  5410.    
  5411.     return "Farm converted to network storage"
  5412. end
  5413.  
  5414. local function createFarm(R, extend) -- 31
  5415.     -- if extend ~= nil then this has been called from createFarmExtension()
  5416.     -- extend "right" or "forward". only adds a single new farm.
  5417.  
  5418.     if extend == nil then
  5419.         extend = ""
  5420.     end
  5421.    
  5422.     local lib = {}
  5423.  
  5424.     function lib.addWaterSource(R, pattern, storage)
  5425.         -- pattern = {"d","c","c","d"} t = place crafting  instead of dirt
  5426.         -- place(self, blockType, damageNo, direction, leaveExisting, signText)
  5427.         T:go("D1x2C2", false, 0, false, R.useBlockType)
  5428.         for i = 1, 4 do
  5429.             T:dig("forward")
  5430.             if pattern[i] == "d" then
  5431.                 T:place("dirt", "forward", false)
  5432.             elseif pattern[i] == "t" then
  5433.                 --if not T:place("minecraft:crafting_table", "forward", false) then
  5434.                 if T:place(storage, "forward", false) then
  5435.                     if T:dropItem("crafting", "forward", 0) then
  5436.                         print("Crafting table -> buried storage")
  5437.                     end
  5438.                 else
  5439.                     T:place("dirt", "forward", false) -- dirt if no storage available
  5440.                 end
  5441.             else
  5442.                 T:place(R.useBlockType, "forward", false)
  5443.             end
  5444.             T:turnRight(1)
  5445.         end
  5446.         T:up(1)
  5447.         T:placeWater("down")    -- ends facing same direction as started, over water source
  5448.     end
  5449.    
  5450.     function lib.placeDirt(count, atCurrent)
  5451.         if atCurrent then
  5452.             local blockType = T:getBlockType("down")
  5453.             if blockType:find("dirt") == nil and blockType:find("grass_block") == nil then
  5454.                 T:place("dirt", "down", false)
  5455.             end
  5456.         end
  5457.         for  i = 1, count do
  5458.             T:forward(1)
  5459.             T:dig("up")
  5460.             local blockType = T:getBlockType("down")
  5461.             if blockType:find("dirt") == nil and blockType:find("grass_block") == nil then
  5462.                 T:place("dirt", "down", false)
  5463.             end
  5464.         end
  5465.     end
  5466.  
  5467.     function lib.placeStorage(storage, storageBackup)
  5468.         T:dig("down")
  5469.         if not T:place(storage, "down", false) then-- place barrel/chest below
  5470.             T:place(storageBackup, "down", false) -- place chest below
  5471.         end
  5472.     end
  5473.    
  5474.     local blockType = ""
  5475.     -- current position if extend == nil is on ground/farm level facing N
  5476.     -- if extend = "right", "front" above water source facing crops
  5477.     local numPlots = 0
  5478.     local storage, storageBackup = utils.setStorageOptions()
  5479.     if extend == "right" then
  5480.         T:up(1)
  5481.         utils.goBack(1)
  5482.         repeat
  5483.             T:forward(11)
  5484.             numPlots = numPlots + 1
  5485.         until not utils.isStorage("down") -- on top of chest, barrel or modem
  5486.         T:go("R1F1R2")  -- move to front right corner of last plot on right side
  5487.         --T:go("R1U1 F1L1 F10L1") -- move to front right corner
  5488.     elseif extend == "forward" then
  5489.         T:go("L1U1")
  5490.         utils.goBack(1)
  5491.         repeat
  5492.             T:forward(11)
  5493.             numPlots = numPlots + 1
  5494.         until not utils.isStorage("down") -- on top of chest, barrel or modem
  5495.         T:go("L1F1 R1x2")   -- could dig tree or sapling + block below
  5496.         --T:go("L2U1 F1R1 F10")
  5497.     elseif extend == "convertWithStorage" then
  5498.         -- convert normal (storage) primary plot farm to networked storage
  5499.         --T:turnLeft(1) -- over water source, facing N
  5500.         local response = createFarmNetworkStorage(R, true, true)
  5501.         return {response}
  5502.     elseif extend == "convert" or extend == "convertWithStorage" then
  5503.         -- convert normal (storage) extended farm to networked storage
  5504.         --T:turnLeft(1) -- over water source, facing N
  5505.         createFarmNetworkStorage(R, false, true)
  5506.         return {"Farm converted to network storage"}
  5507.     else -- new farm.
  5508.         T:up(1) -- assume on the ground, go up 1
  5509.     end
  5510.    
  5511.     -- design change: sapling placed 2 blocks above corner for ease of walking round
  5512.     if R.networkFarm then   -- barrel on corner of plot, modems on each side N/E
  5513.         T:place("barrel", "down", false)
  5514.     else
  5515.         T:place(R.useBlockType, "down", false)
  5516.     end
  5517.     -- stage 2 place sapling
  5518.     T:up(3)
  5519.     T:place("dirt", "down")
  5520.     T:up(1)
  5521.     T:place("sapling", "down") -- plant sapling
  5522.     T:go("F1D4")
  5523.     if R.networkFarm then
  5524.         T:place("modem", "down", false)
  5525.         network.attachModem()
  5526.         T:forward(1)
  5527.         T:place(R.useBlockType, "down", false)
  5528.     else
  5529.         -- stage 2 place double barrel/chest
  5530.         T:go("L1")
  5531.         lib.placeStorage(storage, storageBackup)
  5532.         T:go("R1F1L1")
  5533.         lib.placeStorage(storage, storageBackup)
  5534.         T:turnRight(1)
  5535.     end
  5536.     if extend == "right" then -- cobble wall exists so go forward to its end
  5537.         T:forward(9)
  5538.     else -- new farm or extend forward
  5539.         for i = 1, 9 do -- complete left wall to end of farm
  5540.             T:go("F1 x0x2C 2", false, 0, false, R.useBlockType)
  5541.         end
  5542.     end
  5543.     T:go("R1F1 R1x0 x2C2 F1D1", false, 0, false, R.useBlockType)-- turn round ready for first dirt col
  5544.     lib.addWaterSource(R, {"d","c","c","d"}, storage) -- water at top of farm
  5545.     lib.placeDirt(9, false)     -- place dirt back to start
  5546.     if R.networkFarm then   -- water source next to modem
  5547.         lib.addWaterSource(R, {"c","c","d","d"}, storage)
  5548.         -- T:go(path, useTorch, torchInterval, leaveExisting, preferredBlock
  5549.         T:go("U1F1L1")
  5550.         T:place("modem", "down", false)
  5551.         T:go("F1C2 L1F1 D1", false, 0, false, R.useBlockType)
  5552.     else
  5553.         lib.addWaterSource(R, {"c","c","t","d"}, storage)  -- put barrel / chest in floor
  5554.         T:go("U1F1R2")
  5555.         if T:getBlockType("down"):find(storage) == nil and T:getBlockType("down"):find(storageBackup) == nil then
  5556.             lib.placeStorage(storage, storageBackup)
  5557.         end
  5558.         T:go("R1F1L1")
  5559.         if T:getBlockType("down"):find(storage) == nil and T:getBlockType("down"):find(storageBackup) == nil then
  5560.             lib.placeStorage(storage, storageBackup)
  5561.         end
  5562.         T:go("F1D1")
  5563.     end
  5564.     lib.placeDirt(9, true)
  5565.     local turn = "R"
  5566.     for i = 1, 7 do
  5567.         T:go("F1U1x0C2"..turn.."1F1"..turn.."1x0 x2C2 F1D1", false, 0, false, R.useBlockType)
  5568.         lib.placeDirt(9, true)
  5569.         if turn == "R" then
  5570.             turn = "L"
  5571.         else
  5572.             turn = "R"
  5573.         end
  5574.     end
  5575.     T:go("F1U1x0C2"..turn.."1F1"..turn.."1x0x2C2F1D1", false, 0, false, R.useBlockType)
  5576.     lib.addWaterSource(R, {"d","c","c","d"}, storage)   -- bottom right
  5577.     lib.placeDirt(9, false)
  5578.     lib.addWaterSource(R, {"c","c","d","d"}, storage)   -- top right, facing away from plot
  5579.     T:go("F1U1 R1C2 x0F1 x0x2 C2R1", false, 0, false, R.useBlockType)
  5580.     for i = 1, 11 do    -- build right wall from top of plot to bottom
  5581.         T:go("F1x0x2C2", false, 0, false, R.useBlockType)
  5582.     end
  5583.     T:go("R1F10")               -- ends on top of front storage/ modem facing tree
  5584.     if R.networkFarm then       -- network storage
  5585.         network.attachModem()
  5586.         T:go("R1F1D1R1")    -- over water source, facing E (crops)
  5587.         if extend == "" then    -- primary plot
  5588.             createFarmNetworkStorage(R, true)
  5589.         else
  5590.             createFarmNetworkStorage(R, false)
  5591.         end
  5592.     end
  5593.     if extend == "right" then
  5594.         T:up(1)
  5595.         utils.goBack(numPlots * 11)
  5596.         T:down(1)
  5597.     elseif extend == "forward" then
  5598.         T:go("R1U1F".. numPlots * 11 .."D1L1")
  5599.     end
  5600.     return {"Modular farm completed"}
  5601. end
  5602.  
  5603. local function createFarmExtension(R) -- 32
  5604.     -- assume inventory contains 4 chests, 64 cobble, 128 dirt, 4 water, 1 sapling
  5605.     -- check position by rotating to face tree/sapling
  5606.    
  5607.     --T:setUseLog(true, "farmCreateLog.txt", true)
  5608.     --dbug = true   -- set dbug flag
  5609.     --utils.waitForInput("Logging and debugging enabled")   --utils.waitForInput(message)
  5610.    
  5611.     local extend = "right" -- default
  5612.     if R.subChoice == 1 then
  5613.         extend = "forward"
  5614.     end
  5615.    
  5616.     R = utils.checkFarmPosition(R)
  5617.    
  5618.     if not R.ready then
  5619.         return {"Unable to determine starting position"}
  5620.     end
  5621.    
  5622.     createFarm(R, extend)
  5623.     return {"Modular crop farm extended"}
  5624. end
  5625.  
  5626. local function createFloorCeiling(R) -- 79 size integer 1 to 4
  5627.     --[[
  5628.     R.up = true for ceiling
  5629.     R.down = true for floor
  5630.     R.height = 0 for normal
  5631.     R.height combined with R.up/R.down used for remote access
  5632.     R.data == "random" for random floor placement
  5633.     ]]
  5634.    
  5635.     local lib = {}
  5636.    
  5637.     function lib.goToRemote(R)
  5638.         R.silent = true
  5639.         local depth = 0
  5640.         if R.down then -- floor could be under water
  5641.             while turtle.down() do
  5642.                 depth = depth + 1
  5643.             end
  5644.         elseif R.up then
  5645.             while turtle.up() do
  5646.                 depth = depth + 1
  5647.                 if depth > R.height + 3 then
  5648.                     break
  5649.                 end
  5650.             end
  5651.         end
  5652.         if not(R.height - depth <= 2 or depth - R.height <= 2) then
  5653.             T:up(depth)
  5654.             return "Measured depth/height of "..depth.." > setting: "..R.height
  5655.         end
  5656.         -- not returned so depth acceptable
  5657.         return ""
  5658.     end
  5659.    
  5660.     function lib.checkPosition(R)
  5661.         -- check if block above/below
  5662.         local blockBelow = turtle.detectDown()
  5663.         local blockAbove = turtle.detectUp()
  5664.         if R.subChoice == 2 then -- New floor over existing
  5665.             -- if no block below, assume in correct position and continue
  5666.             -- else move up 1 and continue
  5667.             if blockBelow then T:up(1) end
  5668.         elseif R.subChoice == 4 then -- New ceiling under existing
  5669.             -- if no block above, assume in correct position and continue
  5670.             -- else move down 1 and continue
  5671.             if blockAbove then T:down(1) end
  5672.         end
  5673.     end
  5674.    
  5675.     function lib.placeRow(R, direction, waterPresent)
  5676.         for y = 1, R.length do
  5677.             local blockType = T:getBlockType("forward")
  5678.             if not waterPresent then
  5679.                 if blockType:find("lava") ~= nil or blockType:find("water") ~= nil then
  5680.                     waterPresent = true
  5681.                 end
  5682.             end
  5683.             R = lib.getRandomBlock(R)   -- changes block randomly ONLY if R.data == "random"
  5684.             R = lib.changeCheckered(R)  -- changes to next block type ONLY if R.data == "checked"
  5685.             -- ONLY if R.data == "striped" has already been changed for whole row
  5686.             T:place(R.inventory.useBlock, direction, false) -- leaveExisting = false
  5687.             if y < R.length then
  5688.                 T:forward(1)
  5689.             end
  5690.         end
  5691.         return waterPresent, R
  5692.     end
  5693.    
  5694.     function lib.getRandomBlock(R)
  5695.         if R.data == "random" then
  5696.             local index = math.random(1, #R.inventory.names)    -- random index between 1 and no of block types
  5697.             local success = false
  5698.             for i = index, R.inventory.blockTypeCount do        -- iterate from index upwards
  5699.                 if R.inventory.quantities[i] > 0 then           -- at least 1 block left
  5700.                     R.inventory.useBlock = R.inventory.names[i]
  5701.                     R.inventory.quantities[i] = R.inventory.quantities[i] - 1
  5702.                     R.inventory.blockCount = R.inventory.blockCount - 1
  5703.                     success = true
  5704.                     break
  5705.                 end
  5706.             end
  5707.            
  5708.             if not success then                             -- no blocks left in the range of index -> no of block types
  5709.                 for i = index, 1, -1 do                     -- go backwards from index to 1
  5710.                     if R.inventory.quantities[i] > 0 then       -- at least 1 block left
  5711.                         R.inventory.useBlock = R.inventory.names[i]
  5712.                         R.inventory.quantities[i] = R.inventory.quantities[i] - 1
  5713.                         R.inventory.blockCount = R.inventory.blockCount - 1
  5714.                         success = true
  5715.                         break
  5716.                     end
  5717.                 end
  5718.             end
  5719.         end
  5720.        
  5721.         return R
  5722.     end
  5723.    
  5724.     function lib.changeStripe(R)
  5725.         if R.data == "striped" then
  5726.             R.size = R.size + 1
  5727.             if R.size > R.inventory.blockTypeCount then
  5728.                 R.size = 1
  5729.             end
  5730.             R.inventory.useBlock = R.inventory.names[R.size]
  5731.         end
  5732.         return R
  5733.     end
  5734.    
  5735.     function lib.changeCheckered(R)
  5736.         --swap between 2 block types
  5737.         if R.data == "checked" then
  5738.             if R.inventory.useBlock == R.inventory.names[1] then
  5739.                 R.inventory.useBlock = R.inventory.names[2]
  5740.             else
  5741.                 R.inventory.useBlock = R.inventory.names[1]
  5742.             end
  5743.         end
  5744.         return R
  5745.     end
  5746.    
  5747.     local waterPresent = false
  5748.    
  5749.     if R.data == "random" then
  5750.         math.randomseed(R.inventory.blockCount)
  5751.         print("Using random blocks")
  5752.     elseif R.data == "striped" then
  5753.         print("Using striped pattern")
  5754.     elseif R.data == "checked" then
  5755.         print("Using checkered pattern")
  5756.     end
  5757.     local direction = "down"
  5758.     if R.up then
  5759.         direction = "up"
  5760.     end
  5761.    
  5762.     if R.height > 0 then -- remote placing. go up/down R.height first
  5763.         local message = lib.goToRemote(R)
  5764.         if message ~= "" then   -- error encountered
  5765.             return {message}
  5766.         end
  5767.     end
  5768.    
  5769.     lib.checkPosition(R)
  5770.     -- based on clearRectangle code
  5771.     if R.width == 1 then                    -- single block ahead only
  5772.         waterPresent, R = lib.placeRow(R, direction, waterPresent)
  5773.         T:turnRight(2)                      -- turn at the top of the run
  5774.         T:forward(R.length - 1)             -- return to start
  5775.         T:turnRight(2)                      -- turn round to original position
  5776.     else
  5777.         local iterations = 0                -- R.width = 2, 4, 6, 8 etc
  5778.         if R.width % 2 == 1 then            -- R.width = 3, 5, 7, 9 eg R.width 7
  5779.             iterations = (R.width - 1) / 2  -- iterations 1, 2, 3, 4 for widths 3, 5, 7, 9
  5780.         else
  5781.             iterations = R.width / 2        -- iterations 1, 2, 3, 4 for widths 2, 4, 6, 8
  5782.         end
  5783.         R = lib.changeStripe(R)
  5784.         R = lib.changeCheckered(R)
  5785.         for i = 1, iterations do            -- eg 3 blocks wide, iterations = 1
  5786.             waterPresent, R = lib.placeRow(R, direction, waterPresent)
  5787.             T:go("R1F1R1")
  5788.             R = lib.changeStripe(R)
  5789.             --R = lib.changeCheckered(R)
  5790.             waterPresent, R = lib.placeRow(R, direction, waterPresent)
  5791.             -- if 1 less than end, reposition for next run
  5792.             if i < iterations then
  5793.                 T:go("L1F1L1", false, 0, false)
  5794.                 R = lib.changeStripe(R)
  5795.             end
  5796.         end
  5797.         if R.width % 2 == 1 then  -- additional run and return to base needed
  5798.             T:go("L1F1L1", false, 0, false)
  5799.             R = lib.changeStripe(R)
  5800.             waterPresent, R = lib.placeRow(R, direction, waterPresent)
  5801.             T:turnRight(2)
  5802.             T:forward(R.length - 1)
  5803.         end
  5804.         T:go("R1F"..R.width - 1 .."R1", false, 0, false)
  5805.     end
  5806.  
  5807.     if waterPresent then
  5808.         return {"water or lava found"}
  5809.     end
  5810.    
  5811.     return {""}
  5812. end
  5813.  
  5814. local function createIceCanal(R) -- 55
  5815.     --[[
  5816.         R.subChoice =
  5817.         1: move right into canal, build left towpath
  5818.         2: build left towpath
  5819.         3: build right towpath
  5820.         4: move left into canal, build right towpath
  5821.         5/8: clear path 3 blocks high, place slabs/torches
  5822.         6/7: build alternating ice road / create 3 block high air gap (2 over water)
  5823.     ]]
  5824.     local oTurn = "R"
  5825.     if R.side == "R" then
  5826.         oTurn = "L"
  5827.     end
  5828.     local lib = {}
  5829.    
  5830.     function lib.convertTowpath(R)
  5831.         -- only used to convert existing water canal, so assume towpath already present
  5832.         -- starting position 1 block above existing towpath
  5833.         for i = 1, R.length do
  5834.             if turtle.detectDown() then                         -- eg existing torch
  5835.                 T:dig("down")
  5836.             end
  5837.             local placeSlab = true
  5838.             if R.torchInterval > 0 then                         -- place torches
  5839.                 if i == 1 or i % R.torchInterval == 0 then      -- ready to place torch
  5840.                     --go(path, useTorch, torchInterval, leaveExisting, preferredBlock)
  5841.                     T:go("C2x0", false, 0, false, R.useBlockType)-- place solid block below
  5842.                     T:go("F1R2x0")
  5843.                     T:place("torch", "forward")             -- place torch, move forward
  5844.                     T:turnRight(2)                              -- turn round
  5845.                     placeSlab = false
  5846.                 end
  5847.             end
  5848.             if placeSlab then
  5849.                 T:dig("up")
  5850.                 if not T:place("slab", "down") then         -- break if out of slabs
  5851.                     break
  5852.                 end
  5853.                 if i < R.length then
  5854.                     T:forward(1)                                -- move forward
  5855.                 end
  5856.             end
  5857.         end
  5858.     end
  5859.    
  5860.     function lib.iceCanalTowpath(R)
  5861.         -- move forward placing slabs along the way. If existing water canal, ceiling will be raised by 1 block
  5862.         local torchInterval = 0
  5863.         local placeIce = true
  5864.         for i = 1, R.length do
  5865.             local addTorch = false
  5866.             if R.torchInterval > 0 then
  5867.                 if i == 1 then addTorch = true end
  5868.                 if torchInterval >= R.torchInterval then
  5869.                     addTorch = true
  5870.                     torchInterval = 0
  5871.                 end
  5872.             end
  5873.             lib.iceCanalEdge(R, addTorch, i, placeIce)
  5874.             torchInterval = torchInterval + 1
  5875.             placeIce = not placeIce
  5876.         end
  5877.         T:go(oTurn.."1")
  5878.     end
  5879.    
  5880.     function lib.iceCanalEdge(R, addTorch, numBlocks, placeIce)
  5881.         -- starting position facing canal side
  5882.         local blockType = T:getBlockType("forward")         -- ? air / slab / other
  5883.         local isWater, isSource, isIce = T:isWater("down")  -- water / ice below
  5884.         if blockType:find("slab") == nil then               -- add slab
  5885.             if addTorch then                                -- check if torch needed
  5886.                 -- T:place(blockType, direction, leaveExisting, signText)
  5887.                 if not T:place("stone", "forward", false) then
  5888.                     T:checkInventoryForItem({"stone"}, {math.ceil(R.length / R.torchInterval)}, true)
  5889.                 end
  5890.                 T:go("U1x1 U1x1 D1")                    -- water level to 2 above water level
  5891.                 if T:getItemSlot("minecraft:torch") > 0 then
  5892.                     T:place("torch", "forward", false)
  5893.                 end
  5894.                 T:down(1)                               -- back to above water level
  5895.                 if not isIce then                       -- break block below if NOT ice
  5896.                     T:dig("down")
  5897.                 end
  5898.             else
  5899.                 if not T:place("slab", "forward", false) then
  5900.                     T:checkInventoryForItem({"slab"}, {R.length - numBlocks}, true)
  5901.                 end
  5902.                 if not isSource then                    -- NOT on water, so dig above
  5903.                     T:go("U1x1 U1x1 D2")
  5904.                 end
  5905.                 if not isIce and not isSource then
  5906.                     T:dig("down")
  5907.                 end
  5908.             end
  5909.         else -- slab already on side
  5910.             if addTorch then                            -- check if torch needed
  5911.                 T:dig("forward")
  5912.                 if not T:place("stone", "forward", false) then
  5913.                     T:checkInventoryForItem({"stone"}, {math.ceil(R.length / R.torchInterval)}, true)
  5914.                 end
  5915.                 T:go("U1x1 U1x1 D1")                    -- water level to 2 above water level
  5916.                 if T:getItemSlot("minecraft:torch") > 0 then
  5917.                     T:place("torch", "forward", false)
  5918.                 end
  5919.                 T:down(1)                               -- back to above water level
  5920.                 if not isIce then                       -- break block below if NOT ice
  5921.                     T:dig("down")
  5922.                 end
  5923.             end
  5924.         end
  5925.         if R.data == "ice" and placeIce and not isIce then -- R.data = "ice", placeIce = true, not already ice present
  5926.             T:place("ice", "down", true)
  5927.         end
  5928.         if numBlocks < R.length then
  5929.             T:go(oTurn.."1F1"..R.side.."1")
  5930.         end
  5931.     end
  5932.    
  5933.     function lib.convertIcepath(length)
  5934.         -- use only for placing ice to convert a water canal
  5935.         -- place ice on alternate blocks until length reached or run out of ice
  5936.         local placeIce = true
  5937.         for i = 1, length do
  5938.             if T:getBlockType("down"):find("ice") == nil then -- no ice below
  5939.                 T:dig("down") -- remove any existing block
  5940.                 if placeIce then
  5941.                     if not T:place("ice", "down", true) then -- out of ice
  5942.                         break
  5943.                     end
  5944.                     if i == length - 1 then
  5945.                         break
  5946.                     end
  5947.                 end
  5948.             else -- ice already below
  5949.                 placeIce = true
  5950.             end
  5951.             T:go("U1x0 D1F1")
  5952.             placeIce = not placeIce -- reverse action
  5953.         end
  5954.     end
  5955.    
  5956.     function lib.convertToAir(length)
  5957.         -- use only for converting a water canal. start at ground level
  5958.         -- dig up/down/forward to clear space
  5959.         for i = 1, length + 1 do
  5960.             T:go("U1x0D1")
  5961.             if i < length + 1 then
  5962.                 T:go("x2F1")
  5963.             else
  5964.                 T:dig("down")
  5965.             end
  5966.         end
  5967.     end
  5968.    
  5969.     function lib.initialise(R)
  5970.         if R.subChoice == 1 or R.subChoice == 4 then       
  5971.             local blockType = T:getBlockType("down")        -- ? at ground/water level or on top of existing slab
  5972.             if blockType:find("slab") ~= nil then           -- slab already present
  5973.                 T:go(oTurn.."1F1D1"..R.side.."2")           -- move right/left forward, down onto ice canal top, face canal wall
  5974.             elseif blockType:find("torch") ~= nil then
  5975.                 T:go(oTurn.."1F1D2"..R.side.."2")           -- move right/left forward, down onto ice canal top, face canal wall
  5976.             else                                            -- assume on ground / water level
  5977.                 T:go(oTurn.."1F1"..R.side.."2")             -- move right/left forward onto ice canal top, face canal wall
  5978.             end
  5979.         else
  5980.             if T:isWater("forward") then                    -- player put turtle inside canal water
  5981.                 T:up(1)
  5982.             end
  5983.             T:go(R.side.."1")                               -- face canal wall
  5984.         end
  5985.     end
  5986.    
  5987.     if R.length == 0 then R.length = 1024 end
  5988.  
  5989.     if R.subChoice <= 4  then                               -- towpath 1,2,3,4
  5990.         lib.initialise(R)                                   -- reposition
  5991.         lib.iceCanalTowpath(R)                              -- build towpath
  5992.     elseif R.subChoice == 5 or R.subChoice == 8 then        -- assume placed on towpath
  5993.         if T:getBlockType("down"):find("slab") ~= nil then  -- slab already present
  5994.             T:go("F1")
  5995.         else
  5996.             T:up(1)
  5997.         end
  5998.         lib.convertTowpath(R)
  5999.     elseif R.subChoice == 6 or R.subChoice == 7 then   
  6000.         if R.data == "ice" then                         -- assume placed on existing ice or initial ice position
  6001.             lib.convertIcepath(R.length)-- place ice
  6002.         else                                            -- assume placed on empty path
  6003.             lib.convertToAir(R.length)                  -- clear 3 high area
  6004.         end
  6005.     end
  6006.    
  6007.     return {}
  6008. end
  6009.  
  6010. local function createLadder(R) -- 12
  6011.     -- createLadder(R.data ="bedrock", R.height = 70, R.depth = -48)
  6012.     -- go(path, useTorch, torchInterval, leaveExisting)
  6013.     -- place(blockType, damageNo, direction, leaveExisting)
  6014.     local lib = {}
  6015.    
  6016.     function lib.placeLadder(direction, ledge, i, height)
  6017.         -- 1 check both sides and behind
  6018.         local fluid = false
  6019.         local block = T:isWaterOrLava("forward", ledge)
  6020.         if block:find("water") ~= nil or block:find("lava") ~= nil then
  6021.             --[[ surround 2 block shaft with blocks ]]
  6022.             T:go("R1C1 R1C1 R1C1 R1F1 L1C1 R1C1 R1C1 R1C1F1 R2C1 x1")
  6023.         else
  6024.             --[[ no water/lava so prepare ladder site]]
  6025.             T:go("F1 L1C1 R1C1 R1C1 L1B1", false, 0, true)
  6026.         end
  6027.         if not T:place("ladder", "forward", false) then
  6028.             T:checkInventoryForItem({"ladder"}, {height - i}, false)
  6029.         end
  6030.         -- 3 check if ledge, torch
  6031.         if ledge == 0 and i > 1 then -- place block above unless new ladder
  6032.             T:place("common", direction, false) -- any common block
  6033.         elseif ledge == 1 then
  6034.             T:place("minecraft:torch", direction, false)
  6035.         elseif ledge == 2 then
  6036.             ledge = -1
  6037.         end
  6038.        
  6039.         return ledge
  6040.     end
  6041.    
  6042.     local retValue = {}
  6043.     local ledge = 0
  6044.     local height = math.abs(R.depth - R.height) --height of ladder
  6045.     local blockType = T:getBlockType("forward")
  6046.     if R.up then -- create ladder from current level to height specified
  6047.         for i = 1, height do -- go up, place ladder as you go
  6048.             ledge = lib.placeLadder("down", ledge, i, height) -- ladder placed forward, stone ledge for torch placed down
  6049.             if i <  height then
  6050.                 T:up(1)
  6051.                 ledge = ledge + 1
  6052.             end
  6053.         end    
  6054.     else -- R.down = true: ladder towards bedrock      
  6055.         local success = true
  6056.         local numBlocks, errorMsg = 0, ""
  6057.         T:down(1)
  6058.         for i = 1, height do -- go down, place ladder as you go
  6059.             ledge = lib.placeLadder("up", ledge, i, heigt) -- ladder placed forward, stone torch placed up
  6060.             --success, blocksMoved, errorMsg, blockType = clsTurtle.down(self, steps, getBlockType)
  6061.             if i < height then
  6062.                 success, numBlocks, errorMsg, blockType = T:down(1, true) -- true = return blockType
  6063.                 ledge = ledge + 1
  6064.             end
  6065.             -- if looking for stronghold then check for stone_bricks
  6066.             if blockType:find("stone_bricks") ~= nil then
  6067.                 table.insert(retValue, "Stronghold discovered")
  6068.                 break -- stop descent at stronghold
  6069.             end
  6070.         end
  6071.         -- if user requested shelter create chamber at this level
  6072.         if R.data == "chamber" then -- user has chosen to build a chamber
  6073.             table.insert(retValue, "Shelter constucted at level".. R.depth)
  6074.             if blockType:find("bedrock") ~= nil then
  6075.                 T:findBedrockTop(0) -- test to check if on safe level immediately above tallest bedrock
  6076.             end
  6077.             -- In shaft, facing start direction, on lowest safe level
  6078.             -- create a square space round shaft base, end facing original shaft, 1 space back
  6079.             T:go("L1n1 R1n3 R1n2 R1n3 R1n1", false, 0, true)
  6080.             T:go("U1Q1 R1Q3 R1Q2 R1Q3 R1Q1 R1D1", false, 0, true)
  6081.         end
  6082.     end
  6083.    
  6084.     return retValue
  6085. end
  6086.  
  6087. local function createLadderToWater(R) -- 86
  6088.     -- go down to water/lava with alternaate solid/open layers
  6089.     -- create a working area at the base
  6090.     -- Return to surface facing towards player placing ladders
  6091.     local inAir = true
  6092.     local numBlocks, errorMsg = 0, ""
  6093.     local height = 2
  6094.     local blockType = T:getBlockType("down")
  6095.     if blockType ~= "" then -- not over air
  6096.         T:forward(1)
  6097.     end
  6098.     T:go("R2D1") -- face player, go down 2
  6099.     while inAir do --success = false when hits water/lava
  6100.         blockType = T:isWaterOrLava("down")
  6101.         if blockType:find("water") ~= nil or blockType:find("lava") ~= nil then
  6102.             inAir = false
  6103.         end
  6104.         T:go("C1R1 C1R2 C1R1", false, 0, false) -- surround front  and sides with cobble
  6105.         if inAir then
  6106.             T:down(1)
  6107.             height = height + 1
  6108.         end
  6109.         T:place("ladder", "up")
  6110.     end
  6111.     -- In shaft, facing opposite start direction, on water/lava, ladders above
  6112.     T:go("C2", false, 0, false)
  6113.     utils.goBack(1)
  6114.     T:place("ladder", "forward")
  6115.     T:up(3)
  6116.     height = height - 3
  6117.     for i = 1, height do
  6118.         if i < height then
  6119.             T:go("C2U1", false, 0, false)
  6120.         else
  6121.             T:go("C2", false, 0, false)
  6122.         end
  6123.     end
  6124.  
  6125.     return {}
  6126. end
  6127.  
  6128. local function createMine() -- 11
  6129.     -- go(path, useTorch, torchInterval, leaveExisting, preferredBlock)
  6130.     T:clear()  
  6131.     T:go("m32U1R2M16", true, 8, true) -- mine ground level, go up, reverse and mine ceiling to mid-point
  6132.     T:go("U2D2") -- create space for chest
  6133.     T:place("minecraft:chest", "up", false)
  6134.     T:emptyTrash("up")
  6135.     T:go("D1R1m16U1R2M16", true, 8, true) -- mine floor/ceiling of right side branch
  6136.     T:emptyTrash("up")
  6137.     T:go("D1m16U1R2M16", true, 8, true) -- mine floor/ceiling of left side branch
  6138.     T:emptyTrash("up")
  6139.     T:go("L1M15F1R1D1", true, 8, true) -- mine ceiling of entry corridor, turn right
  6140.     T:go("F1x0 F1x0 n14 R1n32 R1n32 R1n32 R1n14 F1x0 F1U1", true, 8, true)-- mine floor of 36 x 36 square corridor
  6141.     T:go("R1F16R2") --return to centre
  6142.     T:emptyTrash("up")
  6143.     T:go("F16R1") --return to entry shaft
  6144.     T:go("F2Q14R1Q32R1Q32R1Q32R1Q14F2R1", true, 8, true) --mine ceiling of 36x36 square corridor. return to entry shaft + 1
  6145.     T:go("F16R2") --return to centre
  6146.     T:emptyTrash("up")
  6147.     -- get rid of any remaining torches
  6148.     while T:getItemSlot("minecraft:torch") > 0 do
  6149.         turtle.select(T:getItemSlot("minecraft:torch"))
  6150.         turtle.dropUp()
  6151.     end
  6152.     T:go("F16R1F1R1") --return to shaft + 1
  6153.     for i = 1, 8 do
  6154.         T:go("N32L1F1L1", true, 8, true)
  6155.         T:go("N16L1F"..(i * 2).."R2", true, 8, true)
  6156.         T:emptyTrash("up")
  6157.         if i < 8 then
  6158.             T:go("F"..(i * 2).."L1N16R1F1R1", true, 8, true)
  6159.         else
  6160.             T:go("F"..(i * 2).."L1N16L1", true, 8, true)
  6161.         end
  6162.     end
  6163.     T:go("F17L1") -- close gap in wall, return to ladder + 1
  6164.     for i = 1, 8 do
  6165.         T:go("N32R1F1R1", true, 8, true)
  6166.         T:go("N16R1F"..(i * 2).."R2", true, 8, true)
  6167.         T:emptyTrash("up")
  6168.         if i < 8 then
  6169.             T:go("F"..(i * 2).."R1N16L1F1L1", true, 8, true)
  6170.         else
  6171.             T:go("F"..(i * 2).."R1N16R1", true, 8, true)
  6172.         end
  6173.     end
  6174.     T:go("F16R1")
  6175.     T:clear()
  6176.     return{"Mining operation complete"}
  6177. end
  6178.  
  6179. local function createMobFarmCube(R) -- 61, 62
  6180.     --[[
  6181.     Part 1 / 3 Mob Spawner Farm
  6182.     blaze = true: blaze spawner in nether
  6183.         R.subChoice is set to:
  6184.         1 = on top of spawner
  6185.         2 = line of sight
  6186.         3 = room below
  6187.     blaze = false: overworld mob spawner
  6188.         R.subChoice is set to:
  6189.         1 = on top of spawner
  6190.         2 = bottom left corner
  6191.         3 = top left corner
  6192.         4 = bottom right, corner
  6193.         5 = top right corner
  6194.         R.width / R.length set by player (external size)
  6195.     ]]
  6196.     local blaze = true
  6197.     if R.data == "spawner" or R.data == "chest" then
  6198.         blaze = false
  6199.     end
  6200.     local continue = false
  6201.     if R.data == "restart" then
  6202.         continue = true
  6203.     end
  6204.     R.direction = "clock"       -- starting on right side
  6205.     if not blaze then
  6206.         print("R.width: "..tostring(R.width))
  6207.         R.width = R.width - 2       -- internal width
  6208.         R.length = R.length - 2     -- internal length
  6209.        
  6210.         if R.subChoice == 2 or R.subChoice == 3 then
  6211.             R.direction = "anticlock"
  6212.         end
  6213.     end
  6214.     -- R.data allows for 2-part operation "blaze" = main cube, "restart" = killzone
  6215.     T:clear()
  6216.     local lib = {}
  6217.    
  6218.     function lib.floorSection(length)
  6219.         for i = 1, length do        -- starts on top left corner
  6220.             T:go("C2")
  6221.             if i < length then
  6222.                 T:forward(1)
  6223.             else
  6224.                 T:go("R1F1")
  6225.             end
  6226.         end
  6227.     end
  6228.    
  6229.     function lib.wallSection(blaze)
  6230.         blaze = blaze or false
  6231.         for i = 1, 4 do
  6232.             for j = 1, 11 do
  6233.                 if blaze then
  6234.                     T:place("slab", "up", false)
  6235.                     T:go("C2", false, 0, false)
  6236.                 else
  6237.                     T:go("C0C2", false, 0, false)
  6238.                 end
  6239.                 if j < 11 then
  6240.                     T:forward(1)
  6241.                     T:go("R2C1L2", false, 0, false)
  6242.                 else
  6243.                     T:turnRight(1)
  6244.                 end
  6245.             end
  6246.         end
  6247.     end
  6248.    
  6249.     function lib.ceiling(blaze)
  6250.         -- T:go(path, useTorch, torchInterval, leaveExisting, preferredBlock)
  6251.         -- all outer walls complete, now for remaining 9x9 ceiling
  6252.         blaze = blaze or false
  6253.         for i = 1, 9 do
  6254.             for j = 1, 9 do
  6255.                 if blaze then
  6256.                     T:place("slab", "up", false)
  6257.                     T:dig("down")
  6258.                 else
  6259.                     T:go("C0x2", false, 0, true)
  6260.                 end
  6261.                 if j < 9 then
  6262.                     T:forward(1)
  6263.                 else    -- end of length
  6264.                     local place = false
  6265.                     if i%2 == 1 then -- odd numbers 1,3,5,7,9
  6266.                         if i < 9 then
  6267.                             place = true
  6268.                             T:go("R1F1 R1")
  6269.                         end
  6270.                     else
  6271.                         place = true
  6272.                         T:go("L1F1 L1")
  6273.                     end
  6274.                     if place then
  6275.                         if blaze then
  6276.                             T:place("slab", "up", false)
  6277.                             T:dig("down")
  6278.                         else
  6279.                             T:go("C0x2", false, 0, true)
  6280.                         end
  6281.                     end
  6282.                 end
  6283.             end
  6284.         end
  6285.     end
  6286.    
  6287.     function lib.clearWall(length)
  6288.         for i = 1, 4 do
  6289.             for j = 1, length do
  6290.                 if j < length then
  6291.                     T:go("x0x2F1")
  6292.                 else
  6293.                     T:go("x0x2R1")
  6294.                 end
  6295.             end
  6296.         end
  6297.     end
  6298.    
  6299.     function lib.isSpawner()
  6300.         local blockType = T:getBlockType("down")
  6301.         if blockType:find("spawner") ~= nil then
  6302.             return true, "top"
  6303.         end
  6304.         blockType = T:getBlockType("up")
  6305.         if blockType:find("spawner") ~= nil then
  6306.             return true, "bottom"
  6307.         end
  6308.         blockType = T:getBlockType("forward")
  6309.         if blockType:find("spawner") ~= nil then
  6310.             return true, "forward"
  6311.         end
  6312.         return false, ""
  6313.     end
  6314.    
  6315.     function lib.placeFloor(width, length, blockType)
  6316.         -- T:place(blockType, direction, leaveExisting, signText)
  6317.         for i = 1, width do            
  6318.             for j = 1, length do
  6319.                 T:place(blockType, "down", false)
  6320.                 if j < length then
  6321.                     T:forward(1)
  6322.                 else
  6323.                     if i%2 == 1 then -- odd numbers 1,3,5,7,9
  6324.                         if i < width then
  6325.                             T:go("R1F1R1", false, 0, true)
  6326.                         end
  6327.                     else
  6328.                         T:go("L1F1L1", false, 0, true)
  6329.                     end
  6330.                 end
  6331.             end
  6332.         end
  6333.     end
  6334.    
  6335.     function lib.searchStrip(distance)
  6336.         --go forward until hit either a wall or a chest
  6337.         local blocks = 1
  6338.         print("Searching strip, distance = "..distance)
  6339.         while blocks < distance do --max travel is external width/length
  6340.             if turtle.forward() then
  6341.                 blocks = blocks + 1
  6342.             else
  6343.                 print("Checking for chest")
  6344.                 if lib.isChest("forward") then
  6345.                     if turtle.forward() then
  6346.                         blocks = blocks + 1
  6347.                     end
  6348.                 else
  6349.                     break
  6350.                 end
  6351.             end
  6352.         end
  6353.     end
  6354.    
  6355.     function lib.findChests(R)
  6356.         for i = 1, 4 do
  6357.             print("Searching wall "..i)
  6358.             if i == 1 or i == 3 then
  6359.                 lib.searchStrip(R.width)    -- find and empty chests along width
  6360.             else
  6361.                 lib.searchStrip(R.length)   -- find and empty chests along length
  6362.             end
  6363.             if R.direction == "clock" then
  6364.                 T:turnRight(1)
  6365.             else
  6366.                 T:turnLeft(1)
  6367.             end
  6368.         end
  6369.  
  6370.         -- checked all walls
  6371.         T:go("F".. math.floor(R.width / 2)) -- now at mid-dungeon, next to wall
  6372.         if T:getItemSlot("chest") > 0 then
  6373.             if R.direction == "clock" then
  6374.                 T:turnLeft(1)
  6375.             else
  6376.                 T:turnRight(1)
  6377.             end
  6378.             -- now at mid-dungeon, facing wall
  6379.             -- deposit chest outside the dungeon
  6380.             while turtle.back() do end -- backing spawner
  6381.             T:go("x0 F1x0 F1x0 F1x0 F1x0 F1x0 F1x0 x1x2")
  6382.             -- place(self, blockType, damageNo, direction, leaveExisting, signText)
  6383.             T:place("chest", "forward", false)
  6384.             -- empty out all except stone and slab
  6385.             T:emptyInventorySelection("forward", {"cobble", "tuff", "slab", "granite", "andesite", "diorite" }, {0,0,0,0,0,0})
  6386.             T:turnLeft(2)
  6387.             T:checkInventoryForItem({"stone"}, {512}, false, "Full cube uses ~700 blocks\nEstimate your requirements")
  6388.             while turtle.forward() do end -- facing spawner
  6389.         else
  6390.             if R.direction == "clock" then
  6391.                 T:turnRight(1)
  6392.             else
  6393.                 T:turnLeft(1)
  6394.             end
  6395.         end -- now at mid-dungeon, facing spawner
  6396.     end
  6397.    
  6398.     function lib.enterDungeon(R)
  6399.         --[[ find and empty any chests, return to dungeon wall ]]
  6400.         print("Entering dungeon")
  6401.         local blockType = T:getBlockType("forward")
  6402.         if blockType == "" then -- nothing in front.Error
  6403.             return false, "No block in front: Check position."
  6404.         else -- attempt entry into dungeon wall
  6405.             if R.subChoice == 2 then        -- bottom left
  6406.                 T:go("R1F1 L1U2 F2R1")
  6407.             elseif R.subChoice == 3 then    -- top left
  6408.                 T:go("R1F1 L1D1 F2R1")
  6409.             elseif R.subChoice == 4 then    -- bottom right
  6410.                 T:go("L1F1 R1U2 F2L1")
  6411.             elseif R.subChoice == 5 then    -- top right
  6412.                 T:go("L1F1 R1D1 F2L1")
  6413.             end
  6414.             while turtle.down() do end -- either on floor or chest
  6415.             if lib.isChest("down") then
  6416.                 turtle.down()
  6417.             end
  6418.             print("Searching for chests")
  6419.             lib.findChests(R)   -- go round inside walls  emptying chests. Finish mid-wall
  6420.         end
  6421.         return true, ""-- success, message
  6422.     end
  6423.    
  6424.     function lib.isChest(direction)
  6425.         direction = direction or "forward"
  6426.         local blockType = T:getBlockType(direction)
  6427.         if blockType:find("chest") ~= nil then -- chest found. early stages so empty and break it
  6428.             print("Chest found")
  6429.             while T:suck(direction) do end
  6430.             T:dig(direction, false) -- false prevents checking for chests
  6431.             return true
  6432.         elseif blockType:find("torch") ~= nil then
  6433.             print("Torch found")
  6434.             T:dig(direction, false) -- false prevents checking for chests
  6435.             return true
  6436.         end
  6437.         return false
  6438.     end
  6439.        
  6440.     function lib.findSpawner(blaze)
  6441.         local moves  = 0
  6442.         local quit = false
  6443.         -- assume turtle placed on centre of inside spawner wall in front of spawner
  6444.         -- or as close as possible in Nether
  6445.         print("Checking if next to spawner")
  6446.         local found, position = lib.isSpawner() -- true/false, top/bottom/nil
  6447.         if not found then -- move forward towards spawner
  6448.             print("Not close to spawner")
  6449.             while turtle.forward() and not quit do
  6450.                 moves = moves + 1
  6451.                 if moves > 16 then
  6452.                     quit = true
  6453.                 end
  6454.             end
  6455.             found, position = lib.isSpawner() -- true/false, top/bottom/nil
  6456.             if not found then
  6457.                 if blaze then -- could be behind a wall
  6458.                     print("Assuming blaze spawner behind a wall")
  6459.                     T:forward(1)
  6460.                     moves = moves + 1
  6461.                     while turtle.forward() and not quit do
  6462.                         moves = moves + 1
  6463.                         if moves > 16 then
  6464.                             quit = true
  6465.                         end
  6466.                     end
  6467.                     found, position = lib.isSpawner() -- true/false, top/bottom/nil
  6468.                     if not found then
  6469.                         T:go("R2F"..moves + 2 .."R2")
  6470.                     end
  6471.                 end
  6472.             end
  6473.         end
  6474.        
  6475.         return found, position
  6476.     end
  6477.    
  6478.    
  6479.     if not continue then -- new mob cube either dungeon or blaze
  6480.         -- clsTurtle.go(self, path, useTorch, torchInterval, leaveExisting, preferredBlock)
  6481.         -- determine spawner position level 4, move to top of spawner (level 6)
  6482.         print("Checking if already at spawner")
  6483.         local found, position = lib.isSpawner() -- already on spawner?
  6484.         if blaze then
  6485.             if not found then -- away from spawner
  6486.                 if R.subChoice == 3 then
  6487.                     T:go("U5")
  6488.                 end
  6489.                 found, position = lib.findSpawner(blaze)
  6490.             end
  6491.         else -- go to bottom of dungeon and empty chests
  6492.             if not found then --outside dungeon
  6493.                 local success, message = lib.enterDungeon(R)
  6494.                 if not success then
  6495.                     return {message}
  6496.                 end
  6497.                 found, position = lib.findSpawner(blaze) -- is spawner in front / above / below?
  6498.             end
  6499.         end
  6500.        
  6501.         if found then -- true: move to correct starting position
  6502.             --[[
  6503.             1 |c|c|c|c|c|c|c|c|c|c|c|
  6504.             2 |w| | | | | | | | | |w|
  6505.             3 |w| | | | | | | | | |w|
  6506.             4 |w| | | | | | | | | |w|
  6507.             5 |w| | | | | | | | | |w|
  6508.             6 |w| | | | |s| | | | |w|
  6509.             7 |w| | | | | | | | | |w|
  6510.             8 |w| | | | | | | | | |w|
  6511.             9 |w| | | | | | | | | |w|
  6512.            10 |w| | | | | | | | | |w| exit level for overworld
  6513.            11 |f|f|f|f|f|f|f|f|f|f|f|
  6514.            12 |f|f|f|f|f|f|f|f|f|f|f| sub floor for overworld
  6515.                1 2 3 4 5 6 7 8 9 1 1
  6516.                                  0 1
  6517.             ]]
  6518.             -- T:go(path, useTorch, torchInterval, leaveExisting, preferredBlock)
  6519.             if position == "bottom" then
  6520.                 T:go("B1U2F1")
  6521.             elseif position == "forward" then
  6522.                 T:go("U1F1")
  6523.             end
  6524.            
  6525.             T:up(1)
  6526.             T:place("slab", "down", true)               -- place slab on top T:place(blockType, direction, leaveExisting)
  6527.             -- go up 2 blocks, forward 5, right, forward 5, right
  6528.             T:go("U2F5 R1F5 R1")                            -- Level 2: now placed 1 below ceiling inside wall, top right corner of new dungeon
  6529.             lib.wallSection(blaze)                          -- fix layers 1, 2, 3 including ceiling margin turtle at Level 2           
  6530.             T:go("F1R2 C1L1 F1R2 C1R1", false, 0, false)    -- exit wall, repair behind, still Level 2
  6531.             lib.ceiling(blaze)                              -- fix ceiling, end opposite corner to start
  6532.             T:go("R2D3")                                    -- clear the inner walls inside original dungeon
  6533.             lib.clearWall(9)                                -- clear the 9 x 9 area around the spawner
  6534.             T:go("F1R1F1L1")
  6535.             lib.clearWall(7)                                -- clear the 7 x 7 area around the spawner
  6536.             T:go("F1R1F1L1")
  6537.             lib.clearWall(5)                                -- clear the 5 x 5 area around the spawner. Also needed for cave spiders
  6538.             T:go("R2F1R1F1R1")
  6539.             T:go("F7R1 F8L1F1R2", false, 0, false)          -- return from ceiling, enter wall below previous section: Level 5
  6540.             lib.wallSection()                               -- deal with areas from spawner level up (4,5,6). walls only   
  6541.             T:go("F1R2 C1L1 F1R2 C1R1 D3", false, 0, false) -- exit wall, repair behind, embed 1 below original floor: Level 8
  6542.             clearRectangle({width = 9, length = 9, up = true, down = true}) -- clear levels 7,8,9
  6543.             T:go("L1F1 L1F1L2", false, 0, false)            -- go inside wall sectio, ready for next wall section
  6544.             lib.wallSection()                               -- deal with walls on levels 7,8,9
  6545.             T:go("F1R2 C1L1 F1R2 C1R1 D3", false, 0, false) -- exit wall, repair behind, embed 4 below original floor: Level 11
  6546.             --print("Check: about to clear 3 floors 3 below spawner") read()
  6547.             clearRectangle({width = 9, length = 9, up = true, down = true}) -- clear levels 10,11,12
  6548.            
  6549.             if blaze then
  6550.                 T:go("L1F1 L1F1L2", false, 0, false)            -- ready for next wall section
  6551.                 lib.wallSection()                               -- wall on layers 10,11,12
  6552.                 T:go("F1R2 C1L1 F1R2 C1R1 D3", false, 0, false) -- exit wall, repair behind, embed 1 below original floor: Level 8
  6553.                 clearRectangle({width = 9, length = 9, up = true, down = true}) -- clear levels 13, 14, 15
  6554.             end
  6555.            
  6556.             T:go("L1F1L1F1L2", false, 0, false) -- ready for next wall section
  6557.             --print("Check: level 11, 5 north, 5 east")
  6558.             --read()
  6559.             lib.wallSection()   -- wall on layers 10,11,12 or 12,13,14 if blaze
  6560.             T:go("F1R1 F1R2 C1R1 U1", false, 0, false) -- exit wall, repair behind: Level 10, facing entry point top right corner
  6561.             T:down(1)
  6562.             if blaze then
  6563.                 lib.placeFloor(9, 9, brick)             -- place brick floor on level 14
  6564.                 T:go("L1F4 R1F2 U4")                    -- ends facing out to lower chamber ?below staircase
  6565.                 -- now needs to build killzone
  6566.             else
  6567.                 lib.placeFloor(9, 9, "stone")           -- ends facing wall on entrance side
  6568.                 T:go("U1R2")
  6569.                 lib.placeFloor(9, 9, "stone")           -- ends facing wall on opposite side
  6570.                 -- return to mid-point front
  6571.                 T:go("R2F8 R1F4 L1F2")                  -- exit at bottom of dungeon
  6572.                 T:go("x1U1 x1U1 x1U1 x1D3 R2")          -- rise to chest, then return ready for next stage 
  6573.                 -- ends with turtle facing spawner, in front of exit hole              
  6574.             end
  6575.         else
  6576.             return
  6577.             {
  6578.                 "Spawner not found. Place me on top,",
  6579.                 "immediately below, or facing it.",
  6580.                 "\nEnter to quit"
  6581.             }
  6582.         end
  6583.     end
  6584.     if continue then
  6585.         T:clear()
  6586.         local text =
  6587. [[~yellow~Items required to continue:
  6588.  
  6589. ~lightGray~slabs            81
  6590. ~orange~nether bricks    88
  6591. ~yellow~lava buckets     4
  6592. ~brown~chest / barrel   1
  6593. ~gray~hopper           1      
  6594.  
  6595. ~red~WARNING ~yellow~Inventory ejects items if you
  6596. continue! ~orange~(Choose at next menu.)
  6597.  
  6598. ~white~Next -> Enter]]
  6599.         menu.colourText(nil, text, true)
  6600.         read()
  6601.         pp.itemColours = {colors.red, colors.lime}
  6602.         local choice = menu.new("Choose your option", {"Continue with build", "Re-start later"}, pp, "Type number + Enter") -- 1 = continue, 2= quit
  6603.         if choice == 1 then
  6604.             choice = menu.new("Choose your option", {"Turtle is above me", "Turtle is below or next to me"}, pp, "Type number + Enter") -- 1 = up, 2= down
  6605.             if choice == 1 then
  6606.                 T:emptyInventory("down")
  6607.             else
  6608.                 T:emptyInventory("up")
  6609.             end
  6610.         else
  6611.             return {"Blaze spawner partial build halted"}
  6612.         end
  6613.        
  6614.         --clsTurtle.getItemSlot(self, item, useDamage): return slotData.lastSlot, slotData.leastModifier, total, slotData
  6615.         T:checkInventoryForItem({brick}, {88})
  6616.         T:checkInventoryForItem({"slab"}, {81})
  6617.         T:checkInventoryForItem({"lava"}, {4})
  6618.         T:checkInventoryForItem({"sign"}, {1})
  6619.         T:checkInventoryForItem({"hopper"}, {1})
  6620.         T:checkInventoryForItem({"chest", "barrel"}, {1, 1})
  6621.         print("Stand clear. Starting in 2 secs")
  6622.         os.sleep(2)    -- pause for 2 secs to allow time to press esc
  6623.         -- place upper floor and slabs under it
  6624.         T:go("R2D4F2 L1F4 R1U4")   
  6625.         lib.placeFloor(9, 9, brick)             -- place brick floor on level 10
  6626.         T:down(2)
  6627.         T:place(brick, "up")
  6628.         T:go("D1R2")
  6629.         lib.ceiling(true)                       -- true uses slabs
  6630.        
  6631.         -- place lava
  6632.         T:go("L1F4 L1F4 U4")                    -- through hole in ceiling
  6633.         T:go("F4 L1F4 R2")
  6634.         for i = 1, 4 do
  6635.             T:place("lava", "down")
  6636.             T:go("F8 R1")
  6637.         end
  6638.         T:go("F4 R1F4 D2 F1 R2")
  6639.         -- place sign and repair above
  6640.         T:dig("forward")
  6641.         T:place("sign", true, "")
  6642.         T:down(1)
  6643.         T:place(brick, "up")
  6644.         T:forward(1)
  6645.         -- place blocks under exit hole
  6646.         for i = 1, 4 do
  6647.             T:place(brick, "forward")
  6648.             T:turnRight(1)
  6649.         end
  6650.         -- place chest and hopper
  6651.        
  6652.         T:go("D2x2")
  6653.         if not T:place("chest", "down") then
  6654.             T:place("barrel", "down")
  6655.         end
  6656.         T:up(1)
  6657.         T:place("hopper", "down")
  6658.         utils.goBack(1)
  6659.         T:place("slab", "forward")
  6660.         T:go("R2F5U3")-- return to starting point and create entrance
  6661.     end
  6662.     return {}
  6663. end
  6664.  
  6665. local function floodMobFarm(R) -- 63
  6666.     --[[Part 2 / 3 Mob Spawner Farm turtle on floor, pointing towards water source wall, single hole]]
  6667.     local lib ={}
  6668.    
  6669.     function lib.setPosition(addWater)
  6670.         local width = 0
  6671.         while turtle.forward() do end                   -- move forward until hit wall
  6672.         T:go("U1L1")
  6673.         while turtle.forward() do end                   -- move forward until hit left wall
  6674.         if addWater then
  6675.             T:placeWater("down")                        -- place water down
  6676.         end
  6677.         T:turnLeft(2)                                   -- turn round
  6678.         while turtle.forward() do
  6679.             width = width + 1
  6680.         end         -- go forward 7
  6681.         if addWater then                                -- back 1
  6682.             T:placeWater("down")                        -- place water                 
  6683.         end
  6684.         T:go("L2F".. math.floor(width / 2) .."L1")      -- turn round, go forward 3 (centre of wall), turn left
  6685.     end
  6686.  
  6687.     function lib.digFloor()
  6688.         T:go("x2F1 x2")                                 -- first block, move forward
  6689.        
  6690.         T:turnLeft(1)                                   -- left turn, go back into right side, facing left
  6691.         utils.goBack(1)
  6692.         T:go("x2 F1x2 F1x2 R1F1")                       -- go right to left dig 3 blocks, forward on left side
  6693.        
  6694.         T:turnRight(1)                                  -- right turn, go back into left side, facing right
  6695.         utils.goBack(1)
  6696.         T:go("x2 F1x2 F1x2 F1x2 F1x2 L1F1")             -- go left to right dig 5 blocks, forward on right side
  6697.        
  6698.         T:turnLeft(1)                                   -- left turn, go back into right side, facing left
  6699.         utils.goBack(1)
  6700.         T:go("x2 F1x2 F1x2 F1x2 F1x2 F1x2 F1x2 R1F1")   -- go right to left dig 7 blocks, forward on left side
  6701.        
  6702.         T:turnRight(1)                                  -- right turn, go back into left side, facing right
  6703.         utils.goBack(1)
  6704.         T:go("x2 F1x2 F1x2 F1x2 F1x2 F1x2  F1x2  F1x2  F1x2 L1")    -- go left to right dig 5 blocks, face forward on right side
  6705.     end
  6706.    
  6707.     lib.setPosition(false)                  -- move to back of cube and verify position
  6708.     if R.subChoice == 1 then
  6709.         T:forward(3)                        -- forward 4 (centre of chamber)
  6710.     else
  6711.         T:forward(2)                        -- forward 3
  6712.     end
  6713.     T:down(1)
  6714.     lib.digFloor()
  6715.     if R.subChoice == 1 then       
  6716.         T:go("D1F1 L1F8")
  6717.         utils.goBack(4)
  6718.         T:go("L1U1")
  6719.         lib.setPosition(true)               -- place water sources
  6720.         T:go("F8D2")
  6721.         -- go down 2, check floor, up 1, place fence
  6722.         T:go("D2C2U1", false, 0, true)
  6723.         T:place("fence", "down", false)
  6724.         T:go("F1D1C2U1", false, 0, true)
  6725.         T:place("fence", "down", false)
  6726.         T:go("F1U1R2", false, 0, true)
  6727.         T:go("F1R1U1")
  6728.         T:place("sign", "down", false)
  6729.         T:go("U1C0D1")
  6730.         T:place("slab", "up", false)
  6731.         T:go("R2F1R2")
  6732.         T:place("sign", "forward", false)
  6733.         T:go("R1F1R2C1R1F1D1L1") --sitting on soul sand/dirt facing spawner
  6734.         if not T:place("minecraft:soul_sand", "down", false) then
  6735.             T:place("minecraft:dirt", "down", false)
  6736.         end
  6737.     else
  6738.         T:go("D1F1 L1F8")
  6739.         T:go("R1F1 R1F8")
  6740.         utils.goBack(4)
  6741.         T:go("R1U1")
  6742.         lib.setPosition(true)       -- place water sources
  6743.         T:go("F8D2 F1C2C0 F1")      -- exit leaving single hole in wall, facing away from spawner
  6744.     end
  6745.    
  6746.     return {}
  6747. end
  6748.  
  6749. local function createMobBubbleLift(R) -- 64
  6750.     -- Part 3 / 3 Mob Spawner Farm
  6751.     -- R.subChoice = 1 or 2 (left/right)
  6752.     local lib = {}
  6753.    
  6754.     function lib.initialise()
  6755.         local blockType = T:getBlockType("down")
  6756.         if blockType ~= "minecraft:soul_sand" then
  6757.             T:dig("down")
  6758.             if not T:place("minecraft:soul_sand", "down", false) then
  6759.                 return {"Unable to find or place soulsand."}
  6760.             end
  6761.         end
  6762.         -- check facing sign, rotate if not
  6763.         blockType = T:getBlockType("forward")
  6764.         local turns = 0
  6765.         while blockType:find("sign") == nil do
  6766.             T:turnRight(1)
  6767.             turns = turns + 1
  6768.             if turns == 4 then
  6769.                 return {"Unable to find sign."}
  6770.             end
  6771.             blockType = T:getBlockType("forward")
  6772.         end
  6773.         return {""}
  6774.     end
  6775.    
  6776.     function lib.createWaterSource(oTurn)
  6777.         T:go(oTurn.."1")            -- turn to opposite side of dropzone
  6778.         T:go("F1x0D1")
  6779.         for i = 1, 3 do
  6780.             T:go("C2 L1C1 R2C1 L1")
  6781.             if i < 3 then
  6782.                 T:go("F1x0")
  6783.             end
  6784.         end
  6785.         T:go("R2F1")                -- face column move to middle
  6786.         T:placeWater("forward")
  6787.         T:go("R2")                  -- face away
  6788.         T:placeWater("forward")
  6789.         T:go("U1x0 R2F1x0 F1x0"..oTurn.."1")    -- face column, return to soul sand, face spawner
  6790.     end
  6791.    
  6792.     function lib.goToWater(moves)
  6793.         T:down(moves)
  6794.         T:getWater("down")
  6795.         sleep(0.1)
  6796.         T:getWater("down")
  6797.     end
  6798.    
  6799.     function lib.placeCollector(turn, oTurn)
  6800.         local hopperSlot = T:getItemSlot("hopper")
  6801.         local chestSlot = T:getItemSlot("chest")
  6802.         if hopperSlot > 0 and chestSlot > 0 then
  6803.             T:dig("down")
  6804.             T:place("chest", "down")
  6805.             T:go(turn.."1F1"..oTurn.."1")
  6806.             T:dig("down")
  6807.             T:place("chest", "down")
  6808.             T:go(turn.."1")
  6809.             utils.goBack(3)
  6810.             T:go("D1x1")
  6811.             T:place("hopper", "forward")
  6812.             T:go("U1C2F2"..oTurn.."1")
  6813.         end
  6814.     end
  6815.    
  6816.     function lib.up()
  6817.         local moves = 0
  6818.         while turtle.detect() do
  6819.             turtle.up()
  6820.             moves = moves + 1
  6821.         end
  6822.  
  6823.         return moves
  6824.     end
  6825.    
  6826.     function lib.createChamber()
  6827.         local D = {}
  6828.         D.width  = 4
  6829.         D.length = 7
  6830.         D.height = 4
  6831.         D.ceiling = true
  6832.         D.floor = true
  6833.         D.vDirection = "D"
  6834.         D.hDirection = "RL"
  6835.         D.goHome = true
  6836.  
  6837.         utils.createWalledSpace(D)
  6838.     end
  6839.    
  6840.     function lib.mobTransporter()
  6841.         for i = 1, 9 do -- fill in a solid block bar from bubble column for 8 spaces
  6842.             T:go("F1C0C2 R1C1 R1C1 R1C1 R1C1")
  6843.         end
  6844.         T:go("D1C2C1 R1C1 R2C1 L1 C0x0") -- move down column
  6845.         for i = 1, 8 do -- go under solid block bar
  6846.             T:go("F1C2 R1C1 L2C1 R1x0")
  6847.         end
  6848.     end
  6849.    
  6850.     local turn = "R"
  6851.     local oTurn = "L"
  6852.     if R.subChoice == 1 then
  6853.         turn = "L"
  6854.         oTurn = "R"
  6855.     end
  6856.    
  6857.     local data = lib.initialise()           -- check if in the right position
  6858.     if data[1] ~= "" then
  6859.         return data                         -- eg {"Unable to find sign."}
  6860.     end
  6861.     lib.createWaterSource(oTurn)            -- everything in place, build a water source, facing spawner
  6862.     for i = 1, 3 do     -- fill in back and one side, go up
  6863.         T:go(turn.."1C1"..turn.."1C1"..turn.."1x1"..turn.."1U1", false, 0, true)
  6864.     end
  6865.     -- dungeon wall, above mob exit, facing spawner
  6866.     local colHeight = 19
  6867.     for i = 1, colHeight do     -- tunnel up, filling 3 sides
  6868.         T:go(turn.."1C1"..turn.."1C1"..turn.."1x1"..turn.."1C1 U1", false, 0, true)
  6869.     end
  6870.     -- facing spawner 20 blocks up. move either left/right 8 blocks, repairing ceiling and sides
  6871.     T:go("C0"..turn.."2C1"..turn.."1F1 C0C1"..turn.."1C1"..turn.."2C1"..oTurn.."1", false, 0, true) -- fill top of column
  6872.    
  6873.     lib.mobTransporter()
  6874.     T:go("x2x0 F1x2x0 F1x2x0 R2") -- over water source
  6875.     -- now create bubble column
  6876.     T:down(colHeight + 2)
  6877.     local moves = 0
  6878.     repeat
  6879.         lib.goToWater(moves)    -- get water
  6880.         moves = lib.up()
  6881.         T:go("F1")
  6882.         T:placeWater("forward")
  6883.         T:go("U1C2")
  6884.         T:placeWater("forward")
  6885.         utils.goBack(1)
  6886.         T:go("C1")
  6887.         moves = moves + 1
  6888.     until moves >= colHeight + 1
  6889.     lib.goToWater(moves)    -- get water for last time
  6890.     lib.up()                -- finishes above lower part of the transport bar
  6891.     T:go("F2R2C1D1")        -- seal off bubble column
  6892.     utils.goBack(1)
  6893.     T:placeWater("forward") -- place source on top of bubble column
  6894.     utils.goBack(7)         -- over down shaft
  6895.     T:down(1)               -- start shaft, facing bubble column
  6896.     for i = 1, 17 do
  6897.         -- tunnel down, filling all 4 sides
  6898.         T:go("R1C1 R1C1 R1C1 R1C1 D1", false, 0, true)
  6899.     end
  6900.     lib.createChamber()
  6901.     T:go("x0")
  6902.     T:go(oTurn.."1C1".. turn.."1D1C2"..oTurn.."1C1"..turn.."1F1".. oTurn.."1F1"..oTurn.."1") -- facing end wall ready to place slabs
  6903.     for i = 1, 6 do
  6904.         T:place("slab", "down")
  6905.         if i == 6 then
  6906.             T:go(oTurn.."1")
  6907.         end
  6908.         utils.goBack(1)
  6909.         T:go("C1")
  6910.     end
  6911.    
  6912.     T:go("D2F2"..turn.."1F5")
  6913.     T:placeWater("forward")
  6914.     T:go(turn.."1F1"..oTurn.."1") -- facing down mob channel floor
  6915.     for i = 1, 5 do
  6916.         T:go("C1")
  6917.         utils.goBack(1)
  6918.     end
  6919.     T:go("C1"..turn.."1F1"..turn.."1F1")
  6920.     for i = 1, 7 do
  6921.         T:go("C2x0")
  6922.         if i < 7 then
  6923.             T:forward(1)
  6924.         end
  6925.     end
  6926.     T:go("U3R2")
  6927.     for i = 1, 7 do
  6928.         T:go("F1x2")
  6929.     end
  6930.     T:go("D3")
  6931.    
  6932.     lib.placeCollector(turn, oTurn) -- if hopper / chests present
  6933.    
  6934.     return {}
  6935. end
  6936.  
  6937. local function createPlatform(R) -- 56
  6938.     -- T:go(path, useTorch, torchInterval, leaveExisting, preferredBlock)
  6939.     local forward = true
  6940.     for w = 1, R.width do
  6941.         for l = 1, R.length do
  6942.             T:go("x2C2", false, 0, false, R.useBlockType)
  6943.             if R.up then
  6944.                 T:dig("up")
  6945.             end
  6946.             if l < R.length then
  6947.                 T:forward(1)
  6948.             end
  6949.         end
  6950.         if w < R.width then
  6951.             if forward then
  6952.                 if R.up then
  6953.                     T:go("R1F1 x0R1")
  6954.                 else
  6955.                     T:go("R1F1 R1")
  6956.                 end
  6957.             else
  6958.                 if R.up then
  6959.                     T:go("L1F1 x0L1")
  6960.                 else
  6961.                     T:go("L1F1 L1")
  6962.                 end
  6963.             end
  6964.         end
  6965.         forward = not forward
  6966.     end
  6967.     return {}
  6968. end
  6969.  
  6970. local function createPortal(R) -- 42
  6971.     --[[
  6972.     R.length = length of portal NOT width default 4
  6973.     R.height = height of portal default 5
  6974.     R.width = thickness of portal default 1
  6975.     R.data = "bury" to embed bottom into ground
  6976.     R.subChoice 1 = facing portal, 2 = aligned
  6977.     ]]
  6978.     local lib = {}
  6979.    
  6980.     function lib.buildBase()
  6981.         --T:go(path, useTorch, torchInterval, leaveExisting, preferredBlock)
  6982.         T:go("C2", false, 0, false, R.useBlockType)
  6983.         T:forward(1)
  6984.         for i = 1, R.length - 2 do -- R.length = 4: place when i=1,2
  6985.             T:place("minecraft:obsidian", "down", false)
  6986.             T:forward(1)
  6987.         end
  6988.         T:go("C2", false, 0, false, R.useBlockType)
  6989.     end
  6990.    
  6991.     function lib.buildLayer()
  6992.         T:place("minecraft:obsidian", "down", false)
  6993.         for i = 1, R.length - 1 do -- R.length = 4: forward when i=1,2,3
  6994.             T:forward(1)
  6995.         end
  6996.         T:place("minecraft:obsidian", "down", false)
  6997.     end
  6998.    
  6999.     if R.data ~= "bury" then
  7000.         T:up(1)
  7001.     end
  7002.     if R.subChoice == 1 then
  7003.         T:go("F1R1")
  7004.     end
  7005.     local out = true
  7006.     for width = 1, R.width do
  7007.         lib.buildBase()
  7008.         for i = 1, R.height - 2 do
  7009.             T:go("R2U1")
  7010.             out = not out
  7011.             lib.buildLayer()
  7012.         end
  7013.         T:go("R2U1")
  7014.         out = not out
  7015.         lib.buildBase()
  7016.         if out then
  7017.             T:go("R2F"..R.length - 1)
  7018.         end
  7019.         if width < R.width then
  7020.             T:go("R1F1D"..R.height.."R1")
  7021.             out = true
  7022.         else
  7023.             T:go("L1F"..R.width.."D"..R.height - 1 .."R2")
  7024.             if R.data ~= "bury" then
  7025.                 T:down(1)
  7026.             end
  7027.         end
  7028.     end
  7029.    
  7030.     return {}
  7031. end
  7032.  
  7033. local function createPortalPlatform() -- 48
  7034.     --[[ Used in End World to use a trapdoor to push player through portal ]]
  7035.     local lib ={}
  7036.    
  7037.     function lib.findPortal()
  7038.         local found = false
  7039.         local onSide = false
  7040.         for i = 1, 64 do
  7041.             if not turtle.up() then -- hit block above
  7042.                 found = true
  7043.                 break
  7044.             end
  7045.         end
  7046.         if found then
  7047.             -- are we under the centre block, or one of the sides?
  7048.             if turtle.detect() then -- under a side
  7049.                 onSide = true
  7050.             else    -- nothing in front, probably under centre, or facing wrong direction so check
  7051.                 for i = 1, 4 do
  7052.                     turtle.turnRight()
  7053.                     if turtle.detect() then
  7054.                         onSide = true
  7055.                         break
  7056.                     end
  7057.                 end
  7058.             end
  7059.             if onSide then-- move to centre
  7060.                 T:go("D1F1")
  7061.             end
  7062.         end
  7063.         local height = 3 -- allows for 2 bedrock + starting space
  7064.         while turtle.down() do
  7065.             height = height + 1
  7066.         end
  7067.         return found, height
  7068.     end
  7069.    
  7070.     function lib.addFloor(length)
  7071.         for i = 1, length do
  7072.             if i < length then
  7073.                 T:go("C2F1", false, 0, true)
  7074.             else
  7075.                 T:go("C2", false, 0, true)
  7076.             end
  7077.         end
  7078.     end
  7079.    
  7080.     function lib.buildLadder(height)
  7081.         for i = 1, height do
  7082.             --T:go("F1C1 R1C1 L2C1 L1F1L2", false, 0, true)
  7083.             T:go("F1C1 R1C1 L2C1 R1", false, 0, true)
  7084.             utils.goBack(1)
  7085.             if i > 3 then
  7086.                 T:go("C2")
  7087.             end
  7088.             T:place("minecraft:ladder", "forward", true)
  7089.             T:up(1)
  7090.         end
  7091.     end
  7092.    
  7093.     local found, height = lib.findPortal()
  7094.     if found then   -- position under centre of beacon
  7095.         -- build ladder up and create platform
  7096.         T:go("L1F1L1F2L2")
  7097.         T:checkInventoryForItem({"minecraft:ladder"},{height})
  7098.         T:checkInventoryForItem({"stone"},{height * 4 + 18})
  7099.         T:checkInventoryForItem({"trapdoor"},{1})
  7100.         lib.buildLadder(height) -- ends facing ladder, 1 block above
  7101.        
  7102.         T:go("R1")
  7103.         utils.goBack(1)
  7104.         T:go("C2F1 C2F1 C2F1 C2")
  7105.         T:go("R1F1R1")
  7106.         T:go("C2F1 C2F1 C2F1 C2")
  7107.         utils.goBack(2)
  7108.         T:go("R1F1")            -- facing portal entrance
  7109.         T:place("trapdoor", "up", false)
  7110.     else
  7111.         return {"Portal not found. Move me under","the centre if possible.", "wait for purple beacon."}
  7112.     end
  7113.     return {}
  7114. end
  7115.  
  7116. local function createRailway(R) -- 93
  7117.     -- go(path, useTorch, torchInterval, leaveExisting, preferredBlock)
  7118.     --[[
  7119.         Build steps up or down ready for railtrack
  7120.         R.depth = headroom, default 2
  7121.     ]]
  7122.     if R.down then
  7123.         if R.height == 0 then
  7124.             local blockType = ""
  7125.             while blockType == "" do
  7126.                 T:go("F1D1", false, 0, true)
  7127.                 blockType = T:getBlockType("down")
  7128.                 if blockType == "" then
  7129.                     T:go("C2", false, 0, true)
  7130.                 end
  7131.             end
  7132.         else
  7133.             for i = 1, R.height - 1 do
  7134.                 --T:go("U1x0 D1x1 F1x0x1 D1x1 C2", false, 0, false)
  7135.                 T:go("U"..R.depth - 1 .."x0 D"..R.depth - 1 .."x1 F1x0x1 D1x1 C2", false, 0, false)
  7136.             end
  7137.         end
  7138.     elseif R.up then
  7139.         for i = 1, R.height do
  7140.             --T:go("C1U2 x0D1F1", false, 0, false) --put stone in front, up 2 excavate 1, down 1, forward 1
  7141.             T:go("C1U"..R.depth.." x0D"..R.depth.."F1", false, 0, false) --put stone in front, up 2 (or headroom) excavate 1, down 1, forward 1
  7142.         end
  7143.     end
  7144.     return {}
  7145. end
  7146.  
  7147. local function createRectanglePath(R) -- 710, 83 direct commands
  7148.     -- allow user to control length / width of each path
  7149.     -- T:go(path, useTorch, torchInterval, leaveExisting, preferredBlock)
  7150.     --local pp = utils.getPrettyPrint()
  7151.     local lib = {}
  7152.    
  7153.     function lib.forward(R)
  7154.         if R.subChoice == 1 then
  7155.             for i = 1, R.length do
  7156.                 T:go("F1x0C2", false, 0, true)
  7157.             end
  7158.         else
  7159.             for i = 1, R.length do
  7160.                 T:go("F1U1C0D1C2", false, 0, true)
  7161.             end
  7162.         end
  7163.     end
  7164.    
  7165.     function lib.back(R)
  7166.         for i = 1, R.length do
  7167.             turtle.back()
  7168.         end
  7169.     end
  7170.    
  7171.     function lib.left(R)
  7172.         T:turnLeft(R.length)
  7173.     end
  7174.    
  7175.     function lib.right(R)
  7176.         T:turnRight(R.length)
  7177.     end
  7178.    
  7179.     function lib.up(R)
  7180.         if R.subChoice == 1 then
  7181.             T:go("U2R2 x1R2 D1C2", false, 0, true)
  7182.         else
  7183.             T:go("U2R2 x1R2C0 D1C2", false, 0, true)
  7184.         end
  7185.     end
  7186.    
  7187.     function lib.down()
  7188.         T:go("D1C2", false, 0, true)
  7189.     end
  7190.    
  7191.     if R.data == "menu" then
  7192.         -- mimics direct commands using f.lua, r.lua etc with space between commands and number
  7193.         local width = 0
  7194.         local length = 0
  7195.         local choices =
  7196.         {
  7197.             "Forward 1 block",
  7198.             "Forward # blocks",
  7199.             "Back 1 block",
  7200.             "Back # blocks",
  7201.             "Turn Right",
  7202.             "Turn Left",
  7203.             "Up 1 block",
  7204.             "Down 1 block",
  7205.             "Quit"
  7206.         }
  7207.         local choice, modifier
  7208.         pp.itemColours = {colors.lime, colors.lime, colors.green, colors.green, colors.orange, colors.orange, colors.cyan, colors.cyan, colors.gray}
  7209.         while choice ~= 9 do
  7210.             choice, modifier = menu.menu("Choose next step", choices, pp, "Type number + Enter ")
  7211.             if choice == 1 then
  7212.                 R.length = 1
  7213.                 lib.forward(R)
  7214.             elseif choice == 2 then
  7215.                 -- getInteger(prompt, minValue, maxValue, row, fg, bg, default)
  7216.                 R.length = menu.getInteger("Move forward how many blocks?", 1, 250, nil, colors.lime)
  7217.                 lib.forward(R)
  7218.             elseif choice == 3 then
  7219.                 R.length = 1
  7220.                 lib.back(R)
  7221.             elseif choice == 4 then
  7222.                 R.length = menu.getInteger("Move back how many blocks?", 1, 250, nil, colors.green)
  7223.                 lib.back(R)
  7224.             elseif choice == 5 then
  7225.                 R.length = 1
  7226.                 lib.right(R)
  7227.             elseif choice == 6 then
  7228.                 R.length = 1
  7229.                 lib.left(R)
  7230.             elseif choice == 7 then
  7231.                 lib.up(R)
  7232.             elseif choice == 8 then
  7233.                 lib.down()
  7234.             end
  7235.         end
  7236.     else
  7237.         local instructions =
  7238. [[~lightGray~Commands:
  7239.  
  7240. direction + ~blue~number ~yellow~eg ~white~f2 ~yellow~= forward ~blue~2
  7241. ~lightGray~direction without number = ~blue~1
  7242.  
  7243. ~yellow~f = forward  ~orange~b = backward
  7244. ~lime~l = left     ~red~r = right
  7245. ~lightGray~u = up       ~cyan~d = down
  7246.  
  7247. ~red~q = quit
  7248.  
  7249. ]] -- Direct control
  7250.         local cmd = ""
  7251.         while cmd ~= "q" do
  7252.             local line = menu.clear()
  7253.             line = menu.colourText(line, instructions)
  7254.             -- menu.getString(prompt, withTitle, minValue, maxValue, row, fg, bg, default)
  7255.             input = menu.getString("command ", false, 1, 5, line, colors.yellow, colors.black):lower()
  7256.             -- remove spaces
  7257.             input = input:gsub( " ", "")
  7258.             cmd = input:sub(1,1)
  7259.             R.length = 1
  7260.             if #input > 1 then
  7261.                 R.length = tonumber(input:sub(2))
  7262.             end
  7263.             if cmd == "q" then
  7264.                 return{"User has quit application"}
  7265.             end
  7266.             if cmd == "f" then
  7267.                 lib.forward(R)
  7268.             elseif cmd == "b" then
  7269.                 lib.back(R)
  7270.             elseif cmd == "l" then
  7271.                 lib.left(R)
  7272.             elseif cmd == "r" then
  7273.                 lib.right(R)
  7274.             elseif cmd == "u" then
  7275.                 lib.up(R)
  7276.             elseif cmd == "d" then
  7277.                 lib.down()
  7278.             end
  7279.         end
  7280.     end
  7281.     return {}
  7282. end
  7283.  
  7284. local function createRetainingWall(R) -- 34, 82
  7285.     -- facing direction wall will take
  7286.     -- will need to rotate 180 to build
  7287.     -- if R.height > 0 then build to specified depth
  7288.     local lib = {}
  7289.        
  7290.     function lib.checkFloor()
  7291.         local newDepth = 0
  7292.         place = utils.clearVegetation("down") -- in case col in front is deeper
  7293.         while place do -- loop will be entered at least once
  7294.             T:down(1)
  7295.             newDepth = newDepth + 1
  7296.             place = utils.clearVegetation("down")
  7297.         end
  7298.         if newDepth > 0 then
  7299.             for j = 1, newDepth do  -- go up until column base is met
  7300.                 T:go("U1C2")
  7301.             end
  7302.         end
  7303.     end
  7304.    
  7305.     function lib.patchMissingBlock()
  7306.         if turtle.back() then
  7307.             T:go("C1")
  7308.         else
  7309.             T:go("B1C1")
  7310.         end
  7311.     end
  7312.    
  7313.     function lib.placeSingle(height)
  7314.         local y = 0
  7315.         if height > 0 then
  7316.             T:go("D"..height)
  7317.             y = height
  7318.         else
  7319.             local place = utils.clearVegetation("down")
  7320.             while place do -- loop will be entered at least once
  7321.                 place = utils.clearVegetation("down")
  7322.                 if place then
  7323.                     T:down(1)
  7324.                     y = y + 1
  7325.                 end
  7326.             end
  7327.         end
  7328.         -- return to surface, placing below
  7329.         for i = 1, y do
  7330.             T:go("U1C2", false, 0, true)
  7331.         end
  7332.     end
  7333.    
  7334.     function lib.placeDouble(height)
  7335.         --T:go(path, useTorch, torchInterval, leaveExisting, preferredBlock)
  7336.         local y = 0
  7337.         if not turtle.back() then
  7338.             T:go("B1")
  7339.         end
  7340.         if height > 0 then
  7341.             for i = 1, height do
  7342.                 T:go("C1D1", false, 0, true)
  7343.             end
  7344.             T:go("C1", false, 0, true)
  7345.             y = height
  7346.         else
  7347.             local place = utils.clearVegetation("down")
  7348.             -- build back column
  7349.             while place do -- loop will be entered at least once
  7350.                 place = utils.clearVegetation("down")
  7351.                 if place then
  7352.                     T:go("C1D1", false, 0, true)
  7353.                     y = y + 1
  7354.                 end
  7355.             end
  7356.             -- reached bottom. floor in front could drop down
  7357.             T:go("F1") -- move under first column
  7358.             lib.checkFloor()
  7359.             turtle.back() -- back at starting point
  7360.             T:go("C1", false, 0, true)
  7361.         end
  7362.         -- return to surface, placing below
  7363.         for i = 1, y do
  7364.             T:go("U1C2", false, 0, true)
  7365.         end
  7366.     end
  7367.        
  7368.     function lib.placeTriple(height)
  7369.         --T:go(path, useTorch, torchInterval, leaveExisting, preferredBlock)
  7370.         local y = 0
  7371.         if turtle.back() then
  7372.             T:go("C1")
  7373.         else
  7374.             T:go("B1C1")
  7375.         end
  7376.         if height > 0 then
  7377.             for i = 1, height do
  7378.                 T:go("C1D1", false, 0, true)
  7379.             end
  7380.             for i = 1, height do
  7381.                 T:go("C1D1", false, 0, true)
  7382.             end
  7383.             T:go("C1R2C1", false, 0, true) --fill last block, then turn 180 to build opposite side
  7384.             y = height
  7385.         else
  7386.             local place = utils.clearVegetation("down")
  7387.             -- build back column
  7388.             while place do -- loop will be entered at least once
  7389.                 place = utils.clearVegetation("down")
  7390.                 if place then
  7391.                     T:go("C1D1", false, 0, true)
  7392.                     y = y + 1
  7393.                 end
  7394.             end
  7395.             -- reached bottom. floor in front could drop down
  7396.             T:go("F1") -- move under first column
  7397.             lib.checkFloor()
  7398.             T:go("B1C1R2F1", false, 0, true)
  7399.             lib.checkFloor()
  7400.             T:go("B1C1")
  7401.             -- return to surface , placing below and to front
  7402.         end
  7403.         for i = 1, y do
  7404.             T:go("C1U1C2", false, 0, true)
  7405.         end
  7406.         T:go("F1R2C1", false, 0, true)
  7407.         -- facing back again inside edge of col 3
  7408.         return y -- depth of this run
  7409.     end
  7410.    
  7411.     local topInPlace = false -- topInPlace = true already a path across the water eg monument rectangle
  7412.     if R.data == "withPath" then
  7413.         topInPlace = true
  7414.     end
  7415.     local place = false
  7416.     local inWater = false
  7417.     local onWater = false
  7418.     if not topInPlace then
  7419.         if R.length > 1 then
  7420.             inWater, onWater = utils.getWaterStatus() -- returns whether above water, or immersed
  7421.         end
  7422.     end
  7423.    
  7424.     local maxDepth = 5 --initial estimated value
  7425.     -- start at surface, move back 1 block
  7426.     -- each iteration completes 3 columns
  7427.     local numBlocks = T:getSolidBlockCount()
  7428.     print("Solid blocks in inventory: "..numBlocks)
  7429.    
  7430.     if R.length == 1 then -- single column down to water bed
  7431.         lib.placeSingle(R.height)
  7432.     elseif R.length == 2 then--down then up: 2 cols
  7433.         inWater, onWater = utils.startWaterFunction(onWater, inWater, 2, true) -- move into water
  7434.         T:go("R2") -- move to face player
  7435.         lib.placeDouble(R.height)
  7436.         if not inWater then
  7437.             T:go("U1C2", false, 0, true)
  7438.         end
  7439.     else -- R.length 3 or more
  7440.         if topInPlace then
  7441.             T:down(1) -- break through top
  7442.         else
  7443.             inWater, onWater = utils.startWaterFunction(onWater, inWater, 2, true) -- move into water
  7444.         end
  7445.         T:go("R2") -- move to face player
  7446.         -- now at water surface
  7447.         local remain = R.length
  7448.         while remain >= 3 do
  7449.             numBlocks = T:getSolidBlockCount()
  7450.             print("Inventory blocks: "..numBlocks.." depth: "..maxDepth)
  7451.             if numBlocks < maxDepth * 3 then
  7452.                 --ask player for more
  7453.                 T:checkInventoryForItem({"stone"}, {maxDepth * remain}, false)
  7454.             end
  7455.             local y = lib.placeTriple(R.height) -- moves back, places col in front, centre and behind. R.height = 0 for auto depth
  7456.             if y > maxDepth then
  7457.                 maxDepth = y
  7458.             end
  7459.             remain = remain - 3
  7460.             if remain > 1 then
  7461.                 lib.patchMissingBlock()
  7462.             end
  7463.         end
  7464.         if remain == 1 then -- 1 more column
  7465.             lib.patchMissingBlock()
  7466.             lib.placeSingle(R.height)
  7467.         elseif remain == 2 then -- 2 cols
  7468.             lib.placeDouble(R.height)
  7469.         end
  7470.         T:go("U1C2") -- above surface
  7471.     end
  7472.     return {}
  7473. end
  7474.  
  7475. local function createSafeDrop(R) -- 14
  7476.     -- dig down height blocks, checking for blocks on all sides
  7477.     local drop = 0
  7478.     local isBedrock = false
  7479.     T:down(2)
  7480.     drop = 2
  7481.     for i = 1, R.height - 1 do
  7482.         for j = 1, 4 do
  7483.             -- go(path, useTorch, torchInterval, leaveExisting, preferredBlock)
  7484.             T:go("C1R1", false, 0, true)
  7485.         end
  7486.         if T:down(1) then
  7487.              drop = drop + 1
  7488.         else
  7489.             isBedrock = true
  7490.             break
  7491.         end
  7492.         if T:isWaterOrLava("up") ~= "" then
  7493.             T:go("C0x0", false, 0, false) -- delete water/ lava block
  7494.         end
  7495.     end
  7496.     if not isBedrock then -- not sitting on bedrock, so break foot level exit block
  7497.         T:go("U1R2x1")
  7498.     end
  7499.     T:placeWater("down")
  7500.     if isBedrock then
  7501.         T:go("U1x1 U1x1")
  7502.     else
  7503.         T:go("U1x1")
  7504.     end
  7505.     T:up(drop - 2)
  7506.    
  7507.     return {"Safe drop completed "..drop .. " blocks"}
  7508. end
  7509.  
  7510. local function createSlopingWater(R) -- 34, 89
  7511.     --[[
  7512.     creates a sloping water area above existing lake/river/ocean
  7513.     R.width is usually 7 with an existing wall on 8th row
  7514.     R.length  is user choice, limited to bucket/slab quantities
  7515.     Places slabs into existing surface, places row of sources
  7516.     Removes slabs
  7517.     ]]
  7518.     local lib = {}
  7519.    
  7520.     function lib.fillBuckets()
  7521.         local emptyBuckets = utils.getEmptyBucketCount()
  7522.         for i = 1, emptyBuckets do
  7523.             if utils.fillBucket("down") then
  7524.                 print("Bucket filled down")
  7525.                 sleep(0.3)
  7526.             else
  7527.                 print("Unable to fill bucket ".. i .." / "..emptyBuckets)
  7528.             end
  7529.         end
  7530.         return utils.getWaterBucketCount()
  7531.     end
  7532.    
  7533.     local outbound = true
  7534.     local inWater, onWater = utils.getWaterStatus()
  7535.     inWater, onWater = utils.startWaterFunction(onWater, inWater, 2 ,false) -- move above water, max descent 2
  7536.     local waterBuckets = lib.fillBuckets()
  7537.     for w = 1, R.width do
  7538.         for l = 1, R.length do
  7539.             T:place("slab", "down", false)
  7540.             if l < R.length then
  7541.                 T:forward(1)
  7542.             end
  7543.         end
  7544.         if w < R.width then
  7545.             if outbound then
  7546.                 T:go("R1F1R1")
  7547.             else
  7548.                 T:go("L1F1L1")
  7549.             end
  7550.             outbound = not outbound
  7551.         end
  7552.     end
  7553.     if outbound then
  7554.         T:go("L1F"..R.width - 1 .."L1")
  7555.     else
  7556.         T:go("R1F"..R.width - 1 .."R1")
  7557.     end
  7558.     T:placeWater("up")  -- place in corner
  7559.     local move = true
  7560.     while move do
  7561.         move = turtle.forward()
  7562.         move = turtle.forward() -- false if at end of run
  7563.         T:placeWater("up")  -- alternate positions + end of run
  7564.     end
  7565.     T:go("R2D1")
  7566.     T:sortInventory() -- get all buckets into 1 slot
  7567.     for w = 1, R.width do
  7568.         for l = 1, R.length do
  7569.             if l < R.length then
  7570.                 T:forward(1)
  7571.             end
  7572.         end
  7573.         if w < R.width then
  7574.             if outbound then
  7575.                 T:go("R1F1R1")
  7576.             else
  7577.                 T:go("L1F1L1")
  7578.             end
  7579.             outbound = not outbound
  7580.         end
  7581.     end
  7582.     if outbound then
  7583.         T:go("L1F"..R.width - 1 .."L1")
  7584.     else
  7585.         T:go("R1F"..R.width - 1 .."R1")
  7586.     end
  7587.     T:go("U2")
  7588.    
  7589.     return {}
  7590. end
  7591.  
  7592. local function createSquidFarmBase(R) -- 34
  7593.     -- dig down to y=45. assume starting at water level (64)
  7594.     local depth = 0
  7595.     local message =
  7596. [[Place the turtle under the grid
  7597. left corner above current pit
  7598. opening.
  7599. It will descend to the base and
  7600. continue to level y=45.
  7601.  
  7602. Enter to continue...
  7603. ]]
  7604.     read()
  7605.     while turtle.down()do
  7606.         depth = depth + 1
  7607.     end
  7608.     T:down(1)
  7609.     R.height = 21 - depth
  7610.     R.width = 15
  7611.     R.length = 15
  7612.     R.silent = true
  7613.     R.data = "down"
  7614.     clearSolid(R)
  7615.     T:up(depth + 4)
  7616.     return {}
  7617. end
  7618.  
  7619. local function createSandWall(R) -- 81
  7620.     local success = true
  7621.     --move above water
  7622.     local maxMove = 2
  7623.     while turtle.detectDown() and maxMove > 0 do
  7624.         T:forward(1)
  7625.         maxMove = maxMove - 1
  7626.     end
  7627.     if R.length > 0 then
  7628.         for i = 1, R.length - 1 do
  7629.             success = utils.dropSand()
  7630.             T:forward(1, false)
  7631.         end
  7632.         success = utils.dropSand()
  7633.     else
  7634.         while not turtle.detectDown() do -- over water
  7635.             while not turtle.detectDown() do -- nested to allow forward movement
  7636.                 success = utils.dropSand() -- drops sand and checks supplies
  7637.             end
  7638.             if success then
  7639.                 T:forward(1, false)
  7640.             else -- out of sand
  7641.                 break
  7642.             end
  7643.         end
  7644.     end
  7645.     return {}
  7646. end
  7647.  
  7648. local function createSinkingPlatform(R) -- 58
  7649.     local lib = {}
  7650.    
  7651.     function lib.stage1a(R)                         -- build side wall left side
  7652.         for l = 1, R.length do                      --            | |*| |
  7653.             T:go("L1C1 R1C2", false, 0, false)      -- |*|>| | to |*|>| | place left wall
  7654.             if l == 1 then                          -- first iteration
  7655.                 T:go("U1C2 D1 F1C2", false, 0, false)-- |*|>| | to |*|*|>| up/down block to delete source at corner
  7656.             elseif l < R.length then                -- mid run
  7657.                 T:go("F1C2", false, 0, false)       -- |*|>| | to |*|*|>| move forward
  7658.             else                                    -- end of run
  7659.                 T:go("C1U1 C2D1", false, 0, false)  -- |*|>| | to |*|>|*| place end wall
  7660.             end
  7661.         end
  7662.     end
  7663.    
  7664.     function lib.stage1b(R)                         -- same as stage1a on right side
  7665.         for l = 1, R.length do
  7666.             T:go("R1C1 L1C2", false, 0, false)
  7667.             if l == 1 then
  7668.                 T:go("U1C2 D1 F1C2", false, 0, false)
  7669.             elseif l < R.length then
  7670.                 T:go("F1C2", false, 0, false)
  7671.             else
  7672.                 T:go("C1U1 C2D1", false, 0, false)
  7673.             end
  7674.         end
  7675.     end
  7676.    
  7677.     function lib.stage2(forward)
  7678.         if forward then
  7679.             T:go("C1R1 F1L1 C1R2", false, 0, false)
  7680.         else
  7681.             T:go("C1L1 F1R1 C1L2", false, 0, false)
  7682.         end
  7683.     end
  7684.        
  7685.     local forward = true
  7686.     local goingRight = true
  7687.     local blockType = T:getBlockType("down")
  7688.     if blockType:find("water") ~= nil or blockType:find("lava") ~= nil then
  7689.         T:up(1)
  7690.     end
  7691.     for h = 1, R.height do                      -- repeatedly create a platform, move down and repeat
  7692.         T:down(1)                               -- move down into existing platform
  7693.         if goingRight then                      -- first side
  7694.             if forward then                     -- complete left side
  7695.                 T:go("R2C1 L2", false, 0, false) -- | |>| | to |*|<| | to |*|>| |
  7696.                 lib.stage1a(R)                  -- build left wall
  7697.                 T:go("R1F1 L1C1 R2C2", false, 0, false)         -- turn ready for next side
  7698.             else
  7699.                 T:go("L2C1 R2", false, 0, false) -- block 1, 1
  7700.                 lib.stage1b(R)                  -- turn ready for next side
  7701.                 T:go("L1F1 R1C1 L2C2", false, 0, false)
  7702.             end
  7703.         else                                    -- on right side so different approach
  7704.             if forward then
  7705.                 T:go("L2C1 R2", false, 0, false) -- | |<| | to | |>|* | to | |<|*|
  7706.                 lib.stage1b(R)                 
  7707.                 T:go("C1L1 F1R1 C1L2 C2", false, 0, false)      -- turn ready for next side
  7708.             else                                -- complete left side
  7709.                 T:go("R2C1 L2", false, 0, false) -- block 1, 1
  7710.                 lib.stage1a(R)                  -- turn ready for next side
  7711.                 T:go("C1R1 F1L1 C1R2 C2", false, 0, false)
  7712.             end
  7713.         end
  7714.         forward = not forward                   -- continue strips across until at far edge
  7715.         for w = 1, R.width - 2 do
  7716.             for l = 1, R.length do
  7717.                 if l < R.length then
  7718.                     T:go("C2F1", false, 0, false)
  7719.                 else
  7720.                     T:go("C2", false, 0, false)
  7721.                 end
  7722.             end
  7723.             if goingRight then
  7724.                 lib.stage2(forward)
  7725.             else
  7726.                 lib.stage2(not forward)
  7727.             end
  7728.             forward = not forward
  7729.         end                                     -- far side
  7730.         if goingRight then
  7731.             if forward then
  7732.                 lib.stage1b(R)
  7733.             else
  7734.                 lib.stage1a(R)
  7735.             end
  7736.         else
  7737.             if forward then
  7738.                 lib.stage1a(R)
  7739.             else
  7740.                 lib.stage1b(R)
  7741.             end
  7742.         end
  7743.         goingRight = not goingRight
  7744.         T:turnRight(2)
  7745.         forward = not forward
  7746.     end
  7747.     return {}
  7748. end
  7749.  
  7750. local function createStaircase(R) -- 13
  7751.     -- R# L# F# B# U# D# +0 -0 = Right, Left, Forward, Back, Up, Down, up while detect and return, down while not detect
  7752.     -- dig:           x0,x1,x2 (up/fwd/down)
  7753.     -- suck:          s0,s1,s2
  7754.     -- place chest:   H0,H1,H2
  7755.     -- place sapling: S0,S1,S2
  7756.     -- place Torch:   T0,T1,T2
  7757.     -- place Hopper:  P0,P1,P2
  7758.     -- mine floor:    m# = mine # blocks above and below, checking for valuable items below, and filling space with cobble or dirt
  7759.     -- mine ceiling:  M# = mine # blocks, checking for valuable items above, and filling space with cobble or dirt
  7760.     -- mine ceiling:  N# same as M but not mining block below unless valuable
  7761.     -- place:         C,H,r,S,T,P,^ = Cobble / cHest / DIrT / Sapling / Torch / hoPper /stair in direction 0/1/2 (up/fwd/down) eg C2 = place cobble down
  7762.    
  7763.     -- 3| |B| |
  7764.     --   - - -
  7765.     -- 2|A| |C|
  7766.     --   - - -
  7767.     -- 1|^|D| |
  7768.     --   - - -
  7769.     --   1 2 3
  7770.     local lib = {}
  7771.    
  7772.     function lib.checkFluids()
  7773.         local isFluid = false
  7774.         -- check if water or lava present
  7775.         for i = 1, 4 do
  7776.             blockType = T:isWaterOrLava("forward")
  7777.             if blockType:find("lava") ~= nil or blockType:find("water") ~= nil then
  7778.                 isFluid = true
  7779.             end
  7780.         end
  7781.         return isFluid
  7782.     end
  7783.    
  7784.     function lib.createStaircaseSection(onGround, levels, level)
  7785.         -- start 1,1,1, n
  7786.         -- stage A
  7787.         local isFluid = lib.checkFluids()
  7788.         local blockType = ""
  7789.         local data = T:getStock("stairs")
  7790.         if data.total == 0 then
  7791.             T:craft('stairs', 4)
  7792.         end
  7793.         if onGround and isFluid then
  7794.             -- add right side and block entrance
  7795.             T:go("R1C1R1C1R2")
  7796.         end
  7797.         if isFluid then
  7798.             T:go("L1C1 R1F1C2 L1C1 R1x1 R1C1 L1C2B1 C1x1 ^1C2", false, 0, true) --start:1,1,1,n stairs A on level 1, going back to 1,1,1,n
  7799.         else
  7800.             T:go("F1x1 R1C1 L1C2B1 ^1C2", false, 0, true)
  7801.         end
  7802.         if not onGround then
  7803.             -- stage A1
  7804.             T:go("L2C1L2", false, 0, true) -- start 1,1,1,n fix corner on level 1 end: 1,1,1,n
  7805.         end
  7806.         -- stage B
  7807.         T:go("U1L1", false, 0, true) -- end  1,1,1,w layer 2
  7808.         level = level + 1
  7809.         isFluid = lib.checkFluids()
  7810.         if isFluid then
  7811.             T:go("C1", false, 0, true) -- end  1,1,1,w layer 2
  7812.         end
  7813.         if not onGround then
  7814.             if isFluid then
  7815.                 T:go("L1C1R1", false, 0, true) -- end  1,1,1,w layer 2
  7816.             end
  7817.         end
  7818.         -- stage C1
  7819.         if isFluid then
  7820.             T:go("R1C1F1C1x1 L1C1 R2C1 L1B1", false, 0, true)
  7821.         else
  7822.             T:go("R1F1 R1C1 L1B1", false, 0, true)
  7823.         end
  7824.         --level = level + 1
  7825.         --if level == levels then
  7826.             --return false, level
  7827.         --end
  7828.         -- stage C2
  7829.         T:go("U1")
  7830.         level = level + 1
  7831.         --if level == levels then
  7832.             --return true, level
  7833.         --end
  7834.         isFluid = lib.checkFluids()
  7835.         if isFluid then
  7836.             T:go("L1C1L1 C1L2 C1F1L1 C1R2 C1L1 B1C2 D1", false, 0, true) -- end 1,1,2,n
  7837.         else
  7838.             T:go("F1R1 C1L1 B1D1", false, 0, true) -- end 1,1,2,n
  7839.         end
  7840.         level = level - 1 -- went down in branch above
  7841.         --if level == levels then
  7842.             --return false, level
  7843.         --end
  7844.         -- stage D
  7845.         isFluid = lib.checkFluids()
  7846.         if isFluid then
  7847.             T:go("C1F1C1F1C1x1L1 C1R1 C1R1", false, 0, true) -- 3,1,2,e
  7848.         else
  7849.             T:go("F2 C1R1", false, 0, true) -- 3,1,2,e
  7850.         end
  7851.        
  7852.         return false, level
  7853.     end
  7854.  
  7855.     --local height = currentLevel -- eg 64 at top or 5 at bedrock
  7856.     local data = T:getStock("stairs")
  7857.     --{rt.total, rt.mostSlot, rt.leastSlot, rt.mostCount, rt.leastCount}
  7858.     local numStairs = data.total
  7859.     local levels = math.abs(R.depth - R.height) --height of stairs
  7860.     local numStairsNeeded = levels
  7861.     numStairsNeeded = numStairsNeeded - numStairs
  7862.     if numStairsNeeded > 40 then
  7863.         print('crafting '..numStairsNeeded..' : '..numStairs.. ' in stock')
  7864.         if T:craft('stairs', 40) then   -- max 40 so repeat
  7865.             data = T:getStock("stairs")
  7866.             if data.total == 0 then
  7867.                 data = T:getStock("stairs")
  7868.             end
  7869.             numStairs = data.total
  7870.             numStairsNeeded = numStairsNeeded - numStairs
  7871.         else
  7872.             return {"Unable to craft stairs"}
  7873.         end
  7874.     end
  7875.     if numStairsNeeded > 0 then
  7876.         T:craft('stairs', numStairsNeeded)
  7877.     end
  7878.     local level = 0
  7879.     if R.down then -- go down towards bedrock
  7880.         local atBedrock = false
  7881.         for i = 1, levels do
  7882.             level = level - 1
  7883.             if not T:down() then
  7884.                 atBedrock = true
  7885.                 break
  7886.             end
  7887.         end
  7888.         if atBedrock then -- hit bedrock so get to level 5 / -59
  7889.             level = T:findBedrockTop(level)
  7890.             T:go("R1F1R1", false, 0, true)
  7891.         end
  7892.     end
  7893.     local onGround = true
  7894.     level = 0
  7895.     while level < levels do
  7896.         onGround, level = lib.createStaircaseSection(onGround, levels, level)
  7897.     end
  7898.     if not turtle.detectDown() then
  7899.         T:go("C2")
  7900.     end
  7901.    
  7902.     return{"Staircase completed"}
  7903. end
  7904.  
  7905. local function createStripMine(R)
  7906.     --[[
  7907.     R.length should be a multiple of 16
  7908.     mine a corridoor repairing floor and ceiling
  7909.     check sides, remove valuable items
  7910.     plug if lava present
  7911.     Every 16 blocks dig a side passage 1 block deep 2 blocks long
  7912.     ]]
  7913.     local lib = {}
  7914.    
  7915.     function lib.seal(R)   
  7916.         if T:isValuable("forward") then -- valuable block in front. If debris then refuse already dumped
  7917.             T:dig("forward")
  7918.         end
  7919.         local blockType = T:getBlockType("forward")
  7920.         if blockType:find("lava") ~= nil then
  7921.             --T:place("stone", "forward", false) -- place does not allow for specific blocktype
  7922.             T:go("C1", false, 0, false, R.useBlockType)
  7923.             return true
  7924.         end
  7925.        
  7926.         return false
  7927.     end
  7928.    
  7929.     function lib.checkSeal(R)
  7930.         local retValue = false
  7931.         T:turnRight(1)
  7932.         if lib.seal(R) then
  7933.             retValue = true
  7934.         end
  7935.         T:turnLeft(2)
  7936.         if lib.seal(R) then
  7937.             retValue = true
  7938.         end
  7939.         T:turnRight(1)
  7940.         return retValue
  7941.     end
  7942.        
  7943.     function lib.alcove(R)
  7944.         -- right side, starting at ceiling
  7945.         T:go("R1F1 C0", false, 0, false, R.useBlockType)-- stone ceiling, facing alcove wall (upper)
  7946.         lib.seal(R)                                     -- seal alcove wall (upper)
  7947.         T:go("D1C2", false, 0, false, "cobble")         -- cobble floor, facing alcove wall (lower)
  7948.         lib.seal(R)                                     -- seal alcove wall (lower)    
  7949.         T:go("L2 F1")                                   -- down 1, turn round, return to corridoor.
  7950.         -- left side   
  7951.         T:go("F1 C2", false, 0, false, "cobble")        -- cobble floor, facing alcove wall (lower)
  7952.         lib.seal(R)                                     -- seal alcove wall (lower)
  7953.         T:go("U1 C0", false, 0, false, R.useBlockType)  -- stone ceiling, still facing alcove wall
  7954.         lib.seal(R)                                     -- seal alcove wall (upper)                    
  7955.         T:go("L2F1L1")                                  -- return to corridoor at ceiling position
  7956.         lib.placeTorch(R)                              
  7957.     end
  7958.    
  7959.     function lib.placeTorch(R)
  7960.         if R.torchInterval > 0 then                     -- torches onboard
  7961.             if T:getItemSlot("minecraft:torch") > 0 then
  7962.                 T:place("minecraft:torch", "down")
  7963.             end
  7964.         end
  7965.     end
  7966.    
  7967.     local seal = false
  7968.     if T:getItemSlot("minecraft:torch") == 0 then
  7969.         R.torchInterval = 0 -- set to default 16 above
  7970.     end
  7971.     for steps = 1, R.length do
  7972.         -- go(path, useTorch, torchInterval, leaveExisting, preferredBlock)
  7973.         T:go("C2U1C0", false, 0, true, R.useBlockType)      -- check and repair floor / ceiling
  7974.         if steps % 16 == 0 or steps % 16 == 1 then
  7975.             lib.alcove(R) -- enter and exit at ceiling position
  7976.         else
  7977.             seal = lib.checkSeal(R)
  7978.         end
  7979.         T:go("F1D1", false, 0, true)
  7980.         seal = lib.checkSeal(R)
  7981.     end
  7982.     if seal then -- water or lava found while tunnelling
  7983.         T:go("U1C0", false, 0, true, R.useBlockType)
  7984.         lib.checkSeal(R)
  7985.         T:go("C1", false, 0, true, R.useBlockType)
  7986.         T:down(1)
  7987.     end
  7988.     return {}
  7989. end
  7990.  
  7991. local function createTreefarm(R) -- 22
  7992.     local lib = {}
  7993.     --go(path, useTorch, torchInterval, leaveExisting, preferredBlock)
  7994.     function lib.buildWallSection(section, useBlockType)
  7995.         -- build a layer 1 block high below turtle
  7996.         for i = 1, #section do
  7997.             local block = section:sub(i, i)
  7998.             if block == "l" then
  7999.                 T:place("log", "down")
  8000.             elseif block == "m" then
  8001.                 T:place("modem", "down")
  8002.             elseif block == "b" then
  8003.                 T:place("barrel", "down")
  8004.             elseif block == "c" then
  8005.                 T:place("cable", "down")
  8006.             else
  8007.                 T:place(useBlockType, "down")
  8008.             end
  8009.             if i < #section then
  8010.                 T:forward(1)
  8011.             end
  8012.         end
  8013.     end
  8014.    
  8015.     function lib.placeFloor(length, useBlockType)
  8016.         for i = 1, length do
  8017.             while turtle.digUp() do end
  8018.             turtle.digDown()
  8019.             T:place(useBlockType, "down")
  8020.             --T:go("x0C2", false, 0, false, useBlockType)
  8021.             if i < length then
  8022.                 while not turtle.forward() do
  8023.                     turtle.dig()
  8024.                 end
  8025.                 --T:forward(1)
  8026.             end
  8027.         end
  8028.     end
  8029.    
  8030.     function lib.placeWater(length)
  8031.         T:placeWater("down")
  8032.         T:go("F"..length.."R1")
  8033.     end
  8034.    
  8035.     function lib.placeCorners(length, numBlocks, useBlockType)
  8036.         for i = 1, numBlocks do
  8037.             T:go("C2F1", false, 0, false, useBlockType)
  8038.         end
  8039.         -- now at 5th space
  8040.         T:forward(length - (numBlocks * 2))
  8041.         for i = 1, numBlocks do
  8042.             T:go("C2", false, 0, false, useBlockType)
  8043.             if i < numBlocks then
  8044.                 T:forward(1)
  8045.             end
  8046.         end
  8047.     end
  8048.    
  8049.     function lib.turn(outward)
  8050.         if outward then
  8051.             T:go("R1F1R1")
  8052.         else
  8053.             T:go("L1F1L1")
  8054.         end
  8055.         return not outward
  8056.     end
  8057.    
  8058.     function lib.findLegacyStart()
  8059.         T:turnRight(1)
  8060.         local block = T:getBlockType("down")
  8061.         if block:find("polished") ~= nil then
  8062.             return ""   -- in correct position, facing centre of front wall
  8063.         end
  8064.         -- assume on left corner
  8065.         T:forward (1)
  8066.         local couint = 0
  8067.         while (T:getBlockType("down")):find("polished") == nil do
  8068.             T:forward(1)
  8069.             count = count + 1
  8070.             if count > 10 then
  8071.                 return "Unable to locate polished block"
  8072.             end
  8073.         end
  8074.         return ""
  8075.     end
  8076.    
  8077.     function lib.floodFarm(R)
  8078.         local outward = true
  8079.         T:sortInventory(false)
  8080.         for i = 1, R.width - 2 do
  8081.             lib.placeFloor(R.length - 2, R.useBlockType)
  8082.             if i < R.width - 2 then
  8083.                 outward = lib.turn(outward)
  8084.             end
  8085.         end
  8086.         T:go("U1R2") -- over 13 x 13 internal area opposite corner
  8087.         -- now add corners
  8088.         lib.placeCorners(R.length - 2, 4, R.useBlockType)
  8089.         outward = lib.turn(outward)
  8090.         lib.placeCorners(R.length - 2, 3, R.useBlockType)
  8091.         outward = lib.turn(outward)
  8092.         lib.placeCorners(R.length - 2, 2, R.useBlockType)
  8093.         outward = lib.turn(outward)
  8094.         lib.placeCorners(R.length - 2, 1, R.useBlockType)
  8095.         if outward then
  8096.             T:go("R1F"..R.width - 9 .."R1")
  8097.         else
  8098.             T:go("L1F"..R.width - 9 .."L1")
  8099.         end
  8100.         outward = not outward
  8101.         lib.placeCorners(R.length - 2, 1, R.useBlockType)
  8102.         outward = lib.turn(outward)
  8103.         lib.placeCorners(R.length - 2, 2, R.useBlockType)
  8104.         outward = lib.turn(outward)
  8105.         lib.placeCorners(R.length - 2, 3, R.useBlockType)
  8106.         outward = lib.turn(outward)
  8107.         lib.placeCorners(R.length - 2, 4, R.useBlockType) -- should be back in starting corner facing front
  8108.        
  8109.         T:go("U1R1")
  8110.         lib.placeWater(R.length - 3) -- place water down then move forward, turn right
  8111.         lib.placeWater(R.width - 3)
  8112.         lib.placeWater(R.length - 3)
  8113.         lib.placeWater(R.width - 3)
  8114.         T:go("F".. math.floor((R.length - 2) / 2).."R1F".. math.floor((R.width - 2) / 2)) -- should be in centre
  8115.        
  8116.         --T:go("D6x2 F1x2 F1x2 F1x2 F1x2 F1x2 F1x2 F1x2 F1x2 F1x2 R1F7 R1U1")
  8117.         T:go("D6F7 R1F7 R1U1") -- end facing back at left corner, 1 block above floor/1 below ceiling
  8118.     end
  8119.    
  8120.     function lib.clearBasement()
  8121.         T:sortInventory(false)
  8122.         T:dropItem("seeds", "forward")
  8123.         T:dropItem("flint", "forward")
  8124.         clearRectangle({width = 15, length = 15, up = true, down = true})
  8125.         -- area has been cleared. starts facing back, 1 block above ground, 1 block below ceiling
  8126.         T:go("R1U1")
  8127.         for i = 1, 14 do    -- place cable into gutter beneath front of farm
  8128.             T:place("cable", "up")
  8129.             if i < 14 then
  8130.                 T:forward(1)
  8131.             end
  8132.         end
  8133.         -- ends right side facing right, just below ceiling
  8134.     end
  8135.    
  8136.     local blockType
  8137.     local blockModifier
  8138.    
  8139.     -- R.data = "new", "left", "right" or "back" to extend tree farm
  8140.     -- R.up = true if clear area
  8141.     -- R.networkFarm to create storage area
  8142.     if R.up then
  8143.         clearArea(R)
  8144.     end
  8145.     -- R.data = "new", "left", "right", "back", "convertStorage", "convert"
  8146.     if R.data == "new" then -- new treeFarm, Start at current position
  8147.         -- build 4 wall sections in 2 passes
  8148.         T:down(1)
  8149.         lib.buildWallSection("---------------", R.useBlockType)     -- left wall (15 blocks)
  8150.         T:go("R1F1")
  8151.         for i = 1, 2 do
  8152.             lib.buildWallSection("--------------", R.useBlockType)  -- back/ right wall (14 blocks)
  8153.             T:go("R1F1")
  8154.         end
  8155.         lib.buildWallSection("------c------", R.useBlockType)       -- front wall (14 blocks) c = network cable
  8156.         T:go("U1F1R1")
  8157.         lib.buildWallSection("---------------", R.useBlockType)     -- left wall top (15 blocks)
  8158.         T:go("R1F1")
  8159.         for i = 1, 2 do
  8160.             lib.buildWallSection("--------------", R.useBlockType)  --back/right wall (14 blocks)
  8161.             T:go("R1F1")
  8162.         end
  8163.         --lib.buildWallSection("-----lmb-----", R.useBlockType) --front wall (14 blocks) log/modem/barrel
  8164.         lib.buildWallSection("-----lmb", R.useBlockType)
  8165.         utils.goBack(1)
  8166.         network.attachModem()
  8167.         T:forward(2)
  8168.         lib.buildWallSection("-----", R.useBlockType)  
  8169.         T:go("R1F1 D2") -- over 13 x 13 internal area
  8170.     elseif R.data == "left" or R.data == "right" or R.data == "back" then
  8171.         -- build 3 wall sections in 2 passes
  8172.         if R.data == "left" then                                        -- should be on left corner of existing
  8173.             T:go("L1F1 D1")                                             -- move left 1 blocks, down 1: <-
  8174.             lib.buildWallSection("------c-------", R.useBlockType)      -- front wall (14 blocks) c = network cable <-
  8175.             T:go("R1F1")
  8176.             lib.buildWallSection("--------------", R.useBlockType)      -- left wall (14 blocks) ^
  8177.             T:go("R1F1")
  8178.             lib.buildWallSection("-------------", R.useBlockType)       -- back wall (13 blocks) ->
  8179.             T:go("U1R2")                                                -- turn round ready to add next layer <-
  8180.             lib.buildWallSection("--------------", R.useBlockType)      -- back wall top (14 blocks) <-
  8181.             T:go("L1F1")
  8182.             lib.buildWallSection("--------------", R.useBlockType)      -- left wall top (14 blocks) v
  8183.             T:go("L1F1")
  8184.             lib.buildWallSection("-----bm", R.useBlockType)             -- front wall (7 blocks) barrel/modem ->
  8185.             network.attachModem()
  8186.             T:forward(1)
  8187.             lib.buildWallSection("l-----", R.useBlockType)              -- front wall (5 blocks) log ->
  8188.             T:go("R2F12 R1F1 D2")                                       -- over 13 x 13 internal area lower left side
  8189.         elseif R.data == "right" then                                   -- should be on right corner of existing
  8190.             T:go("R1F1 D1")                                             -- move right, forward, down 1
  8191.             lib.buildWallSection("------c-------", R.useBlockType)      -- front wall (14 blocks) c = network cable
  8192.             T:go("L1F1")
  8193.             lib.buildWallSection("--------------", R.useBlockType)      -- right wall (14 blocks)
  8194.             T:go("L1F1")
  8195.             lib.buildWallSection("-------------", R.useBlockType)       -- back wall (13 blocks)
  8196.             T:go("U1R2")                                                -- turn round ready to add next layer
  8197.             lib.buildWallSection("--------------", R.useBlockType)      -- back wall top (14 blocks)
  8198.             T:go("R1F1")
  8199.             lib.buildWallSection("--------------", R.useBlockType)      -- left wall top (14 blocks)
  8200.             T:go("R1F1")
  8201.             lib.buildWallSection("-----lmb", R.useBlockType)
  8202.             utils.goBack(1)
  8203.             network.attachModem()
  8204.             T:forward(2)
  8205.             lib.buildWallSection("-----", R.useBlockType)  
  8206.             T:go("R1F1 D2")                                             -- over 13 x 13 internal area
  8207.         elseif R.data == "back" then                                    -- should be on left front corner of existing
  8208.             T:go("R2F1 D4R2 F1")                                        -- move forward 14 blocks, down 1
  8209.             for i = 1, 15 do
  8210.                 T:place("cable", "up")
  8211.                 T:forward(1)
  8212.             end
  8213.             T:up(1)
  8214.             if T:getBlockType("up") == R.useBlockType then              -- already a farm on left side
  8215.                 T:go("U2C2 U1C2 F13R1 F1D1", false, 0, false, R.useBlockType)
  8216.                 lib.buildWallSection("--------------", R.useBlockType)  -- back wall (14 blocks)
  8217.                 T:go("R1F1")
  8218.             else
  8219.                 T:up(2)
  8220.                 lib.buildWallSection("--------------", R.useBlockType)  -- left wall (14 blocks)
  8221.                 T:go("R1F1")
  8222.                 lib.buildWallSection("--------------", R.useBlockType)  -- back wall (14 blocks)
  8223.                 T:go("R1F1")
  8224.             end
  8225.  
  8226.             lib.buildWallSection("-------------", R.useBlockType)       --right wall (13 blocks) no special blocks
  8227.             T:go("U1R2")    -- turn round ready to add next layer
  8228.             for i = 1, 2 do
  8229.                 lib.buildWallSection("--------------", R.useBlockType)  --right wall top (14 blocks) no special blocks
  8230.                 T:go("L1F1")
  8231.             end
  8232.             lib.buildWallSection("-------------", R.useBlockType)       -- left wall top (13 blocks) no special blocks
  8233.             T:go("F1L1 F7x2")
  8234.             T:go("D1x2")
  8235.             T:place("cable", "down")
  8236.             T:up(1)
  8237.             T:place("modem", "down")
  8238.             T:go("F1R2x2")
  8239.             T:place("log", "down")
  8240.             T:go("F2x2")
  8241.             T:place("barrel", "down")
  8242.             utils.goBack(1)
  8243.             network.attachModem()
  8244.             T:go("F6R1 F1D2")
  8245.         end
  8246.     else -- convertStorage or convert
  8247.         -- legacy farm had polished block on positions 4 / (10) from left corner
  8248.          local message = lib.findLegacyStart()
  8249.          if message ~= "" then
  8250.             return {message}
  8251.          end
  8252.          -- now on top of polished block, 4 from left corner, facing Centre
  8253.          T:forward(2)
  8254.          T:place("barrel", "down")
  8255.          T:go("F1D1")
  8256.          T:place("cable", "down")
  8257.          T:up(1)
  8258.          T:place("modem", "down")
  8259.          network.attachModem()
  8260.          T:forward(1)
  8261.          T:place("log", "down")
  8262.          T:go("R1F1 R1F1 D5R1 F1L1 F7R1")-- left corner, facing back ready to clear basement
  8263.     end
  8264.     if (R.data):find("convert") == nil then
  8265.         lib.floodFarm(R)
  8266.     end
  8267.     lib.clearBasement() -- area has been cleared. ends right side facing right, just below ceiling
  8268.  
  8269.     if R.data == "back" then
  8270.         T:go("R2F6 R1D1")
  8271.     else
  8272.         T:go("R2F6 L1F1 U4D5 R2F1") -- make exit for player. end mid floor/ceiling height, facing back, in centre
  8273.     end
  8274.     if R.data == "new" or R.data == "convertStorage" then   -- put storage in this farm
  8275.         for i = 1, 3 do
  8276.             T:place("cable", "up")
  8277.             T:forward(1)
  8278.         end
  8279.         T:place("cable", "up")
  8280.         --T:down(1)
  8281.         utils.createStorage()   -- creates on ground, 1 below current
  8282.         T:go("F1R2")
  8283.         T:place("cable", "forward")
  8284.         T:go("R2")
  8285.         for i = 1, 2 do
  8286.             T:place("cable", "up")
  8287.             T:forward(1)
  8288.         end
  8289.     else
  8290.         for i = 1, 6 do
  8291.             T:place("cable", "up")
  8292.             T:forward(1)
  8293.         end
  8294.     end
  8295.     T:place("modem", "up")
  8296.     T:go("F1R2U2")
  8297.     T:place("barrel", "down")
  8298.     T:drop("down", "dirt", 64)-- override 'slot' with item string
  8299.     T:up(1)
  8300.     T:place("hopper", "down")
  8301.     T:go("F1D1")
  8302.     network.attachModem()
  8303.     T:go("U1C2 U2F6 R2")
  8304.     --network.attachModem()
  8305.  
  8306.     return {"Tree farm ready for planting"}
  8307. end
  8308.  
  8309. local function createWallOrFence(R)
  8310.     local lib = {}
  8311.    
  8312.     function lib.placeBarrel(R)
  8313.         if R.data == "barrel" then
  8314.             utils.goBack(1)
  8315.             T:go("L1F1")
  8316.             T:place("barrel", "down", false)
  8317.             utils.goBack(1)
  8318.             T:go("R1")
  8319.             T:forward(1)
  8320.         end
  8321.     end
  8322.    
  8323.     function lib.placeTorch(R)
  8324.         T:up(1)
  8325.         local up = 1
  8326.         local blockType = T:getBlockType("forward")
  8327.         while blockType:find(R.useBlockType) ~= nil and blockType ~= "" do -- prevent continuous upward travel
  8328.             T:up(1)
  8329.             up = up + 1
  8330.             blockType = T:getBlockType("forward")
  8331.         end
  8332.         T:place("torch", "forward", true)
  8333.         T:down(up)
  8334.     end
  8335.    
  8336.     function lib.buildWall(R, length)
  8337.         -- T:place(blockType, direction, leaveExisting, signText)
  8338.         local blockType = ""
  8339.         local blocks = 0
  8340.         lib.placeBarrel(R)
  8341.         while blocks < length do
  8342.             if turtle.back() then
  8343.                 T:place(R.useBlockType, "forward", true)
  8344.                 if R.torchInterval > 0 then
  8345.                     if blocks == 0 or blocks % R.torchInterval == 0 then
  8346.                         lib.placeTorch(R)
  8347.                     end
  8348.                 end
  8349.                 blocks = blocks + 1 -- still facing start position
  8350.                 while turtle.down() do
  8351.                     T:place(R.useBlockType, "up", true)
  8352.                 end
  8353.             else -- obstruction
  8354.                 T:turnRight(2) -- facing away from start
  8355.                 blockType = T:getBlockType("forward")
  8356.                 if blockType:find("torch") ~= nil then
  8357.                     T:go("F1R2")
  8358.                     T:place(R.useBlockType, "forward")
  8359.                     blocks = blocks + 1 -- facing start
  8360.                 elseif blockType:find("log") ~= nil then
  8361.                     T:harvestTree()
  8362.                     T:turnRight(2)
  8363.                     T:place(R.useBlockType, "forward")
  8364.                     blocks = blocks + 1 -- facing start
  8365.                 elseif T:isVegetation(blockType) then
  8366.                     T:go("F1R2")
  8367.                     T:place(R.useBlockType, "forward")
  8368.                     blocks = blocks + 1 -- facing start
  8369.                 else -- cant go forward, go up instead
  8370.                     while turtle.detect() and blockType:find("torch") == nil and blockType:find("log") == nil and not T:isVegetation(blockType) do -- block ahead, but not torch or tree
  8371.                         while turtle.detectUp() do -- will only run if block above
  8372.                             utils.goBack(1)
  8373.                             blocks = blocks - 1
  8374.                         end
  8375.                         turtle.up()
  8376.                         T:place(R.useBlockType, "down", true)
  8377.                         blockType = T:getBlockType("forward")
  8378.                     end
  8379.                     T:turnRight(2) -- facing start
  8380.                 end
  8381.             end
  8382.         end
  8383.     end
  8384.    
  8385.     T:turnRight(2) --facing start position
  8386.     if R.width == 0 then -- single fence
  8387.         lib.buildWall(R, R.length)
  8388.     else   
  8389.         lib.buildWall(R, R.length - 1)
  8390.         T:go("R1") -- facing start so left turn = turnRight
  8391.         lib.buildWall(R, R.width - 1)
  8392.         T:go("R1")
  8393.         lib.buildWall(R, R.length - 1)
  8394.         T:go("R1")
  8395.         lib.buildWall(R, R.width - 2)
  8396.         T:go("U1")
  8397.         T:place(R.useBlockType, "down", true)
  8398.     end
  8399.    
  8400.     return {"Wall or fence completed"}
  8401. end
  8402.  
  8403. local function createWaterCanal(R) -- 53
  8404.     --[[
  8405.     designed for 4 turtles, but can be done with 2 as well
  8406.     R.data = 1 (2 turtles) or 2 (4 turtles)
  8407.     R.height = 0 (in water/ on canal floor) or 1 (at ground level)
  8408.     R.side = "R" or "L"
  8409.     R.subChoice =
  8410.         1 = left side, ground level (on towpath)
  8411.         2 = left side above canal water (new canal only)
  8412.         3 = right side above canal water (new canal only)
  8413.         4 = right side, ground level (on towpath)
  8414.         5 = left side in canal water (if present)
  8415.         6 = right side in canal water (if present)
  8416.    
  8417.     if R.height = 0 then already at correct height on canal floor
  8418.         check block below, block to left and block above, move forward tunnelling
  8419.         if entering water then move up, onto canal wall and continue pathway
  8420.         if tunnelling then flood canal
  8421.     else R.height = 1 then above water and on path across
  8422.         move forward, checking for water below
  8423.         if water finishes, move into canal, drop down and continue tunnelling
  8424.     ]]
  8425.     local lib = {}
  8426.    
  8427.     function lib.newCanalSide(oTurn)
  8428.         T:go("C1 U1x1 U1x1 D2"..oTurn.."1", false, 0, false)
  8429.     end
  8430.    
  8431.     function lib.newCanalBase(R)
  8432.         local oTurn = "R"                               -- assume R.side = "L"
  8433.         if R.side == "R" then
  8434.             oTurn = "L"
  8435.         end
  8436.         if not turtle.detectDown() then                 -- air / water below: fill floor
  8437.             T:go(oTurn.."1D1")                          -- *|>| | | to *|V| | | turn towards canal centre, go down ready to repair neighbouring canal base
  8438.             if T:getBlockType("forward"):find("turtle") ~= nil then -- turtle in similar position
  8439.                 if R.side == "L" then                   -- only left side repairs, so wait 1 second
  8440.                     sleep(1)
  8441.                     T:go("C1")                          -- repair neighbouring canal base
  8442.                 end
  8443.             else
  8444.                 T:go("C1")                              -- repair neighbouring canal base
  8445.             end
  8446.             T:go(R.side.."1")                           -- *|V| | | to *|>| | | turn to face start
  8447.             T:up(1)                                     -- *|>| | | return to canal floor
  8448.         end
  8449.         T:go("C2", false, 0, false)
  8450.     end
  8451.    
  8452.     function lib.newCanal(R)
  8453.         --  no source in front as this function called                                              -- *|<| | |  facing end wall ready for new canal canal base level
  8454.         local oTurn = "R"                                   -- assume R.side = "L"
  8455.         if R.side == "R" then
  8456.             oTurn = "L"
  8457.         end
  8458.         lib.newCanalSide(oTurn)                             -- *|<| | | wall fixed. to -- *|^| | |
  8459.         lib.newCanalSide(oTurn)                             -- *|^| | | wall fixed. to -- *|>| | |
  8460.         lib.newCanalBase(R)                                 -- *|>| | | base fixed
  8461.         T:go("F1"..R.side.."1")                             -- *|>| | | to *| |>| | to *| |^| |
  8462.         lib.newCanalSide(oTurn)                             -- *| |^| | to *| |>| |
  8463.         lib.newCanalBase(R)                                 -- *| |>| | base fixed
  8464.         T:go("F1"..R.side.."1")                             -- *| |>| | to *| | |>| to *| | |^|
  8465.         lib.newCanalSide(oTurn)                             -- *| | |^| to *| | |>|
  8466.         lib.newCanalBase(R)                                 -- *| | |>| base fixed
  8467.         T:go(R.side.."2")                                   -- *| | |>| to *| | |<|facing existing / back wall
  8468.         T:forward(1)                                        -- *| |<| |
  8469.         T:placeWater("forward")                             -- *|W|<| | placed against start wall  
  8470.         T:go("L2")                                          -- *|W|>| |
  8471.         T:placeWater("forward")                             -- *|W|>|W| placed direction of travel
  8472.         T:go("R2")                                          -- *|W|<|W| rotate
  8473.         if deletesWater then
  8474.             T:up(1)
  8475.         else
  8476.             sleep(0.2)
  8477.         end
  8478.         utils.getWater()                                    -- *|W|<|W|
  8479.         T:go(oTurn.."2F1")                                  -- *|W|<|W|  to *|W|W|>|  ready to go  
  8480.     end
  8481.    
  8482.     function lib.towpathOnly(R)
  8483.         --[[single turtle on towpath only using 4 turtles. Starts at ground level]]
  8484.         for i = 1, R.length do
  8485.             if turtle.detectDown() then                     -- eg dirt, stone
  8486.                 T:dig("down")                               -- remove block below
  8487.             end
  8488.             local placeOnly = true
  8489.             if R.torchInterval > 0 then                     -- place torches
  8490.                 if i == 1 or i % R.torchInterval == 0 then  -- ready to place torch
  8491.                     T:go("C2U1")                            -- place solid block below, up 1
  8492.                     T:place("torch", "down")                -- place torch, move forward
  8493.                     T:go("F1D1")                            -- move forward, down
  8494.                     placeOnly = false      
  8495.                 end
  8496.             end
  8497.             if placeOnly then
  8498.                 T:dig("up")                                 -- clear blocks above
  8499.                 if not T:place("stone", "down") then        -- place stone below
  8500.                     break
  8501.                 end
  8502.                 if i < R.length then
  8503.                     T:forward(1)                            -- move forward
  8504.                 end                        
  8505.             end
  8506.         end
  8507.     end
  8508.    
  8509.     function lib.initialiseCanal(R)
  8510.         --[[ move turtle to correct position. return moves]]
  8511.         local moves = 1
  8512.         local oTurn = "R"
  8513.         if R.side == "R" then
  8514.             oTurn = "L"
  8515.         end
  8516.         local newCanal, isWater, isSource = false, false, false
  8517.         if R.data == 1 then -- use 2 turtles only
  8518.             if R.subChoice == 1 or R.subChoice == 4 then    -- left / right side on towpath- move into canal space
  8519.                 if deletesWater then
  8520.                     T:go(oTurn.."1F1")                      -- turn to face canal centre, go forward over water
  8521.                     isWater, isSource = T:isWater("down")
  8522.                     if isSource then                        -- canal already exists
  8523.                         T:go(oTurn.."1")                    -- stay above water, face existing canal
  8524.                     else                                    -- NOT above source
  8525.                         T:go("D1"..oTurn.."1")              -- go to canal floor, face existing
  8526.                         newCanal = true                     -- flag create 6 water source blocks
  8527.                     end
  8528.                 else                                        -- NOT deletesWater
  8529.                     T:go(oTurn.."1F1D1"..oTurn.."1")        -- move into canal, face back along any existing canal
  8530.                     isWater, isSource = T:isWater("forward")
  8531.                     if isSource then
  8532.                         T:go(R.side.."2")                   -- face forward for new canal
  8533.                     else
  8534.                         newCanal = true
  8535.                     end
  8536.                 end
  8537.             elseif R.subChoice == 2 or R.subChoice == 3 then-- left / right side above canal finishing pos if deletesWater
  8538.                 if deletesWater then
  8539.                     isWater, isSource = T:isWater("down")
  8540.                     if isSource then
  8541.                         T:go(R.side.."2")                   -- face towards existing canal
  8542.                     else
  8543.                         T:go("D1"..R.side.."2")             -- go down and face existing canal                      -- no water ahead
  8544.                         newCanal = true                     -- already in newCanal starting position
  8545.                     end
  8546.                 else                                        -- NOT deletesWater
  8547.                     T:go("D1"..R.side.."2")                 -- facing existing on canal floor
  8548.                     isWater, isSource = T:isWater("forward")
  8549.                     if isSource then                        -- water ahead
  8550.                         T:go(R.side.."2")                   -- face forward for new canal
  8551.                     else                                   
  8552.                         newCanal = true
  8553.                     end
  8554.                 end
  8555.             elseif R.subChoice == 5 or R.subChoice == 6 then -- left / right  side in canal base. Assume NOT deletesWater
  8556.                 T:go(R.side.."2")                           -- check if water behind   
  8557.                 isWater, isSource = T:isWater("forward")
  8558.                 if isSource then
  8559.                     T:go(oTurn.."2")    
  8560.                 else
  8561.                     newCanal = true
  8562.                 end
  8563.             end
  8564.         else    -- use 4 turtles
  8565.             if R.subChoice == 1 or R.subChoice == 4 then    -- left / right side on towpath stays in place for 4 part canal
  8566.            
  8567.             elseif R.subChoice == 2 or R.subChoice == 3 then-- left / right side above canal finishing pos if deletesWater
  8568.            
  8569.             elseif R.subChoice == 5 or R.subChoice == 6 then -- left / right  side in canal base. Assume NOT deletesWater
  8570.            
  8571.             end
  8572.         end
  8573.         if newCanal then                                -- no water ahead, facing start wall of new canal *|<| | |
  8574.             lib.newCanal(R)                             -- start new canal, finish facing new canal 6 block water sources
  8575.             moves = 2
  8576.         end
  8577.        
  8578.         return moves                                    -- facing forward ready for new canal *|>| | |
  8579.     end
  8580.    
  8581.     function lib.side(R, maxLength)
  8582.         -- Already in position facing new canal, 2 water buckets
  8583.         local torch = R.length - maxLength                  -- start torch count at 1-2 depending on lib.initialise()
  8584.         local sourceCount = 0                               -- allow for 1 iteration of placing source blocks when changing from solid to water
  8585.         local numBlocks = 0                                 -- distance travelled
  8586.         local _, isSource = nil, false                      -- initialise variables
  8587.         local oTurn = "R"                                   -- assume R.side == "L" -> oTurn = "R"
  8588.         if R.side == "R" then                               -- R.side = "R"
  8589.             oTurn = "L"                                     -- oTurn = "L"
  8590.         end
  8591.        
  8592.         while numBlocks < maxLength do                      -- loop from here. Facing forwards to extend canal
  8593.             torch = torch + 1                               -- increase torch spacing interval
  8594.             numBlocks = numBlocks + 1                       -- inrease block count
  8595.             if deletesWater then                            -- up to and including mc 1.12.2
  8596.                 T:forward(1)                                -- move forward to extend canal  | |>|
  8597.                 _, isSource = T:isWater("down")             -- check if source water below
  8598.                 if isSource then
  8599.                     sourceCount = sourceCount + 1           -- count source blocks
  8600.                 else                                       
  8601.                     T:down(1)                               -- down to canal floor
  8602.                     sourceCount = 0                         -- reset
  8603.                     T:go("C2", false, 0, false)             -- place block below
  8604.                     T:up(1)                                 -- up to water level
  8605.                 end                                         -- still facing away from canal
  8606.                 T:go(R.side.."1F1C2", false, 0, false)      -- go above towpath, place stone down, facing out
  8607.                 if turtle.detectUp() then                   -- ? block above
  8608.                     T:go(R.side.."2U1F1D1"..R.side.."2")    -- rotate to face canal, up 1 forward over canal, back down to water level, face towpath
  8609.                 else
  8610.                     utils.goBack(1)                         -- face towpath
  8611.                 end
  8612.                 -- facing towpath
  8613.                 if R.torchInterval > 0 then                 -- Are torches required?
  8614.                     if torch == R.torchInterval  or numBlocks == 0 then     -- check if ready to place
  8615.                         torch = 0                           -- reset interval counter
  8616.                         T:go("U1F1")
  8617.                         T:place("torch", "down")        -- place torch
  8618.                         utils.goBack(1)
  8619.                         T:down(1)
  8620.                     end
  8621.                 end                                         -- facing towpath, above water
  8622.                 -- facing towpath
  8623.                 if isSource and scoreCount > 0 then
  8624.                     T:go(oTurn.."1x0")                      -- face new canal direction
  8625.                 else                                        -- not source below, or first time found, ensures continous canal
  8626.                     T:go(R.side.."1x0")                     -- face back to existing canal
  8627.                     T:placeWater("down")                    -- place water
  8628.                     T:forward(1)                            -- forward 1
  8629.                     T:placeWater("down")                    -- place water
  8630.                     utils.getWater()                            -- collects water from below
  8631.                     utils.goBack(1)                         -- back 1
  8632.                     T:go(oTurn.."1x0")                      -- face new canal
  8633.                 end
  8634.             else
  8635.                 T:forward(1)                                -- move forward to extend canal
  8636.                 _, isSource = T:isWater("forward")          -- check if source water ahead
  8637.                 if isSource then                            -- ? source ahead
  8638.                     sourceCount = sourceCount + 1
  8639.                 else
  8640.                     sourceCount = 0
  8641.                     if not turtle.detectDown() then         -- air / water below, but no source in front, so fill it
  8642.                         T:go(oTurn.."1D1")                  -- ready to repair neighbouring canal base
  8643.                         if T:getBlockType("forward"):find("turtle") ~= nil then -- turtle in similar position
  8644.                             if R.side == "L" then           -- only left side repairs, so wait 1 second
  8645.                                 sleep(1)
  8646.                                 T:go("C1")                  -- repair neighbouring canal base
  8647.                             end
  8648.                             T:go(R.side.."1")               -- face forward
  8649.                         else
  8650.                             T:go("C1"..R.side.."1")         -- repair neighbouring canal base if air
  8651.                         end
  8652.                         T:up(1)
  8653.                     end
  8654.                     T:go("C2", false, 0, false)             -- place block below if not already source
  8655.                 end
  8656.                 T:go(R.side.."1C1", false, 0, false)        -- face canal wall, replace with stone
  8657.                
  8658.                 local placeTorch = false
  8659.                 if R.torchInterval > 0 then                 -- Are torches required?
  8660.                     if torch == R.torchInterval  or numBlocks == 0 then     -- check if ready to place
  8661.                         torch = 0                           -- reset interval counter
  8662.                         placeTorch = true
  8663.                     end
  8664.                 end
  8665.                 if (isSource and placeTorch) or not isSource then
  8666.                     T:go("U1x1 U1x1")                       -- up 2 and excavate blocks above tow path
  8667.                     if placeTorch then                      -- torch required
  8668.                         T:forward(1)                        -- move over towpath
  8669.                         T:place("torch", "down")        -- place torch
  8670.                         utils.goBack(1)                     -- move back
  8671.                     end
  8672.                     T:down(2)                               -- return to canal base
  8673.                 end
  8674.                 if isSource and sourceCount > 1 then        -- source ahead found at least once
  8675.                     T:go(oTurn.."1")                        -- face newcanal as no water refill required
  8676.                 else                                        -- not source in front, or first time found, ensures continous canal
  8677.                     T:go(oTurn.."1C1")                      -- *| | | |>| face along new canal and block entrance
  8678.                     utils.goBack(1)                         -- *| | |>| | back 1
  8679.                     T:placeWater("forward")                 -- *| | |>|W| place water
  8680.                     T:go(oTurn.."2")                        -- *| | |<|W| face existing canal
  8681.                     _, isSource = T:isWater("forward")      -- *| |?|<|W| check if source water ahead
  8682.                     if not isSource then
  8683.                         if not T:placeWater("forward") then -- place water again *| |W|<|W|
  8684.                             while not T:getWater("forward") do -- wait for other turtle
  8685.                                 print("Out of water buckets")
  8686.                                 sleep(1)
  8687.                             end
  8688.                             sleep(0.2)
  8689.                             T:getWater("forward")
  8690.                         end
  8691.                     end
  8692.                     utils.getWater()                            -- collects water *| |W|<|W|
  8693.                     T:go(R.side.."2F1")                         -- face along new canal *| |W|>|W| to *| |W|W|>|
  8694.                 end
  8695.             end
  8696.         end
  8697.     end
  8698.        
  8699.     function lib.waterOnly(R, maxLength)
  8700.         -- Already in position facing new canal, 2 water buckets
  8701.         local torch = R.length - maxLength                  -- start torch count at 1-2 depending on lib.initialise()
  8702.         local sourceCount = 0                               -- allow for 1 iteration of placing source blocks when changing from solid to water
  8703.         local numBlocks = 0                                 -- distance travelled
  8704.         local _, isSource = nil, false                      -- initialise variables
  8705.         local oTurn = "R"                                   -- assume R.side == "L" -> oTurn = "R"
  8706.         if R.side == "R" then                               -- R.side = "R"
  8707.             oTurn = "L"                                     -- oTurn = "L"
  8708.         end
  8709.        
  8710.         while numBlocks < maxLength do                      -- loop from here. Facing forwards to extend canal
  8711.             torch = torch + 1                               -- increase torch spacing interval
  8712.             numBlocks = numBlocks + 1                       -- inrease block count
  8713.             if deletesWater then                            -- up to and including mc 1.12.2
  8714.                 T:forward(1)                                -- move forward to extend canal  | |>|
  8715.                 _, isSource = T:isWater("down")             -- check if source water below
  8716.                 if isSource then
  8717.                     sourceCount = sourceCount + 1           -- count source blocks
  8718.                 else                                       
  8719.                     T:down(1)                               -- down to canal floor
  8720.                     sourceCount = 0                         -- reset
  8721.                     T:go("C2", false, 0, false)             -- place block below
  8722.                     T:up(1)                                 -- up to water level
  8723.                 end                                         -- still facing away from canal
  8724.                 if turtle.detectUp() then                   -- ? block above
  8725.                     T:go("U1x0")                            -- break block above
  8726.                 end
  8727.                 if not isSource or scoreCount == 0 then     -- not source below, or first time found, ensures continous canal
  8728.                     T:go(R.side.."2x0")                     -- face back to existing canal
  8729.                     T:placeWater("down")                    -- place water
  8730.                     T:forward(1)                            -- forward 1
  8731.                     T:placeWater("down")                    -- place water
  8732.                     utils.getWater()                        -- collects water from below
  8733.                     utils.goBack(1)                         -- back 1
  8734.                     T:go(R.side.."2")                       -- face new canal
  8735.                 end
  8736.             else
  8737.                 T:forward(1)                                -- move forward to extend canal
  8738.                 _, isSource = T:isWater("forward")          -- check if source water ahead
  8739.                 if isSource then                            -- ? source ahead
  8740.                     sourceCount = sourceCount + 1
  8741.                 else
  8742.                     sourceCount = 0
  8743.                     if not turtle.detectDown() then         -- air / water below, but no source in front, so fill it
  8744.                         T:go(oTurn.."1D1")                  -- ready to repair neighbouring canal base
  8745.                         if T:getBlockType("forward"):find("turtle") ~= nil then -- turtle in similar position
  8746.                             if R.side == "L" then           -- only left side repairs, so wait 1 second
  8747.                                 sleep(1)
  8748.                                 T:go("C1", false, 0, true)  -- repair neighbouring canal base
  8749.                             end
  8750.                             T:go(R.side.."1")               -- face forward
  8751.                         else
  8752.                             T:go("C1"..R.side.."1",false, 0, true)  -- repair neighbouring canal base if air
  8753.                         end
  8754.                         T:up(1)
  8755.                     end
  8756.                     T:go("C2", false, 0, false)             -- place block below if not already source
  8757.                 end
  8758.                 if not isSource then
  8759.                     T:go("U1x0D1")                          -- up 1 and excavate blocks above canal. return to canal base
  8760.                 end
  8761.                 T:go(R.side.."1C1"..oTurn.."1", false, 0, true) -- face canal wall, replace with stone if empty, face forward                                      
  8762.                 if not isSource and sourceCount == 0 then   -- not source in front, or first time found, ensures continous canal                   
  8763.                     T:go("C1", false, 0, true)              -- *| | | |>| face along new canal and block entrance
  8764.                     utils.goBack(1)                         -- *| | |>| | back 1
  8765.                     T:placeWater("forward")                 -- *| | |>|W| place water
  8766.                     T:go(oTurn.."2")                        -- *| | |<|W| face existing canal
  8767.                     _, isSource = T:isWater("forward")      -- *| |?|<|W| check if source water ahead
  8768.                     if not isSource then
  8769.                         if not T:placeWater("forward") then -- place water again *| |W|<|W|
  8770.                             while not T:getWater("forward") do -- wait for other turtle
  8771.                                 print("Out of water buckets")
  8772.                                 sleep(1)
  8773.                             end
  8774.                             sleep(0.2)
  8775.                             T:placeWater("forward")
  8776.                         end
  8777.                     end
  8778.                     utils.getWater()                            -- collects 2 water *| |W|<|W|
  8779.                     T:go(R.side.."2F1")                         -- face along new canal *| |W|>|W| to *| |W|W|>|
  8780.                 end
  8781.             end
  8782.         end
  8783.     end
  8784.        
  8785.     local maxLength = 512
  8786.     if R.length ~= 0 then
  8787.         maxLength = R.length
  8788.     end
  8789.     menu.clear()
  8790.     menu.colourWrite("Building canal "..R.side.." side", colors.yellow, nil, nil, false, true)
  8791.     local moves = lib.initialiseCanal(R) -- move to correct position and/or start new canal
  8792.     if R.data == 1 then -- 2 turtles
  8793.         lib.side(R, maxLength - moves) 
  8794.     else -- 4 turtles
  8795.         if R.subChoice == 1 or R.subChoice == 4 then -- towpath
  8796.             lib.towpathOnly(R)
  8797.             --lib.waterCanalTowpath(R, maxLength - moves)
  8798.         else -- main canal only
  8799.             lib.waterOnly(R, maxLength - moves)
  8800.         end
  8801.     end
  8802.    
  8803.     return {}
  8804. end
  8805.  
  8806. local function createMobGrinder(R)
  8807.     -- go down 5 and create a cavity 9 wide, 5 long, 8 high
  8808.     -- assume facing out from spawner cube at base of exit (6 blocks above ground
  8809.     -- R.subChoice = 1 or 2 (sticky or normal pistons)
  8810.    
  8811.     local lib = {}
  8812.    
  8813.     function lib.getInventory(R)
  8814.         T:clear()
  8815.         menu.colourPrint("All blocks to be ejected. Ready? Enter", colors.red)
  8816.         read()
  8817.         T:emptyInventory("up")
  8818.         T:checkInventoryForItem({"stone"}, {20}, true, "Match existing walls?")
  8819.         if R.subChoice == 1 then -- sticky pistons
  8820.             T:checkInventoryForItem({"computercraft:computer_normal"}, {1})
  8821.             T:checkInventoryForItem({"minecraft:sticky_piston"}, {2})
  8822.         else
  8823.             T:checkInventoryForItem({"computercraft:computer_normal"}, {2})
  8824.             T:checkInventoryForItem({"minecraft:piston"}, {4})
  8825.             T:checkInventoryForItem({"computercraft:cable"}, {6})
  8826.             T:checkInventoryForItem({"computercraft:wired_modem"}, {2})
  8827.         end
  8828.         T:checkInventoryForItem({"polished"}, {23}, true, "Any polished block")
  8829.         T:checkInventoryForItem({"wall"}, {2}, true, "Any wall block ? match")
  8830.         T:checkInventoryForItem({"slab"}, {7}, true, "Any stone slab. ? polished / match")
  8831.         T:checkInventoryForItem({"minecraft:glass"}, {2})
  8832.         T:checkInventoryForItem({"minecraft:stone_pressure_plate"}, {1})
  8833.         T:checkInventoryForItem({"minecraft:dispenser"}, {1})
  8834.         T:checkInventoryForItem({"hopper"}, {1})
  8835.         T:checkInventoryForItem({"chest"}, {2})
  8836.         T:checkInventoryForItem({"minecraft:redstone"}, {9})
  8837.         T:checkInventoryForItem({"lava"}, {1})
  8838.         R.useBlockType = T:getPolishedItem("")
  8839.         if R.useBlockType == "" then
  8840.             R.useBlockType = T:getMostItem("", true)
  8841.         end
  8842.     end
  8843.    
  8844.     function lib.placeRedstone()
  8845.         T:go("D2U2")                                -- in left corner facing right
  8846.         T:place("slab", "down")                 -- place slab layer 1
  8847.         T:up(2)
  8848.         T:place("slab", "down")                 -- place slab layer 2
  8849.         T:up(1)
  8850.         T:place("redstone", "down")
  8851.         T:go("F1D4 C2L1C1 L1U1", false, 0, false)   -- facing ground level slab
  8852.         T:place("redstone", "forward")          -- redstone on ground level slab
  8853.         T:up(1)
  8854.         T:place("slab", "down")                 -- slab on layer 1
  8855.         T:up(2)
  8856.         T:place("slab", "down")                 -- slab on layer 3
  8857.         T:up(1)
  8858.         T:place("redstone", "down")             -- redstone on layer 3 slab
  8859.         utils.goBack(1)
  8860.         T:down(3)                                   -- facing layer 1 slab
  8861.         T:place("redstone", "forward")          -- redstone on layer 1 slab
  8862.         T:go("D2C2", false, 0, false)               -- under end of chest position
  8863.         T:place("redstone", "forward")          -- redstone on level 0 floor
  8864.         T:go("R1C1R1", false, 0, false)             -- face right, repair back wall
  8865.         T:go("F1C2 L1C1 R1", false, 0, false)       -- under chest start
  8866.         T:go("F1C2 L1C1 R1", false, 0, false)       -- under hopper
  8867.         T:go("F1C2 L1C1 R1", false, 0, false)       -- under right side wall block
  8868.         T:go("U1L2")                                -- ready for redstone
  8869.         for i = 1, 4 do
  8870.             T:place("redstone", "down")         -- redstone under chest etc
  8871.             if i < 4 then
  8872.                 T:forward(1)
  8873.             end
  8874.         end
  8875.         -- finishes facing slab at ground level, redstone below
  8876.     end
  8877.    
  8878.     function lib.placeStorage()
  8879.         -- starts facing slab at ground level, redstone below
  8880.         T:go("L1F1L2")                              -- up and out ready for chest placement
  8881.         T:place("chest", "forward")             -- end chest
  8882.         T:go("R1F1 L1F1 R1U2")                      -- at above start chest position facing right
  8883.         T:place("slab", "down")                 -- slab above chest
  8884.         T:turnLeft(1)
  8885.         utils.goBack(1)                            
  8886.         T:down(2)                                   -- in front of start chest position
  8887.         T:place("chest", "forward")             -- start chest
  8888.         T:go("R1F2L2")                              -- at pressure plate position facing left
  8889.         T:place("wall", "forward")              -- place wall
  8890.         T:up(1)
  8891.         T:place("pressure", "down")             -- place pressure plate
  8892.         T:go("R1F1 L1D1")                           -- at second wall position
  8893.         T:place("hopper", "forward")                -- place hopper into chest
  8894.         utils.goBack(1)
  8895.         T:place("wall", "forward")              -- place second wall
  8896.         -- finishes on level 1 (ground) facing right side wall block
  8897.     end
  8898.    
  8899.     function lib.placePistons(R)
  8900.         -- starts on level 1 (ground) facing right side wall block
  8901.         T:go("U1F2 R1")                                 -- go over wall block, facing back wall layer 2
  8902.         T:place(R.useBlockType, "forward", false)   -- replace back wall with polished block layer 2
  8903.         T:up(1)                                         -- layer 3
  8904.         T:place(R.useBlockType, "forward", false)   -- replace back wall with polished block layer 3
  8905.         T:turnRight(1)                                  -- face right side
  8906.         if R.subChoice == 1 then                        -- use sticky pistons x 2
  8907.             T:place(R.useBlockType, "forward", false)-- polished block above second wall layer 3
  8908.             T:go("R2F1")                                -- ready to place lower piston (works ok as slab already in place behind it)
  8909.         else
  8910.             T:go("F2R2")                                -- move forward 2 and face left
  8911.             T:place("piston", "forward")                -- lower replacer piston placed
  8912.             T:go("U1F2D1F1")                            -- go up and over piston forward 1 ready for other piston
  8913.         end
  8914.         T:place("piston", "forward")                    -- lower piston placed
  8915.         utils.goBack(1)
  8916.         T:go("U1R1")                                    -- layer 4
  8917.         T:place(R.useBlockType, "forward", false)   -- polished block back wall layer 4
  8918.         T:go("L1F2R2")                                  -- head left ready for dispenser
  8919.         T:place("dispenser", "forward")             -- dispenser placed
  8920.         T:dropItem("lava", "forward")                   -- drop lava bucket into dispenser
  8921.         T:go("U1F2 D1")                                 -- up and over dispenser, facing right side
  8922.         T:place("slab", "down")                     -- lower piston slab placed
  8923.         if R.subChoice == 1 then
  8924.             T:place(R.useBlockType, "forward", false)-- polished block above second wall layer 4
  8925.             T:turnLeft(1)                               -- facing back wall
  8926.         else
  8927.             T:go("F1L1")                                -- at second computer position, facing back wall
  8928.             utils.goBack(1)                             -- ready to place computer
  8929.             T:place("computercraft:computer_normal", "forward", false)  -- place computer
  8930.             T:go("L1F1R1F1")                            -- facing back wall in mob drop, level 4
  8931.         end
  8932.         T:place(R.useBlockType, "forward", false)   -- polished block back wall layer 4
  8933.         T:turnLeft(1)
  8934.         T:go("U1F3R2")                                  -- level 5 ready for upper piston
  8935.         T:place("piston", "forward")                    -- upper piston placed
  8936.         T:go("U1F3D1R2")                                -- up and over piston, facing left, level 5
  8937.         T:turnRight(1)                                  -- at mob drop, facing wall, level 5
  8938.         T:place(R.useBlockType, "forward", false)   -- polished block back wall layer 5
  8939.         T:turnRight(1)                                  -- facing right side
  8940.         if R.subChoice == 1 then
  8941.             T:place(R.useBlockType, "forward", false)-- right side polished block layer 5, facing right side, in mob drop                           -- layer 5 facing left side, in mob drop       
  8942.         else
  8943.             T:go("F1C1")                                -- move forward, place temporary block
  8944.             utils.goBack(1)
  8945.             T:place("piston", "forward")                -- upper replacer piston placed            
  8946.         end
  8947.         T:turnRight(2) 
  8948.         T:place("slab", "forward")                  -- upper piston slab placed
  8949.         T:turnLeft(1)                                   -- facing away from back wall
  8950.     end
  8951.    
  8952.     function lib.placeComputer(R)
  8953.         -- starts facing away from back wall, layer 5, in mob drop         
  8954.         T:go("F1R1 F2R1 D1")                            -- move to left computerposition, facing back wall, layer 4
  8955.         T:place("computercraft:computer_normal", "forward", false)      -- place computer in gap, layer 4
  8956.     end
  8957.    
  8958.     function lib.placeColumnSection(direction)
  8959.         T:place(R.useBlockType, "forward", false)
  8960.         T:go(direction.."1")
  8961.         T:place(R.useBlockType, "forward", false)
  8962.         T:go(direction.."1")
  8963.         T:place(R.useBlockType, "forward", false)   -- facing right
  8964.     end
  8965.    
  8966.     function lib.placeColumns(R)
  8967.         -- starts facing left computer, layer 4
  8968.         T:go("R1F2 L1")                                 -- facing mob drop level 4
  8969.         T:place("glass", "down", false)
  8970.         T:up(1)                                         -- facing mob drop level 5
  8971.         T:place("glass", "down", false)             -- on top of glass facing back wall at dungeon base level 5
  8972.         T:up(1)
  8973.         T:place(R.useBlockType, "down", false)      -- level 6 dungeon exit
  8974.         T:go("F1L1")
  8975.         lib.placeColumnSection("L")                     -- facing right
  8976.         T:up(1)
  8977.         lib.placeColumnSection("R")                     -- facing left
  8978.         T:up(1)
  8979.         lib.placeColumnSection("L")                     -- facing right
  8980.         T:up(1)
  8981.         lib.placeColumnSection("R")                     -- facing left
  8982.         T:up(1)
  8983.         T:place(R.useBlockType, "forward", false)
  8984.         T:turnLeft(2)
  8985.         T:place(R.useBlockType, "forward", false)   -- facing right
  8986.         T:turnLeft(1)                                   -- at top of column
  8987.         T:go("x1D1 x1D1 x1U2")                          -- open exit for mobs
  8988.         utils.goBack(2)
  8989.         T:place(R.useBlockType, "forward", false)
  8990.         T:down(9)
  8991.     end
  8992.    
  8993.     function lib.placeNetwork()
  8994.         T:go("R1F2L1 F3U3 F1L1F1 L1x1")                             -- go behind second computer and remove block
  8995.         T:place("computercraft:wired_modem", "forward", false)  -- place modem
  8996.         utils.goBack(1)
  8997.         T:place("computercraft:cable", "forward", false)            -- place network cable
  8998.         T:place("computercraft:cable", "forward", false)            -- place network cable (no need to move)
  8999.         T:go("R1F1 x2L1")                                           -- move forward, face back wall
  9000.         T:place("computercraft:cable", "forward", false)
  9001.         T:go("R1F1 x2L1")
  9002.         T:place("computercraft:cable", "forward", false)
  9003.         T:go("R1F1 x2L1F1 x1x2")                                    -- come behind left computer and expose back
  9004.         T:place("computercraft:wired_modem", "forward", false)  -- place modem
  9005.         utils.goBack(1)
  9006.         T:place("computercraft:cable", "forward", false)
  9007.         T:place("computercraft:cable", "forward", false)
  9008.         T:go("L1D3 F4R1 F3U4 D4")                                   -- removes temporary block
  9009.     end
  9010.    
  9011.     local tempSubChoice = R.subChoice   -- store for later use
  9012.     R.width = 4                         -- settings for createFloorCeiling()
  9013.     R.length = 9
  9014.     R.height = 0
  9015.     R.up = true
  9016.     T:turnRight(2)
  9017.     local isWater, isSource, isIce, level, isAir = T:isWater("forward")
  9018.     if not isWater then
  9019.         T:turnRight(2)
  9020.         return{"Not in correct position. Must be water behind"}
  9021.     end
  9022.     --T:go("U1x1 U1L1 F4R2")                    -- go up to starting point
  9023.     T:go("U2L1 F4R2")                   -- go up to starting point
  9024.     local data = createFloorCeiling(R)[1]   -- place ceiling, returns to starting position. if water or lava present is returned as "water or lava found"
  9025.     if data == "water or lava found" then
  9026.         R.height = 10
  9027.         R.silent = true
  9028.         T:up(1)
  9029.         createSinkingPlatform(R)
  9030.         T:go("F4L1 U5x1 U1x1 U3R1 F4C0 R1F3C0 R1F8C0 R1F3C0 R1D9")          -- re-open exit hole, return to floor level
  9031.     else -- no water or lava found so empty out area
  9032.         R.height = 10
  9033.         R.subChoice = 2
  9034.         clearSolid(R) -- 76
  9035.         T:down(R.height - 1)
  9036.         R.width = 4                         -- settings for createFloorCeiling()
  9037.         R.length = 9
  9038.         R.height = 0
  9039.         R.down = true
  9040.         R.up = false
  9041.         R.subChoice = 1                     -- replace existing floor
  9042.         createFloorCeiling(R)
  9043.     end
  9044.    
  9045.     R.subChoice = tempSubChoice
  9046.     lib.getInventory(R)
  9047.     lib.placeRedstone()
  9048.     lib.placeStorage()
  9049.     lib.placePistons(R)
  9050.     lib.placeComputer(R)        -- ends facing computer
  9051.     lib.placeColumns(R)
  9052.     if R.subChoice == 2 then
  9053.         lib.placeNetwork()
  9054.     end
  9055.    
  9056.     return {}
  9057. end
  9058.  
  9059. local function createTrialCover(R)
  9060.     local lib = {}
  9061.    
  9062.     function lib.isSpawner()
  9063.         local blockType = T:getBlockType("down")
  9064.         if blockType:find("spawner") ~= nil then
  9065.             return true, "top"
  9066.         end
  9067.         blockType = T:getBlockType("up")
  9068.         if blockType:find("spawner") ~= nil then
  9069.             return true, "bottom"
  9070.         end
  9071.         blockType = T:getBlockType("forward")
  9072.         if blockType:find("spawner") ~= nil then
  9073.             return true, "forward"
  9074.         end
  9075.         return false, ""
  9076.     end
  9077.    
  9078.     function lib.findSpawner()
  9079.         local moves  = 0
  9080.         local quit = false
  9081.         -- assume turtle placed facing trial spawner
  9082.         print("Checking if next to spawner")
  9083.         local found, position = lib.isSpawner() -- true/false, top/bottom/nil
  9084.         if not found then -- move forward towards spawner
  9085.             print("Not close to spawner")
  9086.             while turtle.forward() and not quit do
  9087.                 moves = moves + 1
  9088.                 if moves > 32 then
  9089.                     quit = true
  9090.                 end
  9091.             end
  9092.             found, position = lib.isSpawner() -- true/false, top/bottom/nil
  9093.             if not found then   -- could be behind a wall
  9094.                 print("Assuming spawner behind a wall")
  9095.                 T:forward(1)
  9096.                 moves = moves + 1
  9097.                 while turtle.forward() and not quit do
  9098.                     moves = moves + 1
  9099.                     if moves > 32 then
  9100.                         quit = true
  9101.                     end
  9102.                 end
  9103.                 found, position = lib.isSpawner() -- true/false, top/bottom/nil
  9104.                 if not found then
  9105.                     T:go("R2F"..moves + 2 .."R2")
  9106.                 end
  9107.             end
  9108.         end
  9109.        
  9110.         return found, position
  9111.     end
  9112.    
  9113.     function lib.attack()
  9114.         local totalHitsF = 0
  9115.         local totalHitsU = 0
  9116.         local totalHitsD = 0
  9117.         while true do
  9118.             local hitF = false
  9119.             local hitU = false
  9120.             local hitD = false
  9121.             if turtle.attackUp() then
  9122.                 hitU = true
  9123.                 totalHitsU = totalHitsU + 1
  9124.             end
  9125.             if turtle.attackDown() then
  9126.                 hitD = true
  9127.                 totalHitsD = totalHitsD + 1
  9128.             end
  9129.             if turtle.attack() then
  9130.                 hitF = true
  9131.                 totalHitsF = totalHitsF + 1
  9132.             end
  9133.             if hitF or hitU or hitD then
  9134.                 print("hits forward: "..totalHitsF..", up: "..totalHitsU..", down: "..totalHitsD)
  9135.             end
  9136.         end
  9137.     end
  9138.    
  9139.     local found, position = lib.findSpawner() -- move forwards until meet Spawner, go through wall if present
  9140.     if not found then --outside dungeon
  9141.         return {"Trial spawner not found"}
  9142.     end
  9143.     if position == "top" then
  9144.         T:go("B2D1")
  9145.     elseif position == "forward" then
  9146.         T:go("B1")
  9147.     else
  9148.         T:go("B2U1")
  9149.     end
  9150.     T:go("R1F2R2")  --on lower right corner
  9151.     -- starts with wall across spawner
  9152.     buildStructure(R)
  9153.     T:go("U3F1 R1F1 L1U1 F1")
  9154.     R.height = 0
  9155.     R.width = 3
  9156.     R.length = 3
  9157.     R.down = true
  9158.     createFloorCeiling(R)
  9159.     T:go("F1R1 B1D1 B1D2 F1")
  9160.     lib.attack()
  9161.     return {}
  9162. end
  9163.  
  9164. local function sandFillArea(R) -- 81
  9165.     --clearRectangle with sand drop
  9166.     -- could be 1 wide x xx length (trench) up and return
  9167.     -- could be 2+ x 2+
  9168.     -- even no of runs return after last run
  9169.     -- odd no of runs forward, back, forward, reverse and return
  9170.     local success
  9171.     local directReturn = true
  9172.     if R.width % 2 == 1 then
  9173.         directReturn = false
  9174.     end
  9175.     if R.width == 1 then -- trench ahead, so fill then return
  9176.         for i = 1, R.length - 1 do
  9177.             success = utils.dropSand()
  9178.             T:forward(1, false)
  9179.         end
  9180.         success = utils.dropSand()
  9181.         T:go("R2F"..(R.length - 1).."R2", false, 0, false)
  9182.     else --2 or more columns
  9183.         if directReturn then -- R.width = 2,4,6,8 etc
  9184.             for i = 1, R.width, 2 do -- i = 1,3,5,7 etc
  9185.                 -- move along R.length, dropping sand
  9186.                 for j = 1, R.length - 1 do
  9187.                     success = utils.dropSand()
  9188.                     T:forward(1, false)
  9189.                 end
  9190.                 success = utils.dropSand()
  9191.                 T:go("R1F1R1") --turn right and return on next column
  9192.                 for j = 1, R.length - 1 do
  9193.                     success = utils.dropSand()
  9194.                     T:forward(1, false)
  9195.                 end
  9196.                 success = utils.dropSand()
  9197.                 if i < R.width - 2 then -- eg R.width = 8, i compares with 6: 1, 3, 5, 7
  9198.                     T:go("L1F1L1")
  9199.                 end
  9200.             end
  9201.             T:go("R1F"..R.width - 1 .."R1") --return home
  9202.         else
  9203.             for i = 1, R.width, 2 do -- i = 1,3,5,7 etc
  9204.                 -- move along R.length, dropping sand
  9205.                 for j = 1, R.length - 1 do
  9206.                     success = utils.dropSand()
  9207.                     T:forward(1, false)
  9208.                 end
  9209.                 success = utils.dropSand()
  9210.                 T:go("R1F1R1") --turn right and return on next column
  9211.                 for j = 1, R.length - 1 do
  9212.                     success = utils.dropSand()
  9213.                     T:forward(1, false)
  9214.                 end
  9215.                 success = utils.dropSand()
  9216.                 T:go("L1F1L1")
  9217.             end
  9218.             -- one more run then return
  9219.             for j = 1, R.length - 1 do
  9220.                 success = utils.dropSand()
  9221.                 T:forward(1, false)
  9222.             end
  9223.             success = utils.dropSand()
  9224.             T:go("R2F"..R.length.."R1F"..R.width - 1 .."R1")
  9225.         end
  9226.     end
  9227.     return {}
  9228. end
  9229.  
  9230. local function deactivateDragonTower() -- 45
  9231.     -- go up centre of tower to bedrock
  9232.     local height = 0
  9233.     --numBlocksMoved, errorMsg = clsTurtle.doMoves(self, numBlocksRequested, direction)
  9234.     local numBlocks, message = T:doMoves(1, "up")
  9235.     while message == nil do
  9236.         numBlocks, message = T:doMoves(1, "up")
  9237.         height = height + 1
  9238.     end
  9239.     -- go round bedrock and destroy crystal
  9240.     T:go("F1R2U2x1U1x1")
  9241.     -- return to start
  9242.     T:down(height + 5)
  9243.     return {}
  9244. end
  9245.  
  9246. local function undermineDragonTowers() -- 44
  9247.     --[[
  9248.             -13, -40....12, -40                     NNW (4)     NNE (5)
  9249.            
  9250.         -34, -25............33, -25             NWW (2)             NEE (9)
  9251.        
  9252.     -42, -1....................42, 0        W (1)                       E (8)
  9253.    
  9254.          -34, 24............33,24               SWW (3)             SEE (10)
  9255.          
  9256.               -13,39....12, 39                      SSW (7)     SSE (6)
  9257.    
  9258.     North towers centres 25 blocks apart, 40 blocks north of axis
  9259.     Mid-North towers 67 blocks apart, 25 blocks north of axis
  9260.     W-E centres 84 blocks apart, on 0 axis
  9261.     Mid-south towers 67 blocks apart, 24 blocks south of axis
  9262.     South towers centres 25 blocks apart, 39 blocks south of axis
  9263.     ]]
  9264.    
  9265.     local lib = {}
  9266.     function lib.findNextTower(maxDistance, withMarker)
  9267.         local distance = 0
  9268.         local blockTypeF = T:getBlockType("forward")
  9269.         local blockTypeD = T:getBlockType("down")
  9270.         for i = 1, maxDistance do
  9271.             if blockTypeF ~= "minecraft:obsidian" and blockTypeD ~= "minecraft:obsidian" then -- not in a tower
  9272.                 if withMarker and blockTypeD ~= "minecraft:obsidian" then -- used to mark 0 coordinate
  9273.                     T:place("cobble", "down", false) -- place cobblestone or cobbled deepslate to mark zero coordinate
  9274.                 end
  9275.             else    -- obsidian found, could still be in an earlier tower
  9276.                 if i > 10 then
  9277.                     break
  9278.                 end
  9279.             end
  9280.             T:go("F1x0")
  9281.             distance = distance + 1
  9282.             blockTypeF = T:getBlockType("forward")
  9283.             blockTypeD = T:getBlockType("down")
  9284.         end
  9285.         if distance == maxDistance then -- obsidian not found ? wrong place/ direction
  9286.             print("Obsidian not found")
  9287.             error()
  9288.         end
  9289.         -- will now be at side of a tower
  9290.         lib.findCentre() -- move into tower to find the other side
  9291.         return distance
  9292.     end
  9293.    
  9294.     function lib.findCentre()
  9295.         local width = 0
  9296.         -- while obsidian in front or below (previously entered tower) measure width and return to centre
  9297.         local blockTypeF = T:getBlockType("forward")
  9298.         local blockTypeD = T:getBlockType("down")
  9299.         while blockTypeF == "minecraft:obsidian" or blockTypeD == "minecraft:obsidian" do
  9300.             T:go("F1x0")
  9301.             width = width + 1
  9302.             blockTypeF = T:getBlockType("forward")
  9303.             blockTypeD = T:getBlockType("down")
  9304.         end
  9305.         -- will always go outside the tower 1 block. width of 5 block tower = 6
  9306.         T:go("R2F"..math.ceil(width / 2)) --return to centre of tower
  9307.         T:turnLeft(1) -- now find another edge of the tower, dig forward until out of obsidian
  9308.         for i = 1, math.ceil(width) do  -- give additional loops if missed centre
  9309.             blockTypeF = T:getBlockType("forward")
  9310.             blockTypeD = T:getBlockType("down")
  9311.             if blockTypeF == "minecraft:obsidian" or blockTypeD == "minecraft:obsidian" then
  9312.                 T:go("F1x0")
  9313.             else
  9314.                 break
  9315.             end
  9316.         end
  9317.         -- now outside different edge of the tower
  9318.         -- reverse and move width/2, dig up + 1 to mark centre, face original direction
  9319.         T:go("L2F"..math.ceil(width / 2).."R1U2x1")
  9320.         T:place("minecraft:end_stone", "forward", false) -- place endstone to mark facing direction
  9321.         T:down(2)
  9322.     end
  9323.    
  9324.     function lib.findPath(maxLength)
  9325.         local blockTypeD = T:getBlockType("down")
  9326.         local distance = 0
  9327.         while blockTypeD:find("cobble") == nil and distance < maxLength do
  9328.             T:go("F1x0")                            -- return to 0 axis,
  9329.             distance = distance + 1
  9330.             blockTypeD = T:getBlockType("down")
  9331.         end
  9332.         return distance
  9333.     end
  9334.    
  9335.     -- start at 0,y,0, facing West
  9336.     T:dig("up")                                 -- in case not already done
  9337.     local maxLength = 0
  9338.     local blockTypeD
  9339.     local distance = lib.findNextTower(45, true)-- find W tower (1) and mark trail with cobble
  9340.     T:turnRight(2)                     
  9341.     for i = 1, 8 do                             -- head back East 8 blocks, turn left (facing north)
  9342.         T:go("F1x0")                            -- this path may be off-axis, so dig double height
  9343.     end
  9344.     T:turnLeft(1)
  9345.     lib.findNextTower(30)                       -- find NWW tower (2)
  9346.     T:turnRight(2)
  9347.     distance = lib.findPath(30)
  9348.     distance = distance + lib.findNextTower(30) -- find SWW tower (3)
  9349.     T:turnRight(2)
  9350.     distance = lib.findPath(30)
  9351.     T:turnRight(1)                              -- should be on cobble path
  9352.     for i = 1, 21 do                            -- move East 21 blocks, turn left facing North
  9353.         T:go("F1x0")
  9354.     end
  9355.     T:turnLeft(1)
  9356.    
  9357.     distance = lib.findNextTower(45)        -- find NNW tower (4)
  9358.     T:turnRight(1)                         
  9359.     distance = lib.findNextTower(30)        -- find NNE tower (5)
  9360.     T:turnRight(1)
  9361.     distance = lib.findNextTower(85)        -- find SSE tower (6)
  9362.     T:turnRight(1)
  9363.  
  9364.     distance = lib.findNextTower(30)        -- find SSW tower (7)
  9365.     T:turnRight(1)
  9366.     distance = lib.findPath(40)             -- head North to 0 axis
  9367.     T:go("R1F13")                           -- return to 0,0 facing East
  9368.     distance = lib.findNextTower(45, true)  -- find E tower (8)
  9369.    
  9370.     T:turnRight(2)                     
  9371.     for i = 1, 9 do
  9372.         T:go("F1x0")                        -- this path may be off-axis, so dig double height
  9373.     end
  9374.     T:turnRight(1)
  9375.    
  9376.     distance = lib.findNextTower(30)        -- find NEE tower (9)
  9377.     T:turnRight(2)
  9378.     distance = lib.findPath(30) -- return to 0 axis
  9379.     distance = lib.findNextTower(30)        -- find SEE tower (10)
  9380.     T:turnRight(2)
  9381.     distance = lib.findPath(30)             -- return to 0 axis
  9382.     T:go("L1F33")                           -- return to 0, 0
  9383.    
  9384.     return {}
  9385. end
  9386.  
  9387. local function demolishPortal(R) -- 43
  9388.     --[[
  9389.     R.length = length of portal NOT width default 4
  9390.     R.height = height of portal default 5
  9391.     R.width = thickness of portal default 1
  9392.     R.data = "bury" to embed bottom into ground
  9393.     R.subChoice 1 = facing portal, 2 = aligned
  9394.     ]]
  9395.     local data = R.data
  9396.     if R.subChoice == 1 then -- facing portal
  9397.         T:go("F"..R.width.."R1")
  9398.     else
  9399.         T:forward(1)
  9400.     end
  9401.     if R.data == "bury" then
  9402.         T:down(1)
  9403.     end
  9404.  
  9405.     R.data = "up"
  9406.     R.silent = true
  9407.     if R.width == 1 then
  9408.         clearWall(R)
  9409.     else
  9410.         clearBuilding(R, true, true)
  9411.     end
  9412.     if data == "bury" then
  9413.         T:up(1)
  9414.     end
  9415.     if R.subChoice == 1 then -- facing portal
  9416.         T:go("R1F1L1F1L1")
  9417.     end
  9418.    
  9419.     return {}
  9420. end
  9421.  
  9422. local function digTrench(R) -- 77
  9423.     local blockType
  9424.     -- go down R.height, move forward
  9425.     if R.length == 0 then
  9426.         R.length = 4096 -- will go out of loaded chunks and stop or max 4096 on a server
  9427.     end
  9428.     for i = 1, R.length do
  9429.         local count = 0
  9430.         for down = 1, R.height do
  9431.             blockType = T:isWaterOrLava("down")
  9432.             -- go down only if no water or lava below
  9433.             if blockType:find("water") == nil and blockType:find("lava") == nil then
  9434.                 T:down(1)
  9435.                 count = count + 1
  9436.             end
  9437.         end
  9438.         T:go("C2", false, 0, true)              -- if empty below fill void
  9439.         T:go("U"..count)                        -- return to surface, continue if block above
  9440.         while turtle.detect() do                -- go up while block in front
  9441.             blockType = T:getBlockType("forward")
  9442.             if T:isVegetation(blockType) then
  9443.                 T:dig("forward")
  9444.                 break
  9445.             elseif blockType:find("log") ~= nil then
  9446.                 T:harvestTree("forward", false)
  9447.             else
  9448.                 T:up(1)
  9449.             end
  9450.         end
  9451.         T:forward(1)                            -- move forward
  9452.         while not turtle.detectDown() do        -- go down until block detected
  9453.             blockType = T:isWaterOrLava("down")
  9454.             if blockType:find("water") == nil and blockType:find("lava") == nil then
  9455.                 T:down(1)
  9456.             else
  9457.                 break
  9458.             end
  9459.         end
  9460.     end
  9461.    
  9462.     return {}
  9463. end
  9464.  
  9465. local function harvestTreeFarm(R) -- 24
  9466.     local lib = {}
  9467.    
  9468.     function lib.getLogCount()
  9469.         local count = 0
  9470.         for i = 1,16 do
  9471.             local item, itemCount = T:getSlotContains(i)
  9472.             if item:find("log") ~= nil then
  9473.                 count = count + itemCount
  9474.             end
  9475.         end
  9476.        
  9477.         return count
  9478.     end
  9479.    
  9480.     function lib.initialise(R)
  9481.         -- assumes legacy tree farm with turtle on polished block 4 blocks from corner
  9482.         local message  = ""
  9483.         R.treeSize = "single"
  9484.         local blockType = T:getBlockType("forward")
  9485.         --local logType = ""
  9486.         --local startHeight = 0
  9487.         local range = 0
  9488.         if blockType == "" then
  9489.             while turtle.forward() do
  9490.                 range = range + 1
  9491.                 if range == 3 then
  9492.                     break
  9493.                 end
  9494.             end
  9495.         end
  9496.         blockType = T:getBlockType("forward")
  9497.         if blockType:find("dirt") ~= nil then   -- dirt found
  9498.             T:go("R1F1L1")
  9499.             blockType = T:getBlockType("forward")
  9500.             if blockType:find("dirt") ~= nil then
  9501.                 R.treeSize = "double"
  9502.             end
  9503.             T:go("L1F1 R1")
  9504.         else    -- dirt NOT found where expected
  9505.             message = "Unable to determine position"
  9506.         end
  9507.         T:up(1)
  9508.         blockType = T:getBlockType("forward")   -- 1 block above dirt
  9509.         if blockType:find("log") ~= nil or blockType:find("sapling") ~= nil or blockType:find("propagule") ~= nil then
  9510.             local parts = T:getNames(blockType)
  9511.             if parts[2] == "dark" then
  9512.                 R.logType = "dark_oak"
  9513.             else
  9514.                 R.logType = parts[2]                -- eg "oak"
  9515.             end
  9516.             if R.logType == "mangrove" then
  9517.                 R.useBlockType = "mangrove_propagule"
  9518.             else
  9519.                 R.useBlockType = parts[2].."_sapling"   -- eg "oak_sapling"
  9520.             end
  9521.         end
  9522.         T:down(1)   -- facing dirt
  9523.        
  9524.         return R, message
  9525.     end
  9526.    
  9527.     function lib.waitForGrowth(R)
  9528.         local pattern = R.treeSize  --"single","double"
  9529.         local elapsed = 0
  9530.         local facing = "left"
  9531.         local ready = {}
  9532.         ready.left = false
  9533.         ready.top = false
  9534.         ready.right = false
  9535.         ready.bottom = false
  9536.         if R.logType == "mangrove" then
  9537.             pattern = "mangrove"
  9538.             local facings = {"left", "top", "right", "bottom"}
  9539.             T:up(1) -- go up from dirt to sapling level
  9540.             while not ready.left or not ready.right or not ready.top or not ready.bottom do
  9541.                 for i = 1, 4 do
  9542.                     local blockType = T:getBlockType("forward")
  9543.                     if blockType:find("propagule") ==  nil then -- either grown or deleted by roots
  9544.                         ready[facings[i]] = true
  9545.                     end
  9546.                     T:turnRight(1)
  9547.                 end
  9548.                 if ready.left and ready.right and ready.top and ready.bottom then
  9549.                     break
  9550.                 else
  9551.                     sleep(15)
  9552.                     elapsed = elapsed + 15
  9553.                     if  elapsed / 60 > 15 then  -- max 15 mins real time before farm is harvested
  9554.                         break
  9555.                     end
  9556.                 end
  9557.                 print("Waiting for mangrove growth "..elapsed / 60 .." minutes")
  9558.                 print("Left = "..tostring(ready.left)..
  9559.                       ", top = "..tostring(ready.top)..
  9560.                       ", right = "..tostring(ready.right)..
  9561.                       ", bottom = "..tostring(ready.bottom))
  9562.                
  9563.             end
  9564.             --T:go("L1D1")
  9565.             T:turnLeft(1)   -- face front
  9566.         else
  9567.             while not ready.left or not ready.right do
  9568.                 T:up(1) -- go up from dirt to sapling level
  9569.                 local blockType = T:getBlockType("forward")
  9570.                 if blockType:find("log") ~=  nil then
  9571.                     ready[facing] = true
  9572.                 end
  9573.                 if pattern == "single" then
  9574.                     --alternate between 2 trees, starts facing left
  9575.                     T:turnRight(2)
  9576.                     if facing == "left" then
  9577.                         facing = "right"
  9578.                     elseif facing == "right" then
  9579.                         facing = "left"
  9580.                     end
  9581.                     blockType = T:getBlockType("forward")
  9582.                     if blockType:find("log") ~=  nil then
  9583.                         ready[facing] = true
  9584.                     end
  9585.                     T:down(1)   -- drop below sapling to dirt level
  9586.                 elseif pattern == "double" then
  9587.                     if ready.left and facing == "left" then-- tree on left now grown. check right
  9588.                         T:go("R2F2")
  9589.                         facing = "right"
  9590.                     end
  9591.                     blockType = T:getBlockType("forward")
  9592.                     if blockType:find("log") ~=  nil then
  9593.                         ready[facing] = true
  9594.                     end
  9595.                     T:down(1)   -- drop below sapling to dirt level
  9596.                 end
  9597.                 T:clear()
  9598.                 print("Farm type: "..pattern)
  9599.                 print("Waiting for tree growth "..elapsed / 60 .." minutes")
  9600.                 print("Left grown = "..tostring(ready.left)..", right grown = "..tostring(ready.right))
  9601.                 if not ready.left or not ready.right then
  9602.                     sleep(15)
  9603.                     elapsed = elapsed + 15
  9604.                     if pattern == "single" and elapsed / 60 > 10 then   -- max 10 mins real time before farm is harvested
  9605.                         break
  9606.                     elseif pattern == "double" and elapsed / 60 > 15 then   -- max 15 mins real time before farm is harvested
  9607.                         break
  9608.                     end
  9609.                 end
  9610.             end
  9611.         end
  9612.         -- growth complete
  9613.         if pattern == "single" then
  9614.             if facing == "right" then
  9615.                 T:turnRight(1)
  9616.             else
  9617.                 T:turnLeft(1)
  9618.             end
  9619.             --T:go("F1R1 F3R1")         -- facing first dirt
  9620.             T:go("F1R1 F1R1")           -- facing first dirt
  9621.         elseif pattern == "double" then -- assume on right side
  9622.             if facing == "right" then
  9623.                 T:go("R1F1 R1F4 R1")
  9624.             else
  9625.                 T:go("L1F1 R1F2 R1")
  9626.             end
  9627.         elseif pattern == "mangrove" then
  9628.             T:go("D2F6 U1F1 R1F6 R1F1 U1")
  9629.         end
  9630.         -- ends facing dirt at base of first tree
  9631.         -- no return needed, function exit so trees are grown
  9632.     end
  9633.    
  9634.     function lib.watchFarm(R)
  9635.         -- rotate to check if dirt on both sides
  9636.         -- R.subChoice == 1: 16 single trees, 2 = 4 doubles
  9637.         if R.logType ~= "mangrove" then
  9638.             R.treeSize = "single"
  9639.             T:turnRight(2)  -- if no dirt then on 4 x doubles
  9640.             if T:getBlockType("forward") == "" then
  9641.                 R.treeSize = "double"
  9642.             end
  9643.             T:saveToLog("lib.watchFarm: R.treeSize set to '"..R.treeSize.."'")
  9644.             T:turnLeft(2)   -- face lower left double dirt
  9645.         end
  9646.         lib.waitForGrowth(R)
  9647.        
  9648.         return R
  9649.     end
  9650.    
  9651.     function lib.harvestSingle(direction, moves)
  9652.         -- if direction == "up": starting inside tree on dirt at dirt level
  9653.         T:saveToLog("lib.harvestSingle('"..direction.."', moves = "..moves)
  9654.         if direction == "up" then
  9655.             while turtle.detectUp() do
  9656.                 T:up(1)
  9657.                 moves = moves + 1
  9658.             end
  9659.         else    -- direction = "down", but could be mid-tree
  9660.             local movesUp = 0
  9661.             while turtle.detectUp() do
  9662.                 T:up(1)
  9663.                 movesUp = movesUp + 1
  9664.             end
  9665.             T:down(movesUp)
  9666.             T:down(moves)
  9667.         end
  9668.         return moves
  9669.     end
  9670.    
  9671.     function lib.harvestSingleRow()
  9672.         -- start next to tree/dirt
  9673.         T:go("F1")
  9674.         local moves = lib.harvestSingle("up", 0)
  9675.         T:go("F2")
  9676.         lib.harvestSingle("down", moves)
  9677.         T:go("F2")
  9678.         moves = lib.harvestSingle("up", 0)
  9679.         T:go("F2")
  9680.         lib.harvestSingle("down", moves)
  9681.     end
  9682.    
  9683.     function lib.harvestDouble()
  9684.         T:forward(1)    -- dig dirt, move into left (S) corner
  9685.         local moves = 0
  9686.         while turtle.detectUp() do
  9687.             turtle.dig()
  9688.             turtle.digUp()
  9689.             turtle.up()
  9690.             moves = moves + 1
  9691.         end
  9692.         turtle.dig()
  9693.         T:go("R1F1L1")  -- move to right corner
  9694.         for i = 1, moves do
  9695.             turtle.dig()
  9696.             turtle.digDown()
  9697.             turtle.down()
  9698.         end
  9699.         turtle.dig()
  9700.         T:go("F1L1F1R1") -- move to left corner (N)
  9701.     end
  9702.        
  9703.     --T:setUseLog(true, "treeFarmLog.txt", true)    -- T:setUseLog(use, filename, delete)
  9704.     --dbug = true                               -- set dbug flag
  9705.    
  9706.     R.silent = true
  9707.     -- if on modem, R.networkFarm has already been set
  9708.     T:saveToLog("harvestTreeFarm(R) R.networkFarm = "..tostring(R.networkFarm))
  9709.  
  9710.     if R.networkFarm then
  9711.         local message = network.loadStorageLists()  -- initialises or creates lists of where an item can be found: GLOBAL LISTS!
  9712.         if message ~= nil then return {message} end
  9713.         network.emptyInventory(R, {"sapling", "propagule", "dirt"}, {"all"}, true)
  9714.     end
  9715.    
  9716.     if turtle.getFuelLevel() < turtle.getFuelLimit() / 2 then
  9717.         local turtleSlot, turtleCount = network.getItemFromNetwork("chest", "log", 16)
  9718.         if turtleSlot > 0 then
  9719.             if turtle.craft() then
  9720.                 turtle.refuel()
  9721.                 T:saveToLog("harvestTreeFarm(R) R.networkFarm = "..tostring(R.networkFarm))
  9722.             end
  9723.         end
  9724.     end
  9725.    
  9726.     if R.networkFarm then           -- monitor trees for growth
  9727.         if R.logType == "mangrove" then
  9728.             T:go("F1D1")
  9729.             T:place("dirt", "up")
  9730.             T:go("F6x2U1L1")    -- move under dirt covering, clear roots from hopper, move level with dirt, face left
  9731.         else
  9732.             T:go("F4R1")                -- either in between 2 trees or in gap between double trees at dirt level
  9733.             if turtle.detect() then     -- single trees, move between 1st and 2nd rows
  9734.                 T:go("R1F1 R1F2 R1F1 L1")
  9735.             else    -- using double trees
  9736.                 T:go("R2F1")
  9737.                 if not turtle.detect() then
  9738.                     return {"Unable to determine position"}
  9739.                 end
  9740.             end
  9741.         end
  9742.         R = lib.watchFarm(R)    -- wait for trees to grow, then start harvest
  9743.     else
  9744.         --logType, startHeight, R = lib.initialise(R) -- possible ID tree type + move to correct position 1 above dirt
  9745.         R, message = lib.initialise(R)      -- possible ID tree type + move to correct position next to first dirt
  9746.         if message ~= "" then
  9747.             return{message}
  9748.         end
  9749.     end
  9750.     T:saveToLog("R.networkFarm = "..tostring(R.networkFarm)..", logType = "..R.logType..", treeSize = "..R.treeSize)
  9751.     -- different clearing for different trees:
  9752.     -- double spruce and jungle: staight up/down or small
  9753.     if R.logType == "mangrove" then
  9754.         clearRectangle({width = 13, length = 13, up = true, down = true })
  9755.         T:go("U2F2 R1F2L1")
  9756.         clearSolid({width = 9, length = 9, height = 18, direction ="up"})
  9757.         T:go("D3R1 F4R1 F3R2")
  9758.         network.emptyInventory(R, {"sapling", "propagule", "dirt", "crafting"}, {"all"}, true)
  9759.     elseif R.logType == "birch" or R.logType == "spruce" or R.logType == "jungle" then  -- use column harvest
  9760.         if R.treeSize == "single" then
  9761.             lib.harvestSingleRow()
  9762.             T:go("F1R1F2R1")
  9763.             lib.harvestSingleRow()
  9764.             T:go("F1L1F2L1")
  9765.             lib.harvestSingleRow()
  9766.             T:go("F1R1F2R1")
  9767.             lib.harvestSingleRow()
  9768.             if R.networkFarm then
  9769.                 T:go("F1R1 F3L1 F3R2")
  9770.                 network.emptyInventory(R, {"sapling", "propagule", "dirt", "crafting"}, {"all"}, true)
  9771.             else
  9772.                 T:go("F1R1 F6L1 F3R2")
  9773.             end
  9774.         else
  9775.             lib.harvestDouble()
  9776.             T:go("F3")
  9777.             lib.harvestDouble()
  9778.             T:go("R1F4")
  9779.             lib.harvestDouble()
  9780.             T:go("R1F4")
  9781.             lib.harvestDouble()
  9782.             if R.networkFarm then
  9783.                 T:go("F1R1 F3L1 F3R2")
  9784.                 network.emptyInventory(R, {"sapling", "propagule", "dirt", "crafting"}, {"all"}, true)
  9785.             else
  9786.                 T:go("F1R1 F6L1 F3R2")
  9787.             end
  9788.         end
  9789.     else    -- use cube harvest, including mangrove
  9790.         local size = 10
  9791.         local start = "L1F1 L1F1 R2"
  9792.         local finish = "R1F1 R1F3 R2"
  9793.         if R.logType == "acacia" then
  9794.             size = 12
  9795.             start = "L1F2 L1F2 R2"
  9796.             finish = "R1F2 R1F2 R2"
  9797.         elseif R.logType == "cherry" then
  9798.             size = 14
  9799.             start = "L1F3 L1F3 R2"
  9800.             finish = "R1F3 R1F1 R2"
  9801.         end
  9802.         T:go("U1F1")    -- into first log at base of tree
  9803.         clearRectangle({width = 7, length = 7, up = true, down = true})
  9804.         --T:go("L1F2 L1F2 R2")  -- rect size 12
  9805.         --T:go("L1F1 L1F1 R2")  -- rect size 10
  9806.         T:go(start)
  9807.         local height = 0
  9808.         local pastLogCount = -1
  9809.         local currentLogCount = lib.getLogCount()
  9810.         --while currentLogCount ~= pastLogCount do
  9811.         while currentLogCount - pastLogCount > 5 do -- ony continue if at least 5 logs harvested last round
  9812.             T:up(3)
  9813.             height = height + 3
  9814.             pastLogCount = currentLogCount -- 0 when loop starts
  9815.             --clearRectangle({width = 12, length = 12, up = true, down = true})
  9816.             clearRectangle({width = size, length = size, up = true, down = true})
  9817.             currentLogCount = lib.getLogCount() -- eg 20 logs from first round
  9818.         end
  9819.         T:down(height + 1)
  9820.         --T:go("R1F2 R1F2 R2")      -- on polished stone rect size 12
  9821.         --T:go("R1F1 R1F3 R2")      -- on polished stone rect size 10
  9822.         T:go(finish)
  9823.         if R.networkFarm then
  9824.             T:go("R1F3 L1")         -- on modem
  9825.             network.emptyInventory(R, {"sapling", "propagule", "dirt", "crafting"}, {"all"}, true)
  9826.         end
  9827.     end
  9828.    
  9829.     return {}   -- if player initiated, stops here. If R.auto then returns to plantTreeFarm()
  9830. end
  9831.  
  9832. local function harvestObsidian(R) -- 41
  9833.     local lib = {}
  9834.    
  9835.     function lib.forward(move)
  9836.         T:isWaterOrLava("forward")  -- automatically use lava ahead to refuel
  9837.         T:isWaterOrLava("down")     -- automatically use lava down to refuel
  9838.         T:go("C2", false, 0, false) -- place / replace block below
  9839.         if move then
  9840.             T:forward(1)
  9841.         end
  9842.     end
  9843.    
  9844.     function lib.home(outward)
  9845.         if outward then
  9846.             T:go("L1F"..R.width - 1 .."L1F"..R.length - 1)
  9847.         else   
  9848.             T:go("R1F"..R.width - 1 .."R1")
  9849.         end
  9850.     end
  9851.    
  9852.     function lib.start()
  9853.         local lavaSlot = T:getItemSlot("lava")
  9854.         if lavaSlot > 0 then
  9855.             turtle.select(slot)
  9856.             turtle.refuel()
  9857.         end
  9858.         T:down(1)
  9859.     end
  9860.    
  9861.     local outward = true
  9862.    
  9863.     lib.start()                     -- use lava bucket if placed, move down into block below
  9864.     for w = 1, R.width do
  9865.         for l = 1, R.length do
  9866.             if l < R.length then
  9867.                 lib.forward(true)
  9868.             else
  9869.                 lib.forward(false)
  9870.             end
  9871.         end
  9872.         if w < R.width then
  9873.             if outward then
  9874.                 T:go("R1F1R1")
  9875.             else
  9876.                 T:go("L1F1L1")
  9877.             end
  9878.             outward = not outward
  9879.         end
  9880.     end
  9881.    
  9882.     lib.home(outward)
  9883.    
  9884.     return {}
  9885. end
  9886.  
  9887. local function harvestShulkers(R)
  9888.     local lib = {}
  9889.    
  9890.     function lib.attackAll()
  9891.         return turtle.attack(), turtle.attackUp(), turtle.attackDown()
  9892.     end
  9893.  
  9894.     function lib.attack()
  9895.         local forward, up, down = lib.attackAll()
  9896.         while forward or up or down do
  9897.             forward, up, down = lib.attackAll()
  9898.             sleep(0.2)
  9899.         end
  9900.     end
  9901.    
  9902.     function lib.approach(direction, limit, dig)
  9903.         -- move forward until stopped by shulker
  9904.         limit = limit or 64
  9905.         dig = dig or false
  9906.         local count = 0
  9907.         local solidDown = false
  9908.         move = turtle.forward
  9909.         if direction == "up" then
  9910.             move = turtle.up
  9911.         elseif direction == "down" then
  9912.             move = turtle.down
  9913.         end
  9914.         local forward, up, down = lib.attackAll()
  9915.         if forward or up or down then -- attacks suceeded
  9916.             return true, 0, solidDown
  9917.         else
  9918.             while move() do
  9919.                 count = count + 1
  9920.                 if turtle.detectDown() then
  9921.                     solidDown = true
  9922.                 end
  9923.                 if count >= limit then
  9924.                     return false, count, solidDown
  9925.                 end
  9926.                 forward, up, down = lib.attackAll()
  9927.                 if forward or up or down then
  9928.                     return true, count, solidDown
  9929.                 else
  9930.                     if dig then
  9931.                         T:dig("forward")
  9932.                     end
  9933.                 end
  9934.             end
  9935.             return false, count, solidDown
  9936.         end
  9937.     end
  9938.    
  9939.     function lib.home(direction, moves)
  9940.         local move = turtle.back
  9941.         if direction == "up" then
  9942.             move = turtle.down
  9943.         elseif direction == "down" then
  9944.             move = turtle.up
  9945.         end
  9946.         for i = 1, moves do
  9947.             move()
  9948.         end
  9949.     end
  9950.    
  9951.     function lib.checkPosition()
  9952.         if T:detect("forward") then         -- wall in front
  9953.             T:turnRight(1)
  9954.             if T:detect("forward") then     -- wall to right
  9955.                 T:turnLeft(1)
  9956.                 return true                 -- position corrected
  9957.             else
  9958.                 T:turnLeft(2)
  9959.                 if T:detect("forward") then -- wall in front. position correct
  9960.                     return true
  9961.                 end
  9962.             end
  9963.         end
  9964.         return false
  9965.     end
  9966.    
  9967.     function lib.getLength()
  9968.         local count = 0
  9969.         while turtle.detectDown() do
  9970.             count = count + 1
  9971.             T:forward(1)
  9972.         end
  9973.         utils.goBack(1)
  9974.         return count
  9975.     end
  9976.    
  9977.     function lib.clearWall()
  9978.         local distance = 0
  9979.         while distance < 9 do   -- smallest external walls are 8 x 8
  9980.             local success, count = lib.approach("forward", 8 - distance)
  9981.             if success then
  9982.                 lib.attack()
  9983.             end
  9984.             distance = distance + count
  9985.         end
  9986.         T:turnLeft(1)
  9987.         if T:detect("forward") then -- larger than 8 blocks
  9988.             T:turnRight(1)
  9989.             while distance < 14 do  -- larger builds are 14 x 14
  9990.                 local success, count = lib.approach("forward", 13 - distance)
  9991.                 if success then
  9992.                     lib.attack()
  9993.                 end
  9994.                 distance = distance + count
  9995.             end
  9996.             T:go("L1x2")
  9997.         end
  9998.     end
  9999.    
  10000.     function lib.roof()
  10001.         -- check position
  10002.         local doContinue = lib.checkPosition()
  10003.        
  10004.         if doContinue then
  10005.             T:go("U2F3 R1F3 R1")
  10006.             local length = lib.getLength()
  10007.             local width = 1
  10008.             local outward = lib.turnRound(true)
  10009.             local success, count, onRoof = false, 0, true  
  10010.             while onRoof do
  10011.                 local distance = 0
  10012.                 while distance < length - 1 do
  10013.                     success, count, onRoof = lib.approach("forward", length - 1 - distance)
  10014.                     if success then
  10015.                         lib.attack()
  10016.                     end
  10017.                     if count == 0 then
  10018.                         turtle.dig()
  10019.                     end
  10020.                     distance = distance + count
  10021.                 end
  10022.                 width = width + 1
  10023.                 outward = lib.turnRound(outward)
  10024.             end
  10025.             if outward then
  10026.                 T:go("F".. 3 .."L1F"..width - 3 .."D2L1")
  10027.             else
  10028.                 T:go("F".. length - 3 .."R1F"..width - 3 .."D2L1")
  10029.             end
  10030.             return {}
  10031.         else
  10032.             return {"Turtle not in front of a wall"}
  10033.         end
  10034.     end
  10035.    
  10036.     function lib.turnRound(outward)
  10037.         if outward then
  10038.             T:go("R1F1R1")
  10039.         else
  10040.             T:go("L1F1L1")
  10041.         end
  10042.         return not outward
  10043.     end
  10044.  
  10045.     function lib.upAndOver()
  10046.         local start, height, forward = 0, 0, 0
  10047.         while turtle.detect() do
  10048.             turtle.up()
  10049.             start = start + 1
  10050.         end
  10051.         while turtle.forward() do
  10052.             forward = forward + 1
  10053.         end
  10054.         turtle.turnRight()
  10055.         while not turtle.detectUp() do
  10056.             local success, count = lib.approach("up", 64, true)
  10057.             if success then
  10058.                 lib.attack()
  10059.             end
  10060.             height = height + count
  10061.         end
  10062.         T:turnRight(2)
  10063.         while not turtle.detectDown() do
  10064.             if lib.approach("down", 64, true) then
  10065.                 lib.attack()
  10066.             end
  10067.         end
  10068.         T:turnLeft(1)
  10069.         for i = 1, forward do
  10070.             turtle.forward()
  10071.         end
  10072.         for i = 1, start do
  10073.             turtle.down()
  10074.         end
  10075.         T:turnRight(2)
  10076.     end
  10077.    
  10078.     function lib.walls()
  10079.         local doContinue = lib.checkPosition()
  10080.         if doContinue then
  10081.             T:go("F2R1 F2R2 D1x2")      -- on corner outside middle of wall, facing along its length, dig light rod
  10082.             for i = 1, 4 do
  10083.                 lib.clearWall()
  10084.             end
  10085.             T:go("F2L1 U1F2 R2")
  10086.         else
  10087.             return {"Turtle not in front of a wall"}
  10088.         end
  10089.     end
  10090.        
  10091.     local direction = "forward"
  10092.     local doContinue = false
  10093.    
  10094.     if R.subChoice == 1 then        -- Shulker is above
  10095.         direction = "up"
  10096.         doContinue = true
  10097.     elseif R.subChoice == 2 then    -- Shulker is ahead
  10098.         doContinue = true
  10099.     elseif R.subChoice == 3 then    -- Shulker is below
  10100.         direction = "down"
  10101.         doContinue = true
  10102.     else
  10103.         if R.subChoice == 4 then    -- Climb tower wall
  10104.             lib.upAndOver()
  10105.         elseif R.subChoice == 5 then    -- Clear roof above
  10106.             return lib.roof()
  10107.         elseif R.subChoice == 6 then    -- Clear outside walls
  10108.             lib.walls()
  10109.         end
  10110.    
  10111.     end
  10112.     if doContinue then
  10113.         local success, moves = lib.approach(direction, 64)
  10114.         if success then
  10115.             lib.attack()
  10116.         end
  10117.         lib.home(direction, moves)
  10118.     end
  10119.    
  10120.     return {}
  10121. end
  10122.  
  10123. local function upgradeFarmland(R)
  10124.     local essences = {":inferium", ":prudentium", ":tertium", ":imperium", ":supremium", ":awakened" , ":insanium"}
  10125.    
  10126.     local lib = {}
  10127.    
  10128.     function lib.isHigherTeir(essence, farmLand)
  10129.         -- eg "ma:prudentium_essence", "mc:farmland"
  10130.         local teir = 0
  10131.         for index = 1, #essences do
  10132.             if farmLand:find(essences[index]) ~= nil then
  10133.                 teir = index    -- 0 if vanilla, else eg 2 for prudentium
  10134.                 break
  10135.             end
  10136.         end
  10137.         for index = 1, #essences do
  10138.             if essence:find(essences[index]) ~= nil then
  10139.                 if index > teir then
  10140.                     return true
  10141.                 end
  10142.             end
  10143.         end
  10144.         return false
  10145.     end
  10146.    
  10147.     function lib.upgrade(essence, slot)
  10148.         -- essence is the slot no of any essence
  10149.         turtle.select(slot)
  10150.         local blockType = T:getBlockType("up")
  10151.         if blockType:find("farmland") ~= nil then           -- farmland found (vanilla or modded)
  10152.             if lib.isHigherTeir(essence, blockType) then    -- eg "ma:inferium", mc:farmland"
  10153.                 turtle.placeUp()
  10154.             end
  10155.         end
  10156.     end
  10157.     -- check if any essence still onboard
  10158.     T:checkInventoryForItem({"essence"}, {95}, false)
  10159.     -- return slotData.lastSlot, total, slotData -- integer, integer, table
  10160.     local name = ""
  10161.     local slot, amount, data = T:getItemSlot("essence")
  10162.     name = data.mostName or ""
  10163.     local empty = T:getFirstEmptySlot()
  10164.    
  10165.     if slot > 0 then
  10166.         T:go("D2")
  10167.         local outward = true
  10168.         for w = 1, 10 do
  10169.             for i = 1, 9 do
  10170.                 if turtle.getItemCount(slot) == 0 then
  10171.                     slot, amount, data = T:getItemSlot("essence")
  10172.                     name = data.mostName or ""
  10173.                     if slot == 0 then
  10174.                         slot = empty
  10175.                     end
  10176.                 end
  10177.                 lib.upgrade(name, slot)
  10178.                 if w == 1 and i == 1 then
  10179.                     T:go("F1R2 C1R2")
  10180.                 elseif w == 10 and i == 1 then
  10181.                    
  10182.                 else
  10183.                     T:forward(1)
  10184.                 end
  10185.             end
  10186.             if outward then
  10187.                 lib.upgrade(name, slot)
  10188.                 if w == 1 then
  10189.                     T:go("L1F1 L2C1 R1")
  10190.                 elseif w == 9 then
  10191.                     T:go("L1F1 L1F1 L2C1 L2")
  10192.                 else
  10193.                     T:go("L1F1L1")
  10194.                 end
  10195.             else
  10196.                 if w < 10 then
  10197.                     lib.upgrade(name, slot)
  10198.                     T:go("R1F1R1")
  10199.                 else
  10200.                     T:go("L1F1 L2C1 R2F8 L1U1 C2U2") -- 1 above normal position
  10201.                 end
  10202.             end
  10203.             outward = not outward
  10204.         end
  10205.         -- sometimes original essence pops out of the ground when upgraded, so rescue it
  10206.         for w = 1, 10 do
  10207.             for i = 1, 9 do
  10208.                 turtle.suckDown()
  10209.                 turtle.suck()
  10210.                 T:forward(1)
  10211.             end
  10212.             if outward then
  10213.                 turtle.suckDown()
  10214.                 turtle.suck()
  10215.                 T:go("L1F1L1")
  10216.             else
  10217.                 if w < 10 then
  10218.                     turtle.suckDown()
  10219.                     turtle.suck()
  10220.                     T:go("R1F1R1")
  10221.                 else
  10222.                     T:go("L1F9 L1D1") -- normal position
  10223.                 end
  10224.             end
  10225.             outward = not outward
  10226.         end
  10227.     end
  10228.     return ({"Farmland Upgraded"})
  10229. end
  10230.  
  10231. local function manageFarm(R)
  10232.     local lib = {}
  10233.        
  10234.     function lib.askPlayerForCrops()
  10235.         local seed  = ""
  10236.         pp.itemColours = {colors.lightGray, colors.red, colors.orange, colors.brown, colors.magenta, colors.yellow}
  10237.         crops = {"minecraft:wheat_seeds", "minecraft:beetroot_seeds", "minecraft:carrots", "minecraft:potatoes", "mysticalagriculture", "none"}
  10238.         choices = {"wheat (seeds)", "beetroot (seeds)", "carrot", "potato", "Mystical Agriculture", "Till soil only"}
  10239.         choice = menu.menu("Choose preferred crop", choices, pp, "Type number of your choice")
  10240.         crop = crops[choice]
  10241.         if crop == "none" then
  10242.             return "", ""
  10243.         elseif crop == "mysticalagriculture" then
  10244.             T:checkInventoryForItem({"seeds"}, {95}, true, "Add one type of M. Agriculture seeds")
  10245.         else
  10246.             T:checkInventoryForItem({crop}, {95}, true, "Do not mix! add as many as you want")
  10247.         end
  10248.         crop = T:getMostItem("", false)     -- not searching for any specific item, not checking stone only
  10249.         -- crop could be wheat/beetroot seeds, carrots, potatoes or mystical agriculture seeds
  10250.         seed, crop = lib.getCropSeed(crop)  -- seed type or "", crop type
  10251.         return seed, crop  
  10252.     end
  10253.        
  10254.     function lib.assessPlot(R)
  10255.         local crop = T:getBlockType("forward")  -- convert ma:inferium_crop to ma:inferium_seeds
  10256.         local seed = lib.getCropSeed(crop)
  10257.         turtle.down()                                   -- into water source
  10258.         local soil = T:getBlockType("forward")
  10259.         turtle.up()                                     -- at crop level
  10260.         return crop, seed, soil
  10261.     end
  10262.    
  10263.     function lib.checkEquipment(R, itemRequired)
  10264.         -- Check if item is equipped and return it to slot
  10265.         T:clear()
  10266.         print("Checking equipment. Please wait...")
  10267.         local equipped, slot = T:getEquipped("right", false)    -- see what is equipped and replace it
  10268.         if equipped == itemRequired then
  10269.             equipped, slot = T:getEquipped("right", true)       -- remove item and return slot
  10270.             return slot
  10271.         end
  10272.  
  10273.         return 0    -- not found
  10274.     end
  10275.        
  10276.     function lib.crossFarm(R)
  10277.         -- used with goHome to find starting point
  10278.         local blockType = ""
  10279.         local isReady, cropType, seed, status
  10280.         isReady, cropType, seed, status, R = lib.isCropReady(R,"down")
  10281.         -- will go forward until chest, barrel, modem or cobble detected below
  10282.         -- if detected within 1 move, this is ignored
  10283.         local numMoves = 0
  10284.         local endOfPath = false
  10285.         while not endOfPath do
  10286.             blockType = T:getBlockType("down")
  10287.             if blockType == "" or cropType ~= "" then --crops or nothing below
  10288.                 turtle.forward()
  10289.             elseif  blockType:find("barrel") ~= nil or
  10290.                     blockType:find("chest") ~= nil or
  10291.                     blockType:find("modem") ~= nil or
  10292.                     blockType:find("cobble") ~= nil then
  10293.                 endOfPath = true
  10294.             end
  10295.             numMoves = numMoves + 1
  10296.         end
  10297.         return R, blockType -- either barrel, chest, modem or cobble
  10298.     end
  10299.    
  10300.     function lib.farmAll(R, isFarmToRight, isFarmToFront)
  10301.         local plotCountR = 0        -- plot a1 harvested
  10302.         local plotCountF = 0        -- plot a1 harvested
  10303.         if isFarmToRight then
  10304.             while isFarmToRight do  -- do all plots on this axis
  10305.                 R, isFarmToRight = lib.farmToRight(R) -- plot harvested and back to plot origin
  10306.                 plotCountR = plotCountR + 1
  10307.             end
  10308.             lib.goToLeft(R, plotCountR)
  10309.         end
  10310.         if isFarmToFront then
  10311.             while isFarmToFront do  -- do all plots on this axis
  10312.                 plotCountR = 0              -- reset
  10313.                 R, isFarmToRight, isFarmToFront = lib.farmInFront(R)
  10314.                 if isFarmToRight then
  10315.                     while isFarmToRight do  -- do all plots on this axis
  10316.                         R, isFarmToRight = lib.farmToRight(R) -- plot harvested and back to plot origin
  10317.                         plotCountR = plotCountR + 1
  10318.                     end
  10319.                     lib.goToLeft(R, plotCountR) -- return home and continue with front
  10320.                 end
  10321.                 plotCountF = plotCountF + 1
  10322.             end
  10323.             lib.goToFront(R, plotCountF)
  10324.         end
  10325.     end
  10326.    
  10327.     function lib.farmInFront(R)
  10328.         --[[ facing crops on first farm. move to next farm in front ]]
  10329.         T:go("U1L1 F11D1 R1")                       -- on next farm, facing crops
  10330.         local seed, crop = lib.manageTree(R)        -- refuel, gather seeds or other crops
  10331.         local isFarmToRight = false
  10332.         R, isFarmToRight = lib.harvest(R, seed, crop)       -- harvest field, store crops
  10333.         return R, isFarmToRight
  10334.     end
  10335.    
  10336.     function lib.farmToRight(R)
  10337.         --[[ facing crops on first farm. move to next farm on right side ]]
  10338.         T:go("U1F11 D1")                                -- on next farm, facing crops
  10339.         local seed, crop = lib.manageTree(R)                    -- refuel, gather seeds and other crops
  10340.         R, isFarmToRight = lib.harvest(R, seed, crop)       -- harvest field, store crops
  10341.         -- now at starting position of current plot
  10342.         return R, isFarmToRight
  10343.     end
  10344.    
  10345.     function lib.getCropSeed(crop)
  10346.         -- change growing crops into equivalent seed names
  10347.         -- crop could be seeds, so return equivalent crop
  10348.         local start = crop:find("_crop")
  10349.         if start ~= nil then    -- only modded seeds have "crop"
  10350.             return crop:sub(1, start).."seeds", crop
  10351.         end
  10352.         if crop:find("seeds") ~= nil then   -- asked to return crop from seed type
  10353.             if crop:find("wheat") ~= nil then
  10354.                 return "minecraft:wheat_seeds", "minecraft:wheat"
  10355.             end
  10356.             if crop:find("beetroot") ~= nil then
  10357.                 return "minecraft:beetroot_seeds", "minecraft:beetroot"
  10358.             end
  10359.         end
  10360.         if crop:find("wheat") ~= nil then
  10361.             return "minecraft:wheat_seeds", "minecraft:wheat"
  10362.         end
  10363.         if crop:find("beetroot") ~= nil then
  10364.             return "minecraft:beetroot_seeds", "minecraft:beetroot"
  10365.         end
  10366.         if crop:find("carrot") ~= nil then
  10367.             return "", "minecraft:carrot"
  10368.         end
  10369.         if crop:find("potato") ~= nil then
  10370.             return "", "minecraft:potato"
  10371.         end
  10372.         -- planted crops are plural, harvested singular: carrots / carrot, pototoes/ potato
  10373.         return "", crop -- no seed for carrot / potato
  10374.     end
  10375.    
  10376.     function lib.getHoe()
  10377.         local message = ""
  10378.         if R.networkFarm then   -- get hoe from storage
  10379.             hoeSlot = network.getItemFromNetwork("barrel", "minecraft:diamond_hoe", 1)
  10380.             if hoeSlot == 0 then
  10381.                 if T:checkInventoryForItem({"minecraft:diamond_hoe"}, {1}, true, "Diamond Hoe required for planting!") ~= nil then
  10382.                     hoeSlot = T:getItemSlot("minecraft:diamond_hoe")
  10383.                 end
  10384.             end
  10385.         else    -- put crafting table into barrel in floor
  10386.             T:go("L1F1")        -- over barrel
  10387.             local isStorage, storageType = utils.isStorage("down")
  10388.             if not isStorage then
  10389.                 message =
  10390. [["Farm upgrade required:
  10391. Previous designs had a crafting table
  10392. next to the water source.
  10393.  
  10394. 1. Place a barrel or chest under the turtle
  10395.     instead.
  10396.    
  10397. 2. Restart"
  10398. ]]
  10399.                 return 0, message
  10400.             end
  10401.             -- must be storage below to get this far
  10402.             T:suck("down")      -- if hoe already in storage get it out
  10403.             hoeSlot = T:getItemSlot("minecraft:diamond_hoe")
  10404.             if hoeSlot == 0 then
  10405.                 if T:checkInventoryForItem({"minecraft:diamond_hoe"}, {1}, true, "Diamond Hoe required for planting!") ~= nil then
  10406.                     hoeSlot = T:getItemSlot("minecraft:diamond_hoe")
  10407.                 end
  10408.             end
  10409.         end
  10410.         return hoeSlot, ""
  10411.     end
  10412.                
  10413.     function lib.getSaplings(R, direction)
  10414.         --[[ get a single sapling from the chest ]]
  10415.         if direction == nil then
  10416.             direction = "forward"
  10417.         end
  10418.         if R.networkFarm then
  10419.             network.getItemFromNetwork("barrel", "sapling", 1)
  10420.             --lib.getItemFromNetworkBarrels(R, "sapling", 1)
  10421.         else
  10422.             while T:suck(direction, 1) do end -- remove saplings/ seeds
  10423.             T:dropItem("sapling", direction, 1) -- drop all but 1 saplings
  10424.             if not T:dropItem("seeds", direction, 0) then-- drop all seeds
  10425.                 T:dropItem("seeds", "up", 0)
  10426.             end
  10427.         end
  10428.        
  10429.         return T:getItemSlot("sapling")
  10430.     end
  10431.    
  10432.     function lib.getSeeds(R, direction, seed)
  10433.         --[[
  10434.             seed = name of growing crops seed or ""
  10435.             turtle facing crops on networked, else facing storage
  10436.             allow for other seeds from mods eg MysticalAgriculture
  10437.             get 1 stack of seeds of whatever type is being farmed
  10438.         ]]
  10439.         if seed == "" then
  10440.             return "", 0
  10441.         end
  10442.         local inventorySlot, seedCount = 0, 0
  10443.         T:saveToLog("Collecting seeds from storage")
  10444.         if R.networkFarm then
  10445.             inventorySlot, seedCount = network.getItemFromNetwork("chest", seed, 64)
  10446.         else
  10447.             while T:suck(direction, 1) do end -- remove saplings/ seeds
  10448.             T:dropItem("sapling", direction, 0) -- drop all saplings
  10449.             T:dropItem("seeds", direction, 64)  -- drop all but 64 of any seed type
  10450.             seedCount = T:getItemCount(seed)   
  10451.         end
  10452.         if seedCount > 0 then
  10453.             T:saveToLog("planting " ..seed)
  10454.         end
  10455.         return seed, seedCount  -- could be: "", 0 or "minecraft:wheat_seeds", 64
  10456.     end
  10457.        
  10458.     function lib.getVeg(R, direction, crop)
  10459.         -- assume only one type of crop per field
  10460.         -- local item, itemName, shortName, slot = "", "","", 0
  10461.         local inventorySlot, cropCount = 0, 0
  10462.        
  10463.         if R.networkFarm then
  10464.             inventorySlot, cropCount = network.getItemFromNetwork("chest", crop, 64)
  10465.         else
  10466.             if not R.mysticalAgriculture then
  10467.                 print("Collecting root crops from storage")
  10468.                 direction = direction or "forward"
  10469.                 while T:suck(direction, 1) do end           -- remove carrots / potatoes / wheat / beetroot / essence
  10470.                 --item, itemName, shortName, slot = T:getFirstItem()
  10471.                 local found = false
  10472.                 if T:getItemSlot("potato") > 0 then
  10473.                     T:dropItem("potato", direction, 64)     -- drop all but 65 potato
  10474.                     found = true
  10475.                     crop = "minecraft:potato"
  10476.                 end
  10477.                 if T:getItemSlot("carrot") > 0 then
  10478.                     T:dropItem("carrot", direction, 64)     -- drop all but 64 carrots
  10479.                     found = true
  10480.                     crop = "minecraft:carrot"
  10481.                 end
  10482.                 if not found then
  10483.                     T:dropAll(direction)                    -- drop everything back
  10484.                 end
  10485.             end
  10486.             cropCount = T:getItemCount(crop)
  10487.         end
  10488.        
  10489.         if cropCount > 0 then
  10490.             print("planting " ..crop)
  10491.         end
  10492.         return crop, cropCount -- could be: "", 0 or "minecraft:potato", 64
  10493.     end
  10494.    
  10495.     function lib.goHome(R)
  10496.         -- after a re-boot go to start
  10497.         local onTree = false
  10498.         -- is tree above or in front
  10499.         -- check if log in front
  10500.         if T:getBlockType("forward"):find("log") ~= nil then -- about to harvest tree
  10501.             lib.harvestTree(R, "forward")       -- will only harvest if fuel below half of limit
  10502.             onTree = true                   -- end on dirt
  10503.         elseif T:getBlockType("up"):find("log") ~= nil then -- harvesting tree
  10504.             lib.harvestTree(R, "up")            -- end on dirt
  10505.             onTree = true
  10506.         elseif T:getItemSlot("log") > 0 then-- tree just harvested
  10507.             onTree = true
  10508.             turtle.up()
  10509.             if T:getBlockType("up"):find("log") ~= nil then -- harvesting tree
  10510.                 lib.harvestTree(R, "up")        -- end on dirt
  10511.             else
  10512.                 while turtle.down() do end  -- end on dirt
  10513.             end
  10514.         end
  10515.        
  10516.         if onTree then
  10517.             local success, storage = false, ""
  10518.             -- tree harvested, sitting on dirt, but direction unknown
  10519.             T:down(1)               -- dig dirt, go down
  10520.             -- if storage type farm chest/barrel will give position
  10521.             for turns = 1, 4 do
  10522.                 T:turnRight(1)
  10523.                 success, storage = utils.isStorage("forward")
  10524.                 if success then
  10525.                     T:go("L2U1")
  10526.                     T:place("dirt", "down")
  10527.                     T:go("B1R1 F1D1R1")
  10528.                     R.ready = true
  10529.                     return R
  10530.                 end
  10531.             end
  10532.             -- if networked will be in air 2 blocks above barrel or stone
  10533.             T:down(1)
  10534.             T:place("dirt", "up")
  10535.             T:down(1)    --should be on corner
  10536.             for turns = 1, 4 do
  10537.                 T:forward(1)
  10538.                 success, storage = utils.isStorage("down")
  10539.                 if storage == "modem" then
  10540.                     T:go("R1F1D1")
  10541.                     R.ready = true
  10542.                     return R
  10543.                 end
  10544.                 T:go("B1R1")
  10545.             end
  10546.         else
  10547.             while turtle.down() do end -- no tree around, no logs onboard
  10548.         end
  10549.            
  10550.         return R    -- R.ready == false
  10551.     end
  10552.    
  10553.     function lib.goToLeft(R, plotCountR)
  10554.         T:go("U1R2 F"..plotCountR * 11 .."D1R2")    -- return home and continue with front
  10555.         if R.config ~= nil then
  10556.             local coord = R.config.currentPlot
  10557.             for i = 1, plotCountR do
  10558.                 coord = lib.configUpdateCoords(R, coord, "left")
  10559.             end
  10560.             R.config.currentPlot = coord
  10561.         end
  10562.     end
  10563.    
  10564.     function lib.goToFront(R, plotCountF)
  10565.         T:go("U1R1F"..plotCountF * 11 .."D1L1")
  10566.         if R.config ~= nil then
  10567.             local coord = R.config.currentPlot
  10568.             for i = 1, plotCountF do
  10569.                 coord = lib.configUpdateCoords(R, coord, "back")
  10570.             end
  10571.             R.config.currentPlot = coord
  10572.         end
  10573.     end
  10574.    
  10575.     function lib.gotoTree(R)
  10576.         -- assume facing crops
  10577.         if not R.networkFarm then
  10578.             T:turnRight(1)                          -- face storage
  10579.         end
  10580.         T:saveToLog("Extracting saplings from storage")
  10581.         lib.getSaplings(R, "forward")               -- gets one sapling only (if present)
  10582.         T:saveToLog("Moving to tree position")
  10583.         if R.networkFarm then
  10584.             T:turnRight(1)                          -- face wall next to corner barrel
  10585.         end
  10586.         T:go("U1F1R1")                              -- move on top of wall/storage. face tree direction
  10587.  
  10588.         -- check if tree or dirt ahead
  10589.         if turtle.forward() then                    -- No tree or sapling ahead. assume tree is missing or 2 blocks above
  10590.             if T:getBlockType("down") == "minecraft:dirt" then
  10591.                 turtle.back()                       -- older design, return
  10592.             else
  10593.                 T:up(1)
  10594.                 if turtle.detectUp() then           -- dirt above = new tree layout
  10595.                     if T:getBlockType("up") == "minecraft:dirt" then
  10596.                         T:go("B1U2")                -- check tree in elevated position
  10597.                     else
  10598.                         turtle.back()               -- not dirt above ? removed or lost
  10599.                     end
  10600.                 end
  10601.             end
  10602.         end
  10603.         lib.harvestTree(R, "forward")               -- fell tree or plant sapling, ends facing tree / dirt / sapling. sticks already used for fuel. excess saplings placed
  10604.         T:go("R1F1D1R1")                            -- return to base, facing crops
  10605.         if R.networkFarm then
  10606.             network.sendItemToNetworkStorage(R, "barrel", "sapling", 64) --move saplings to any attached barrel
  10607.         else
  10608.             T:turnRight(1)                          -- facing plot storage
  10609.             T:saveToLog("Dropping saplings into storage")
  10610.             T:dropItem("sapling", "forward", 0) -- drop all saplings forward into chest/barrel
  10611.             T:turnLeft(1)
  10612.         end
  10613.         -- ends in field facing crops
  10614.     end
  10615.    
  10616.     function lib.harvest(R, seed, crop)
  10617.         --[[
  10618.         cover the field in a set pattern.
  10619.         harvest crop if ripe
  10620.         till soil and plant new ones
  10621.         place seeds / harvest in chests
  10622.         return farm(s) to right / front
  10623.         crop is full name. if mixed seeds only one type returned
  10624.         ]]
  10625.         T:go("U1") --ready to farm field
  10626.         local isFarmToRight = false
  10627.         local isFarmToFront = false
  10628.         local width = 9
  10629.         local length = 10
  10630.         local toRight = true
  10631.         for l = 1, length do
  10632.             for w = 1, width do
  10633.                 lib.replant(R, seed, crop)  -- check and replant crop below
  10634.                 T:forward(1)
  10635.                 if l == 1 and w == width then -- last block of first row at front of farm
  10636.                     T:forward(1)
  10637.                     if utils.isStorage("down") then -- chest, barrel or modem
  10638.                         isFarmToRight = true
  10639.                     end
  10640.                     turtle.back()
  10641.                 end
  10642.             end
  10643.             -- end of the row: change direction
  10644.             if l < length then -- do not turn at final row
  10645.                 lib.replant(R, seed, crop)  -- check and replant crop below
  10646.                 if toRight then
  10647.                     T:go("L1F1L1")
  10648.                 else
  10649.                     T:go("R1F1R1")
  10650.                 end
  10651.                 lib.replant(R, seed, crop)
  10652.             end
  10653.             toRight = not toRight
  10654.         end
  10655.         T:go("R1F1") -- goes over chest/cobble on top wall
  10656.         if utils.isStorage("down") then
  10657.             isFarmToFront = true
  10658.         end
  10659.         T:go("R2F"..length.."D1L1") -- go straight across to seed chest 10 blocks, facing crops
  10660.         lib.storeCrops(R) -- rotates from start to deposit seeds and crops, ends facing crops
  10661.         return R, isFarmToRight, isFarmToFront
  10662.     end
  10663.    
  10664.     function lib.harvestTree(R, direction)
  10665.         --[[
  10666.             start in front of / during tree harvest
  10667.             Check if sapling present
  10668.             Harvest tree if present, replant sapling
  10669.             Dispose of apples. Use sticks as fuel
  10670.             Return to top of storage
  10671.         ]]
  10672.  
  10673.         direction = direction or "forward"
  10674.         local inFront = T:getBlockType("forward")
  10675.         print("Checking tree")
  10676.         if inFront == "" then -- no tree or sapling
  10677.             print("No sapling: planting sapling")
  10678.             T:place("sapling", "forward")
  10679.         elseif inFront:find("log") ~= nil or direction == "up" then -- tree above or in front
  10680.             -- clsTurtle.harvestTree(self, extend, craftChest, direction)
  10681.             print("Harvesting tree")
  10682.             T:harvestTree(false, false, direction) --do not investigate side branches in case chunk unloaded
  10683.             T:back(1) -- face dirt
  10684.             -- place(self, blockType, direction, leaveExisting, signText)
  10685.             T:place("sapling", "forward")
  10686.         end
  10687.         print("Disposing of apples")
  10688.         T:dropItem("apple", "up", 0) -- drop any apples
  10689.         utils.useSticksAsFuel()
  10690.        
  10691.         while turtle.down() do end  -- ends facing tree, on top of plot storage
  10692.     end
  10693.    
  10694.     function lib.manageTree(R)
  10695.         --[[ starting position facing crops ]]
  10696.         local isReady, crop, seed, status, R = lib.isCropReady(R, "forward")
  10697.         if crop == "" and not R.auto then           -- only ask player if not running via startup
  10698.             seed, crop = lib.askPlayerForCrops()    -- "" or  specific seed type,
  10699.         end
  10700.         local logSlot = T:getItemSlot("stick")
  10701.        
  10702.         if logSlot > 0 then
  10703.             turtle.select(logSlot)
  10704.             turtle.refuel()
  10705.         end
  10706.         logSlot = T:getItemSlot("log")
  10707.         local needsFuel = false
  10708.    
  10709.         if turtle.getFuelLevel() < turtle.getFuelLimit() / 2 then
  10710.             needsFuel = true
  10711.         end
  10712.         if needsFuel then
  10713.             T:saveToLog("Running Tree Manager")
  10714.             lib.gotoTree(R)                     -- check for sapling or harvest tree, retuns to facing crops
  10715.         end
  10716.        
  10717.         logSlot = T:getItemSlot("log")
  10718.         if logSlot > 0 then                     --logs onboard, need to equip crafting table
  10719.             if R.networkFarm then
  10720.                 lib.refuelWithLogs(R, logSlot)  -- use any logs for fuel
  10721.             else
  10722.                 T:go("L1F1")                    -- move to buried storage chest/barrel
  10723.                 lib.refuelWithLogs(R, logSlot)  -- use any logs for fuel
  10724.                 T:go("R2F1")                    -- facing seed chest/barrel
  10725.             end
  10726.         else
  10727.             if not R.networkFarm then
  10728.                 T:turnRight(1)
  10729.             end
  10730.         end
  10731.         -- get seeds or veg based on what is growing
  10732.         if seed ~= "" then
  10733.             local seedType, seedCount = lib.getSeeds(R, "forward", seed)    -- table: get 95 of beetroot / wheat / mysticalagriculture seeds
  10734.             if seedCount == 0 then
  10735.                 T:saveToLog("No seeds available.")
  10736.             end
  10737.         else    -- seed  = ""
  10738.             local veg, vegCount = "", 0
  10739.             if R.networkFarm then
  10740.                 veg, vegCount = lib.getVeg(R, "forward", crop)  -- gets any carrots / potatoes
  10741.             else
  10742.                 T:turnRight(1)                  -- face crop chests
  10743.                 veg, vegCount = lib.getVeg(R, "forward", crop)  -- gets any carrots / potatoes
  10744.                 T:turnRight(2)                  -- face crops
  10745.             end
  10746.             if veg ~= "" then
  10747.                 crop = veg
  10748.             end
  10749.         end
  10750.        
  10751.         return seed, crop   -- name of seed / "",  crop / ""
  10752.     end
  10753.    
  10754.     function lib.isCropReady(R, direction)
  10755.         local isReady = false
  10756.         local status = ""
  10757.         local success = false
  10758.         local crop, seed = "", ""
  10759.         local data = {}
  10760.  
  10761.         direction = direction or "forward"
  10762.  
  10763.         if direction == "down" then
  10764.             success, data = turtle.inspectDown()
  10765.         else
  10766.             success, data = turtle.inspect()
  10767.         end
  10768.         --T:saveToLog("lib.isCropReady(R,".. direction..")")
  10769.         if success then         -- inspect() success
  10770.             crop = data.name    -- name of growing plant
  10771.             if crop:find("flower") ~= nil then
  10772.                 T:saveToLog("Flower "..crop.." found")
  10773.                 return true, crop, "open"   -- eg torchflower
  10774.             else
  10775.                 if data.name:find("beetroot") ~= nil then
  10776.                     status = data.state.age.." / 3"
  10777.                     if data.state.age == 3 then
  10778.                         isReady = true
  10779.                     end
  10780.                 else            -- all other crops inc Mystical Agriculture
  10781.                     status = data.state.age.." / 7"
  10782.                     if data.state.age == 7 then
  10783.                         isReady = true
  10784.                     end
  10785.                 end
  10786.             end
  10787.             seed, crop = lib.getCropSeed(crop)
  10788.         end
  10789.         --T:saveToLog("return isReady = "..tostring(isReady)..", crop = "..crop..", seed = "..seed..", status = "..status)
  10790.         -- crop: "", "minecraft:carrots", "minecraft:beetroot", "minecraft:potatoes", "minecraft:wheat", "mysticalagriculture:*_crop"
  10791.         return isReady, crop, seed, status, R   -- eg true, "minecraft:carrots", "7 / 7" or false, "mysticalagriculture:inferium_crop", "1 / 7"
  10792.     end
  10793.    
  10794.     function lib.locateFarm(R)
  10795.         T:saveToLog("? over water = false")
  10796.         for i = 1, 4 do
  10797.             if turtle.detect() then
  10798.                 detected = i
  10799.             end
  10800.             T:turnRight(1)
  10801.         end
  10802.         T:saveToLog("Neighbouring blocks detected at: "..detected)
  10803.         -- check if on corner
  10804.         if detected > 0 then
  10805.             --assume tree / sapling on corner on older farm type
  10806.             T:saveToLog("Assuming next to tree / sapling. Moving..")
  10807.             T:go("R"..detected .."F1D1R1")
  10808.         else    -- no surrounding blocks
  10809.             for i = 1, 4 do
  10810.                 T:forward(1)
  10811.                 success, storage = utils.isStorage("down")
  10812.                 T:saveToLog("i = "..i..",success = "..tostring(success)..". storage = "..storage)
  10813.                 if success then
  10814.                     discovered = discovered .. storage
  10815.                     if storage == "modem" then
  10816.                         R.networkFarm = true
  10817.                     end
  10818.                 else
  10819.                     if storage == "" then
  10820.                         discovered = discovered .. "_"
  10821.                     else
  10822.                         discovered = discovered .. "wall"
  10823.                     end
  10824.                 end
  10825.                 utils.goBack(1)
  10826.                 T:turnRight(1)
  10827.             end
  10828.             T:saveToLog("Neighbouring blocks found:"..discovered)
  10829.         end
  10830.         -- check discovered for patterns eg {_wall_barrel}
  10831.         -- |W|*|B| * = turtle on wall
  10832.         -- | | |M|
  10833.         if R.networkFarm then
  10834.             -- E = wall_barrel_, N = _barrel_wall, W = wall_barrel_, S = _wall_barrel
  10835.             if discovered == "_wall_barrel" then
  10836.                 T:go("F1D1R1")
  10837.             elseif discovered == "barrel_wall_" then
  10838.                 T:go("R1F1D1R1")
  10839.             elseif discovered == "_barrel_wall" then
  10840.                 T:go("R2F1D1R1")
  10841.             elseif discovered == "wall_barrel_" then
  10842.                 T:go("L1F1D1R1")
  10843.             end
  10844.         else    -- normal storage farm
  10845.             if discovered == "_chest_chest" then
  10846.                 T:go("F1D1R1")
  10847.             elseif discovered == "chest__chest" then
  10848.                 T:go("R1F1D1R1")
  10849.             elseif discovered == "_chest_chest" then
  10850.                 T:go("R2F1D1R1")
  10851.             elseif discovered == "chest_chest_" then
  10852.                 T:go("L1F1D1R1")
  10853.             end
  10854.         end
  10855.         blockType = T:getBlockType("down")
  10856.         if blockType:find("water") ~= nil then
  10857.             for i = 1, 4 do
  10858.                 success, storage = utils.isStorage("forward")   -- true/false, chest, barrel, modem / ""
  10859.                 if success and storage == "modem" then
  10860.                     R.networkFarm = true
  10861.                 end
  10862.                 T:turnRight(1)
  10863.             end
  10864.             R.ready = true
  10865.         end
  10866.     end
  10867.    
  10868.     function lib.plantCrop(R, seed, crop, direction)
  10869.         --turtle.digDown("left") -- harvest existing
  10870.         --turtle.digDown("right") -- till soil
  10871.         if crop:find("potato") ~= nil then
  10872.             T:dropItem("poison", "up", 0)
  10873.         end
  10874.         --T:saveToLog("lib.plantCrop(R, seed = "..seed..", crop = "..crop)
  10875.         local success = false
  10876.         -- place(blockType, direction, leaveExisting, signText, doNotAttack)
  10877.         if seed == "" then  -- must be a crop
  10878.             success = T:place(crop, direction, true, "", true)
  10879.             --success = T:place(crop, direction, false, "", true)
  10880.             --T:saveToLog("Seed = "..seed..". Placing "..crop..": success = "..tostring(success))
  10881.         else
  10882.             success = T:place(seed, direction, true, "", true)  -- eg "mysticalagriculture:air_seeds"
  10883.             --success = T:place(seed, direction, false, "", true)   -- eg "mysticalagriculture:air_seeds"
  10884.             --T:saveToLog("Placing "..seed..": success = "..tostring(success))
  10885.         end
  10886.         if not success then
  10887.             success = T:place("seed", direction, true, "", true)            -- failsafe
  10888.             --success = T:place("seed", direction, false, "", true)         -- failsafe
  10889.             --T:saveToLog("Placing generic 'seeds' : success = "..tostring(success))
  10890.         end
  10891.         return success
  10892.     end
  10893.        
  10894.     function lib.replant(R, seed, crop)
  10895.         -- check crop below. If ripe, dig and replant seed
  10896.         local isReady, cropType, seedType, status
  10897.         isReady, cropType, seedType, status, R = lib.isCropReady(R, "down") -- eg true, "minecraft:carrots", "7 / 7" or false, "", ""
  10898.         if cropType == "" then                  -- no crop below (above water, storage or dirt)
  10899.             turtle.digDown("right")             -- use hoe
  10900.             lib.plantCrop(R, seed, crop, "down")-- plant crop
  10901.         elseif isReady then                     -- crop below is ready
  10902.             turtle.digDown("left")              -- use pickaxe
  10903.             lib.plantCrop(R, seedType, cropType, "down")    -- plant crop
  10904.         end
  10905.  
  10906.         return R
  10907.     end
  10908.    
  10909.     function lib.refuelWithLogs(R, logSlot)
  10910.         -- saplings already dropped, apples dumped, sticks used as fuel
  10911.         -- assume positioned in front of crops if networked, or over buried storage
  10912.         -- earlier versions used crafting table buried in the ground
  10913.         -- newer versions have a barrel or chest embedded in the ground, containing the crafting table
  10914.         -- networked version uses remote storage, turtle faces crops and has modem at back
  10915.         local slot, count = 0, 0
  10916.         T:saveToLog("Sorting Inventory")    -- make sure all items collected together
  10917.         T:sortInventory(true)
  10918.         T:saveToLog("Logs present: collecting crafting table")
  10919.         if R.networkFarm then
  10920.             slot, count = network.getItemFromNetwork("barrel", "minecraft:crafting_table", 1)
  10921.             --slot = lib.getItemFromNetworkBarrels(R, "minecraft:crafting_table", 1)
  10922.         else
  10923.             while T:suck("down") do end                 -- empty out barrel/chest to find crafting table
  10924.         end
  10925.         if T:getItemSlot("crafting") == 0 then
  10926.             T:checkInventoryForItem({"crafting"}, {1}, true, "Crafting table required for logs->planks")
  10927.         end
  10928.         if T:equip("right", "minecraft:crafting_table") then -- swap equipment on right side
  10929.             if R.networkFarm then
  10930.                 network.sendItemToNetworkStorage(R, "barrel", "minecraft:diamond_hoe", 1)
  10931.             end
  10932.             for i = 1, 16 do                        -- drop anything except logs down into barrel/chest/pit
  10933.                 if T:getSlotContains(i):find("log") == nil then
  10934.                     T:drop("down", i)   -- into water if networked, buried chest otherwise
  10935.                 else
  10936.                     logSlot = i
  10937.                 end
  10938.             end
  10939.             turtle.select(logSlot)
  10940.             turtle.transferTo(1)
  10941.             turtle.craft()                          -- craft logs to planks
  10942.             logSlot = T:getItemSlot("planks")
  10943.             while logSlot > 0 and turtle.getFuelLevel() < turtle.getFuelLimit() do
  10944.                 turtle.select(logSlot)
  10945.                 turtle.refuel()                     -- refuel using planks
  10946.                 logSlot = T:getItemSlot("planks")              
  10947.             end
  10948.             while T:suck("down") do end         -- recover items from water / storage below
  10949.             --T:unequip("right")
  10950.             if R.networkFarm then
  10951.                 slot = network.getItemFromNetwork("barrel", "minecraft:diamond_hoe", 1)
  10952.             end
  10953.             if not T:equip("right", "minecraft:diamond_hoe") then       -- re-equip hoe/remove crafting table
  10954.                 T:equip("right", "minecraft:diamond_pickaxe")          
  10955.             end
  10956.             if R.networkFarm then
  10957.                 network.sendItemToNetworkStorage(R, "barrel", "minecraft:crafting_table", 1)
  10958.             else
  10959.                 T:dropItem("minecraft:crafting_table", "down")
  10960.             end
  10961.         else
  10962.             print("Unable to equip crafting table.\n\nCheck turtle inventory and chest or barrel below")
  10963.             error()
  10964.         end
  10965.     end
  10966.                
  10967.     function lib.storeCrops(R)
  10968.         -- place crops and seeds into chests. starts facing crops
  10969.         T:dropItem("apple", "up", 0) -- drop all apples
  10970.         T:dropItem("poison", "up", 0) -- drop all poison potatoes
  10971.         if R.networkFarm then
  10972.             network.emptyInventory(R, {"sapling", "diamond_hoe", "crafting"}, {"all"}, true)
  10973.         else
  10974.             T:turnRight(1)  -- face seed storage
  10975.             lib.storeSeeds(R, "forward")
  10976.             T:turnRight(1)
  10977.             if utils.isStorage(direction) then
  10978.                 T:dropAll(direction) -- drops everything including essences
  10979.             end
  10980.             T:turnRight(2) -- facing crops again
  10981.         end
  10982.     end
  10983.        
  10984.     function lib.storeSeeds(R, direction)
  10985.         direction = direction or "forward"
  10986.         if utils.isStorage(direction) then -- chest exists
  10987.             if not R.networkFarm then
  10988.                 if T:getItemSlot("minecraft:wheat_seeds") > 0 then
  10989.                     if not T:dropItem("minecraft:wheat_seeds", direction, 0) then
  10990.                         T:dropItem("minecraft:wheat_seeds", "up", 0)
  10991.                     end -- drop all wheat seeds
  10992.                 elseif T:getItemSlot("minecraft:beetroot_seeds") > 0 then
  10993.                     if not T:dropItem("minecraft:beetroot_seeds", direction, 0) then-- drop all beetroot seeds
  10994.                         T:dropItem("minecraft:beetroot_seeds", "up", 0)
  10995.                     end
  10996.                 elseif T:getItemSlot("seeds") > 0 then
  10997.                     if not T:dropItem("seeds", direction, 0) then   -- drop all other seeds as chest is full
  10998.                         T:dropItem("seeds", "up", 0)
  10999.                         -- or could print a message and wait for player to empty storage
  11000.                     end
  11001.                 end
  11002.             end
  11003.         end
  11004.     end
  11005.    
  11006.     function lib.watchFarm(R)
  11007.         --[[
  11008.         check status of crops in front of turtle.
  11009.         call lib.harvest when ripe
  11010.         return farm(s) found in front or to the right
  11011.         ]]
  11012.         local isReady, crop, seed, status   -- bool, string, string, integer
  11013.         isReady, crop, seed, status, R = lib.isCropReady(R, "forward")      -- eg true, "minecraft:carrots", "7 / 7" or false, "", ""
  11014.         -- check state of crop in front. Harvest if ripe       
  11015.         repeat
  11016.             if not isReady then
  11017.                 if crop == "" then
  11018.                     print("No crops found in front")
  11019.                     print("Plant seeds, carrots, potatoes")
  11020.                     error()
  11021.                 else
  11022.                     print("Waiting for "..crop.." status: "..status)
  11023.                     if crop:find("mysticalagriculture") ~= nil then
  11024.                         R.mysticalAgriculture = true
  11025.                     end
  11026.                 end
  11027.                 sleep(60)
  11028.                 isReady, crop, seed, status, R = lib.isCropReady(R, "forward")          -- eg true, "minecraft:carrots", "7 / 7" or false, "", ""
  11029.             end
  11030.         until isReady
  11031.         T:saveToLog("Local crops ripe", true)
  11032.         seed, crop = lib.manageTree(R) -- "", "" or name of seed, crop
  11033.         return seed, crop
  11034.     end
  11035.    
  11036.     --[[
  11037.         ENTRY POINT**********************************************
  11038.         called from args on start, or from user choice
  11039.         farm already built, needs planting and/or harvesting
  11040.         needs both pickaxe and hoe
  11041.         may start in any position if chunk unloaded while running
  11042.     ]]
  11043.    
  11044.    
  11045.     --T:setUseLog(true, "farmLog.txt", true)    -- T:setUseLog(use, filename, delete)
  11046.     --dbug = true                               -- set dbug flag
  11047.     --if not R.auto then                        -- not running from startup.lua
  11048.         --utils.waitForInput("Logging and debugging enabled")   --utils.waitForInput(message)
  11049.     --end
  11050.    
  11051.     R = utils.checkFarmPosition(R)  -- should be facing crops, placed above water source. R.ready, R.networkFarm is true/false
  11052.     if R.networkFarm then
  11053.         local message = network.loadStorageLists()  -- initialises or creates lists of where an item can be found: GLOBAL LISTS!
  11054.         if message ~= nil then return {message} end
  11055.     end
  11056.     if not R.ready then     -- not in correct starting place
  11057.         R = lib.goHome(R)
  11058.         if not R.ready then -- try to find home
  11059.             return
  11060.             {
  11061.                 "Unable to determine my position.\n",
  11062.                 "Place me in the lower left corner",
  11063.                 "over water, facing the crops with",
  11064.                 "barrel or chest to my right and behind",
  11065.                 "(or modem behind if networked farm)"
  11066.             }
  11067.         end
  11068.     end
  11069.     if not T:isEmpty() then     -- items in turtle inventory
  11070.         local logSlot = T:getItemSlot("log")
  11071.         if logSlot > 0 then
  11072.             if R.networkFarm then
  11073.                 lib.refuelWithLogs(R, logSlot)  -- use any logs for fuel
  11074.             else
  11075.                 T:go("L1F1")                    -- move to buried storage chest/barrel
  11076.                 lib.refuelWithLogs(R, logSlot)  -- use any logs for fuel
  11077.                 T:go("R2F1")                    -- facing seed chest/barrel
  11078.             end
  11079.         end
  11080.         if not T:isEmpty() then                 -- still items in turtle inventory
  11081.             lib.storeCrops(R)
  11082.         end
  11083.     end
  11084.     -- in correct position. Check equipment first, harvest tree, re-equip then harvest crops
  11085.     local hoeSlot = lib.checkEquipment(R, "minecraft:diamond_hoe")  -- returns slot  or 0
  11086.    
  11087.     if hoeSlot == 0 then        -- hoe not in inventory
  11088.         local message
  11089.         hoeSlot, message = lib.getHoe()
  11090.         if message ~= "" then
  11091.             return {message}
  11092.         end
  11093.     end
  11094.     -- hoe present or user cancelled request for hoe. Turtle is ready to exchange items
  11095.     if hoeSlot > 0 then
  11096.         T:equip("right", "minecraft:diamond_hoe", 0) -- equip hoe and put crafting chest into barrel
  11097.         if R.networkFarm then   -- drop crafting table into storage
  11098.             network.sendItemToNetworkStorage(R, "barrel", "minecraft:crafting_table", 1)
  11099.         else    -- put crafting table into barrel in floor
  11100.             T:dropItem("crafting", "down")
  11101.             T:go("B1R1")
  11102.         end
  11103.     else    -- hoe not present: return to start and exit
  11104.         if not R.networkFarm then
  11105.             T:go("B1R1")
  11106.         end
  11107.         return {"Unable to equip hoe."}
  11108.     end
  11109.     -- check if crops already planted
  11110.    
  11111.     local isFarmToRight, isFarmToFront = false, false
  11112.     local isReady, crop, seed, status
  11113.     isReady, crop, seed, status, R = lib.isCropReady(R, "forward")
  11114.     local watch = true  -- assume watching farm already planted
  11115.     local init = false
  11116.     if crop == "" then  -- nothing has been planted
  11117.         seed, crop = lib.askPlayerForCrops()
  11118.         if crop ~= "" or seed  ~= "" then   -- something has been chosen
  11119.             T:saveToLog("Initial planting of "..crop, true)
  11120.             R, isFarmToRight, isFarmToFront = lib.harvest(R, seed, crop)    -- harvest plot a1 plots to right / front recorded 
  11121.             init = true
  11122.         else
  11123.             watch = false   -- not watching, continue with planting
  11124.         end
  11125.     end
  11126.     while true do -- start infinite loop of watching crops, farming all modules
  11127.         if watch and not init then -- init true when farm first planted
  11128.             seed, crop = lib.watchFarm(R) -- waits if required, returns seed / crop
  11129.             init = false
  11130.         end
  11131.         T:saveToLog("Beginning "..crop.. " management", true)
  11132.         R, isFarmToRight, isFarmToFront = lib.harvest(R, seed, crop)    -- harvest plot a1 plots to right / front recorded
  11133.         lib.farmAll(R, isFarmToRight, isFarmToFront)
  11134.         if not R.auto then -- not started from startup.lua
  11135.             if R.networkFarm then
  11136.                 T:unequip("right")  -- unequip hoe
  11137.                 network.getItemFromNetwork("barrel", "minecraft:crafting_table", 1)
  11138.                 network.sendItemToNetworkStorage(R, "barrel", "minecraft:diamond_hoe", 1)
  11139.                 T:equip("right", "minecraft:crafting_table", 0) -- equip crafting_table
  11140.             else
  11141.                 T:go("L1F1")
  11142.                 while T:suck("down") do end                     -- recover items from storagebelow
  11143.                 T:equip("right", "minecraft:crafting_table", 0) -- equip crafting_table
  11144.                 T:dropItem("minecraft:diamond_hoe", "down")                 -- drop hoe into storage
  11145.                 T:dropItem("crafting", "down")                  -- in case spare one picked by mistake
  11146.                 utils.goBack(1)
  11147.                 T:turnRight(1)
  11148.             end
  11149.             return {"Crop management of all modules completed"}
  11150.         end
  11151.     end
  11152.     return {}
  11153. end
  11154.  
  11155. local function manageFarmSetup(R) -- 33
  11156.     local lib = {}
  11157.    
  11158.     function lib.disableAutostart()
  11159.         if fs.exists("start.txt") then
  11160.             fs.delete("start.txt")
  11161.         end
  11162.         if fs.exists("startup.lua") then
  11163.             fs.delete("startup.lua")
  11164.         end
  11165.     end
  11166.    
  11167.     function lib.enableAutostart()
  11168.         if not fs.exists("startup.lua") then
  11169.             local h = fs.open("startup.lua", "w")
  11170.             h.writeLine('function main()')
  11171.             h.writeLine('   if fs.exists("start.txt") then')
  11172.             h.writeLine('       local handle = fs.open("start.txt", "r")')
  11173.             h.writeLine('       local cmd = handle.readLine()')
  11174.             h.writeLine('       handle.close()')
  11175.             h.writeLine('       shell.run("tk.lua "..cmd)')
  11176.             h.writeLine('   end')
  11177.             h.writeLine('end')
  11178.             h.writeLine('main()')
  11179.             h.close()
  11180.         end
  11181.         local h = fs.open("start.txt", "w")
  11182.         if R.data == "farm" then
  11183.             h.writeLine('farm')
  11184.         end
  11185.         h.close()
  11186.         print("Startup files written")
  11187.     end
  11188.    
  11189.     T:clear()
  11190.     --local pp = utils.getPrettyPrint()
  11191.     local choices = {"Plant or harvest this farm complex"}  -- 1.
  11192.     local isManaged = fs.exists("start.txt")
  11193.  
  11194.     if isManaged then
  11195.         table.insert(choices, "Disable automatic farm management") -- 2.
  11196.     else
  11197.         table.insert(choices, "Enable automatic farm management") -- 2.
  11198.     end
  11199.     pp.itemColours = {colors.lime, colors.lightGray}
  11200.     if not R.networkFarm then
  11201.         table.insert(choices, "Convert to Network Storage")     -- 3.
  11202.         table.insert(pp.itemColours, colors.magenta)
  11203.     end
  11204.     if R.mysticalAgriculture then
  11205.         --table.insert(choices, "Convert to Mystical Agriculture")
  11206.         table.insert(choices, "Upgrade Mystical Agriculture soil")  --3/4
  11207.         table.insert(pp.itemColours, colors.green)
  11208.     end
  11209.    
  11210.     local userChoice, modifier = menu.menu("Choose your option", choices, pp) -- 1 to 2
  11211.     if modifier == "q" then -- quit chosen
  11212.         return {"Player has quit"}
  11213.     end
  11214.     R.subChoice = userChoice
  11215.     if R.subChoice == 1 then -- harvest now
  11216.         R.silent = false
  11217.         R.data = "farm"
  11218.         R.auto = false
  11219.         return manageFarm(R)
  11220.     elseif R.subChoice == 2 then -- enable/disable auto farm
  11221.         local line = menu.clear()
  11222.         if isManaged then
  11223.             local message = ( "This turtle has been configured to"..
  11224.                               "start automatically and run the farm"..
  11225.                               "management program.\n")
  11226.             line = menu.colourText(line, message, true, true)
  11227.             if menu.getBoolean("Do you want to disable this? (y/n)", line, colors.yellow, colors.black) then
  11228.                 lib.disableAutostart()
  11229.             end
  11230.             return {"Autostart disabled. Reboot to activate"}
  11231.         else -- not managed
  11232.             local message = ( "~yellow~This turtle can be configured to be\n"..
  11233.                               "a dedicated farm manager.\n\n"..
  11234.                               "~lightGray~It will then start automatically and\n"..
  11235.                               "monitor the farm complex:\n\n"..
  11236.                               "~green~harvesting~yellow~ and ~lime~replanting ~yellow~continuously.\n")
  11237.             line = menu.colourText(line, message, true, true)
  11238.             if menu.getBoolean("Do you want to enable this? (y/n)", line + 2, colors.orange, colors.black) then
  11239.                 lib.enableAutostart()
  11240.             else
  11241.                 return {"Player cancelled operation"}
  11242.             end
  11243.             return {"Autostart enabled. Reboot to activate"}
  11244.         end
  11245.     --elseif R.subChoice == 3 and #choices == 4 then -- convert normal farm to network storage
  11246.     elseif R.subChoice == 3 then -- convert normal farm to network storage
  11247.         local isMain = false
  11248.         local line = menu.clear()
  11249.         local message = ("~yellow~You have chosen to convert this farm "..
  11250.                          "to ~magenta~network storage ~yellow~with modems."..
  11251.                          "All ~brown~chests and barrels ~red~will be removed.\n")
  11252.         line = menu.colourText(line, message, true, true)
  11253.         if menu.getBoolean("Is this the main or only plot? (y/n)", line + 3, colors.orange, colors.black) then
  11254.             isMain = true
  11255.         end
  11256.         T:checkInventoryForItem({"stone"}, {16})
  11257.         T:checkInventoryForItem({"dirt"}, {2})
  11258.         T:checkInventoryForItem({"sapling"}, {1})
  11259.         T:checkInventoryForItem({"barrel"}, {1})
  11260.         T:checkInventoryForItem({"ladder"}, {5})
  11261.         if isMain then
  11262.             T:checkInventoryForItem({"chest"}, {8})
  11263.             T:checkInventoryForItem({"wired_modem_full"}, {3})
  11264.             T:checkInventoryForItem({"computercraft:cable"}, {70})
  11265.             return createFarm(R, "convertWithStorage")
  11266.         else
  11267.             T:checkInventoryForItem({"wired_modem_full"}, {2})
  11268.             T:checkInventoryForItem({"computercraft:cable"}, {57})
  11269.             return createFarm(R, "convert")
  11270.         end
  11271.     --elseif R.subChoice == 3 and #choices == 3 then -- upgrade farmland with essence (network opt not present)
  11272.     --elseif R.subChoice == 3 and #choices == 3 then -- upgrade farmland with essence (network opt not present)
  11273.         --return upgradeFarmland(R)
  11274.     elseif R.subChoice == 4 then -- upgrade farmland with essence
  11275.         return upgradeFarmland(R)
  11276.     end
  11277.    
  11278.     return {}
  11279. end
  11280.  
  11281. local function measure(R) -- 101
  11282.     -- measure height/ depth / length
  11283.     local lib = {}
  11284.    
  11285.     function lib.checkBlocks(R, blocks)
  11286.         local dimension = "height"
  11287.         local message = ""
  11288.         local measure = ""
  11289.         local doContinue = true
  11290.         if R.choice == 102 then
  11291.             dimension = "depth"
  11292.         elseif R.choice == 103 then
  11293.             dimension = "length"
  11294.         end
  11295.         blocks = blocks + 1
  11296.         if blocks > R.size then
  11297.             message = "Max "..dimension.." of "..R.size.." stopped measurement"
  11298.             measure = ""
  11299.             doContinue = false
  11300.         else
  11301.             measure = dimension.." measured: "..blocks.." blocks"
  11302.         end
  11303.         return doContinue, blocks, measure, message
  11304.     end
  11305.    
  11306.     local blocks = 1
  11307.     local method = ""
  11308.     local measure = ""
  11309.     local message = ""
  11310.     local doContinue = true
  11311.     if R.choice == 101 then             -- height
  11312.         if R.subChoice == 1 then        -- obstruction above
  11313.             method = "Method: Until obstruction above"
  11314.             while turtle.up() and doContinue do
  11315.                 doContinue, blocks, measure, message = lib.checkBlocks(R, blocks)
  11316.             end
  11317.         elseif R.subChoice == 2 then    -- end of blocks in front of turtle eg cliff, wall
  11318.             method = "Method: Until no block detected in front"
  11319.             while turtle.detect() and doContinue do
  11320.                 if turtle.up() then
  11321.                     doContinue, blocks, measure, message = lib.checkBlocks(R, blocks)
  11322.                 else
  11323.                     message = "Obstruction above stopped measurement"
  11324.                     measure = ""
  11325.                     doContinue = false
  11326.                 end
  11327.             end
  11328.         elseif R.subChoice == 3 then    -- search for specific block min 3 characters
  11329.             method = "Method:Until search: '"..R.data.."' met"
  11330.             while turtle.detect() and doContinue do
  11331.                 if turtle.up() then
  11332.                     doContinue, blocks, measure, message = lib.checkBlocks(R, blocks)
  11333.                     if doContinue then
  11334.                         local blockType = T:getBlockType("forward")
  11335.                         if blockType:find(R.data) ~= nil then
  11336.                             measure = "Height measured: "..blocks.." blocks"
  11337.                             message = "Found "..blockType
  11338.                             doContinue = false
  11339.                         end
  11340.                     end
  11341.                 else
  11342.                     message = "Obstruction above stopped measurement"
  11343.                     measure = ""
  11344.                     doContinue = false
  11345.                 end
  11346.             end
  11347.         elseif R.subChoice == 4 then    -- When block above changes eg dragon tower height
  11348.             local blockType = T:getBlockType("up")
  11349.             local compare = blockType
  11350.             method = "Method: Until "..blockType.." changes"
  11351.             while blockType == compare and doContinue do
  11352.                 T:up(1)
  11353.                 doContinue, blocks, measure, message = lib.checkBlocks(R, blocks)
  11354.                 if doContinue then
  11355.                     blockType = T:getBlockType("up")
  11356.                 end
  11357.             end
  11358.             measure = "Height measured: "..blocks.." blocks"
  11359.             message = "Found "..blockType
  11360.         end
  11361.        
  11362.         for i = 1, blocks do
  11363.             turtle.down()
  11364.         end
  11365.     elseif R.choice == 102 then -- depth
  11366.         T:go("F1R2D1") -- go off the edge and face cliff/pit wall
  11367.         blocks = blocks + 1
  11368.         if R.subChoice == 1 then        -- obstruction water / lava below
  11369.             local move = true
  11370.             while move do
  11371.                 local blockType = T:getBlockType("down")
  11372.                 if blockType:find("water") ~= nil or blockType:find("lava") ~= nil then
  11373.                     message1 = blockType.." found at "..blocks
  11374.                     move = false
  11375.                 else
  11376.                     move = turtle.down()
  11377.                     if move then
  11378.                         blocks = blocks + 1
  11379.                     else
  11380.                         measure = "Depth measured: "..blocks.." blocks"
  11381.                     end
  11382.                 end
  11383.             end
  11384.             method = "Method: Until obstruction below"
  11385.         elseif R.subChoice == 2 then    -- end of wall in front`
  11386.             while turtle.detect() do
  11387.                 if turtle.down() then
  11388.                     blocks = blocks + 1
  11389.                     measure = "Depth measured: "..blocks.." blocks"
  11390.                 else
  11391.                     message1 = "Obstruction below stopped measurement"
  11392.                     break
  11393.                 end
  11394.             end
  11395.             method = "Method: Until no block detected ahead"
  11396.         elseif R.subChoice == 3 then    -- specific block detected ahead
  11397.             method = "Method:Until search: '"..R.data.."' met"
  11398.             while turtle.detect() do
  11399.                 if turtle.down() then
  11400.                     blocks = blocks + 1
  11401.                     local blockType = T:getBlockType("forward")
  11402.                     if blockType:find(R.data) ~= nil then
  11403.                         measure = "Depth measured: "..blocks.." blocks"
  11404.                         message = "Found "..blockType
  11405.                         break
  11406.                     end
  11407.                 else
  11408.                     message = "Obstruction below stopped measurement"
  11409.                     break
  11410.                 end
  11411.             end
  11412.         end
  11413.         for i = 1, blocks do
  11414.             turtle.up()
  11415.         end
  11416.         T:go("F1R2")
  11417.     elseif R.choice == 103 then -- length
  11418.         if R.subChoice == 1 then        -- obstruction ahead
  11419.             method = "Method: Until obstruction ahead"
  11420.             while turtle.forward() and doContinue  do
  11421.                 doContinue, blocks, measure, message = lib.checkBlocks(R, blocks)
  11422.             end
  11423.         elseif R.subChoice == 2 then    -- end of ceiling above
  11424.             method = "Method: Until no block detected above"
  11425.             while turtle.detectUp() and doContinue do
  11426.                 if turtle.forward() then
  11427.                     doContinue, blocks, measure, message = lib.checkBlocks(R, blocks)
  11428.                 else
  11429.                     message = "Obstruction ahead stopped measurement"
  11430.                     measure = ""
  11431.                     doContinue = false
  11432.                 end
  11433.             end
  11434.         elseif R.subChoice == 3 then    -- end of floor below
  11435.             method = "Method: Until no block detected below"
  11436.             while turtle.detectDown() and doContinue do
  11437.                 if turtle.forward() then
  11438.                     doContinue, blocks, measure, message = lib.checkBlocks(R, blocks)
  11439.                 else
  11440.                     message = "Obstruction ahead stopped measurement"
  11441.                     measure = ""
  11442.                     doContinue = false
  11443.                 end
  11444.             end
  11445.         elseif R.subChoice == 4 then    -- search for specific block up min 3 characters
  11446.             method = "Method:Until search: '"..R.data.."' above met"
  11447.             while turtle.detectUp() and doContinue do
  11448.                 if turtle.forward() then
  11449.                     doContinue, blocks, measure, message = lib.checkBlocks(R, blocks)
  11450.                     if doContinue then
  11451.                         local blockType = T:getBlockType("up")
  11452.                         if blockType:find(R.data) ~= nil then
  11453.                             message = "Found "..blockType
  11454.                             measure = "Length measured: "..blocks.." blocks"
  11455.                             doContinue = false
  11456.                         end
  11457.                     end
  11458.                 else
  11459.                     message = "Obstruction ahead stopped measurement"
  11460.                     measure = ""
  11461.                     doContinue = false
  11462.                 end
  11463.             end
  11464.         elseif R.subChoice == 5 then    -- search for specific block down min 3 characters
  11465.             method = "Method:Until search: '"..R.data.."' below met"
  11466.             --while turtle.detectDown() and doContinue do
  11467.             while doContinue do
  11468.                 if turtle.forward() then
  11469.                     doContinue, blocks, measure, message = lib.checkBlocks(R, blocks)
  11470.                     if doContinue then
  11471.                         local blockType = T:getBlockType("down")
  11472.                         if blockType:find(R.data) ~= nil then
  11473.                             message = "Found "..blockType
  11474.                             measure = "Length measured: "..blocks.." blocks"
  11475.                             doContinue = false
  11476.                         end
  11477.                     end
  11478.                 else
  11479.                     message = "Obstruction ahead stopped measurement"
  11480.                     measure = ""
  11481.                     doContinue = false
  11482.                 end
  11483.             end
  11484.         end
  11485.         T:turnRight(2)  -- head home
  11486.         for i = 1, blocks do
  11487.             turtle.forward()
  11488.         end
  11489.         T:turnRight(2)
  11490.     elseif R.choice == 104 then -- depth of stretch of water
  11491.         --R.length = 0 to auto calculate
  11492.         R.width = 1
  11493.         R.silent = true
  11494.         R.useBlockType = ""
  11495.         R.data = "clearWaterPlants"
  11496.         local data = clearWaterPlants(R)
  11497.         R.height = data[1]
  11498.         local length = data[2]
  11499.         T:go ("R2F"..length - 1 .."R2U1")
  11500.         return {"Greatest depth measured: ".. R.height,"Width of water: "..R.length}
  11501.     end
  11502.     if message == "" then
  11503.         return{method, measure}
  11504.     else
  11505.         return{method, measure, message}
  11506.     end
  11507. end
  11508.  
  11509. local function mineBedrockArea(R) -- 19
  11510.     --[[
  11511.     Assume on level 5 or -59
  11512.     for 1, width do
  11513.         for 1, length do
  11514.             go down until bedrock, digging/replacing all directions
  11515.             return to 5 / -59
  11516.             move forward 1 blocks
  11517.         end
  11518.         turn right/ forward 2 turn right
  11519.     end
  11520.     ]] 
  11521.     local lib = {}
  11522.    
  11523.     function lib.clearColumn(R)
  11524.         local level = 0
  11525.         --T:go("L1x1R2x1L1")
  11526.         local success = T:down(1)
  11527.         while success do
  11528.             level = level + 1
  11529.             if R.data == "leaveExposed" then
  11530.                 T:go("R1x1R1x1R1x1R1x1", false, 0, true)
  11531.             else
  11532.                 T:go("R1C1R1C1R1C1R1C1", false, 0, true)
  11533.             end
  11534.             success = T:down(1)
  11535.         end
  11536.         if R.data == "leaveExposed" then
  11537.             T:go("U"..level)
  11538.         else
  11539.             T:go("U"..level.."C2")
  11540.         end
  11541.     end
  11542.    
  11543.     local goRight = true
  11544.     for i = 1, R.width do
  11545.         for j = 1, R.length do
  11546.             lib.clearColumn(R)
  11547.             T:forward(1)
  11548.         end
  11549.         if goRight then
  11550.             T:go("R1F1R1")
  11551.         else
  11552.             T:go("L1F1L1")
  11553.         end
  11554.         goRight = not goRight
  11555.     end
  11556.     return {}
  11557. end
  11558.  
  11559. local function oceanMonumentColumns(R)
  11560.     -- utility to find corners and build columns to surface
  11561.     local lib = {}
  11562.    
  11563.     function lib.buildColumn()
  11564.         local depth = 0
  11565.         while T:isWater("forward") do
  11566.             T:go("U1C2")
  11567.             depth = depth + 1
  11568.         end
  11569.         return depth
  11570.     end
  11571.    
  11572.     function lib.buildTower()
  11573.         T:go("F1C2 F1C2 F1C2")
  11574.         T:go("L1F1 L1C2 F1C2 F1C2 F1C2")
  11575.         T:go("R1F1 R1C2 F1C2 F1C2 F1C2")
  11576.         T:go("L1F1 L1C2 F1C2 F1C2 F1C2")
  11577.        
  11578.         T:go("R2")
  11579.         for i = 1, 4 do
  11580.             T:go("U1C2")
  11581.             for j = 1, 4 do
  11582.                 T:go("F1C2 F1C2 F1C2 R1")
  11583.             end
  11584.         end
  11585.     end
  11586.    
  11587.     R.silent = true
  11588.     local blockType = T:getBlockType("down")
  11589.     while blockType:find("water") == nil do
  11590.         T:down(1) -- if on a platform will break through
  11591.         blockType = T:getBlockType("down")
  11592.     end
  11593.     --R.useBlockType = "prismarine", R.data = "oceanMonumentColumns" from getTask
  11594.     --local tempData = R.data
  11595.     --R.data = "clearWaterPlants"
  11596.     local result = clearWaterPlants(R)[1]
  11597.     if result ~= "" then
  11598.         return {result}
  11599.     else
  11600.         --on corner of monument, facing out to ocean
  11601.         local depth = lib.buildColumn()
  11602.         -- now above surface, block below at surface level
  11603.         for i = 1, 4 do
  11604.             T:turnRight(1)
  11605.             R.length = 57
  11606.             utils.createPath(R) -- roughly at next corner
  11607.             if i < 4 then
  11608.                 T:down(depth-2) -- roughly at correct depth
  11609.                 local waterBelow = utils.clearVegetation("down")
  11610.                 while waterBelow do
  11611.                     T:down(1)
  11612.                     waterBelow = utils.clearVegetation("down")
  11613.                 end
  11614.                 blockType = T:getBlockType("down")
  11615.                 while blockType:find("prismarine") ~= nil do
  11616.                     T:forward(1)
  11617.                     blockType = T:getBlockType("down")
  11618.                 end
  11619.                 turtle.back()
  11620.                 depth = lib.buildColumn()
  11621.             end
  11622.         end
  11623.         -- completed retaining paths. Build small tower for easy access
  11624.         lib.buildTower()
  11625.     end
  11626.    
  11627.     return {}
  11628. end
  11629.  
  11630. local function placeRedstoneTorch(R) -- 91, 92
  11631.     local moves = 2
  11632.     local blockType = T:getBlockType("down")
  11633.     if R.data == "level" then
  11634.         T:turnLeft(1)
  11635.         utils.goBack(1)
  11636.         if blockType:find("rail") ~= nil then
  11637.              moves = 3
  11638.         end
  11639.         T:down(moves)
  11640.         T:go("F1R1")
  11641.         --clsTurtle.place(self, blockType, damageNo, direction, leaveExisting)
  11642.         T:place(R.useBlockType, "forward", false)
  11643.         utils.goBack(1)
  11644.         T:place("minecraft:redstone_torch", "forward", true)
  11645.         T:turnLeft(1)
  11646.         utils.goBack(1)
  11647.         T:up(moves)
  11648.         T:go("F1R1F1")
  11649.     elseif R.data == "up" then -- sloping rail up/down is relative to dirtection facing
  11650.         moves = 3
  11651.         T:turnLeft(1)
  11652.         utils.goBack(1)
  11653.         if blockType:find("rail") ~= nil then
  11654.              moves = 4
  11655.         end
  11656.         T:down(moves)
  11657.         T:go("F1L1")
  11658.         T:place("minecraft:redstone_torch", "up", false)
  11659.        
  11660.         T:turnRight(1)
  11661.         utils.goBack(1)
  11662.         T:up(moves)
  11663.         T:go("F1R1")
  11664.     end
  11665.     return {}
  11666. end
  11667.  
  11668. local function plantTreefarm(R) -- 23
  11669.     -- already known R.networkFarm, R.subChoice (1 = singe, 2 = double, 3 = mangrove)
  11670.     --T:place(blockType, damageNo, leaveExisting, signText)
  11671.     local lib = {}
  11672.    
  11673.     function lib.checkSaplings(firstChoice, secondChoice)
  11674.         local saplingSlot, sapling, count = T:getSaplingSlot(firstChoice)
  11675.         if count < 4  and secondChoice ~= "" then
  11676.             saplingSlot, sapling, count = T:getSaplingSlot(secondChoice)
  11677.         end
  11678.         if count == 0 then
  11679.             sapling = ""
  11680.         end
  11681.        
  11682.         return sapling, count
  11683.     end
  11684.    
  11685.     function lib.emptyInventory(R)
  11686.         if not T:isEmpty() then
  11687.             utils.useSticksAsFuel()
  11688.             network.sendItemToNetworkStorage(R, "barrel", "sapling", 64)
  11689.             network.sendItemToNetworkStorage(R, "barrel", "propagule", 64)
  11690.             network.sendItemToNetworkStorage(R, "barrel", "apple", 64)
  11691.             network.sendItemToNetworkStorage(R, "barrel", "dirt", 64)
  11692.             network.sendItemToNetworkStorage(R, "chest", "all", 0)
  11693.         end
  11694.     end
  11695.    
  11696.     function lib.getSaplingInventory(R)
  11697.         local saplings = {}
  11698.         saplings.oak_sapling = 0
  11699.         saplings.spruce_sapling = 0
  11700.         saplings.birch_sapling = 0
  11701.         saplings.jungle_sapling = 0
  11702.         saplings.acacia_sapling = 0
  11703.         saplings.dark_oak_sapling = 0
  11704.         saplings.cherry_sapling = 0
  11705.         saplings.mangrove_propagule = 0
  11706.  
  11707.         local firstChoice = ""
  11708.         local firstCount = 0
  11709.         local secondChoice  = ""
  11710.         local secondCount = 0
  11711.         if T:getItemSlot("sapling") > 0 then
  11712.             for i = 1, 16 do
  11713.                 if turtle.getItemCount(i) > 0 then
  11714.                     local data = turtle.getItemDetail(i)
  11715.                     if data.name:find("sapling") ~= nil then
  11716.                         local name = data.name:sub(11) -- removes "minecraft:"
  11717.                         saplings[name] = saplings[name] + data.count
  11718.                     end
  11719.                 end
  11720.             end
  11721.         elseif T:getItemSlot("propagule") > 0 then
  11722.             saplings.mangrove_propagule = turtle.getItemCount(T:getItemSlot("propagule"))
  11723.         else    -- no saplings onBoard: ? automated networked farm
  11724.             if R.networkFarm then
  11725.                 -- networked farms use 1 type of sapling indicated by log embedded to right of modem
  11726.                 -- discovered from utils.assessTreeFarm() on manual Startup
  11727.                 if R.auto then  -- called here as NOT manual startup
  11728.                     R = utils.assessTreeFarm(R)
  11729.                     if R.message ~= "" then
  11730.                         return {R.message}  -- location of turtle error
  11731.                     end
  11732.                 end
  11733.                 local turtleSlot, turtleCount = network.getItemFromNetwork("barrel", R.useBlockType, 16)
  11734.                 if turtleCount == 0 then    -- ask player for saplings
  11735.                     T:checkInventoryForItem({R.useBlockType}, {16}, true, "Saplings required for tree farm")
  11736.                 end
  11737.                 saplings[R.useBlockType] = saplings[R.useBlockType] + turtleCount
  11738.                 turtleSlot, turtleCount = network.getItemFromNetwork("barrel", "minecraft:dirt", 16)
  11739.                 if turtleCount == 0 then
  11740.                     turtleSlot, turtleCount = network.getItemFromNetwork("chest", "minecraft:dirt", 16)
  11741.                 end
  11742.                 if turtleCount == 0 then    -- ask player
  11743.                     T:checkInventoryForItem({"minecraft:dirt"}, {16}, true, "Dirt required for tree farm")
  11744.                 end
  11745.                 return saplings, R.useBlockType, R.useBlockType
  11746.             end
  11747.         end
  11748.         for sapling, count in pairs(saplings) do
  11749.             if count > firstCount then
  11750.                 firstCount = count
  11751.                 firstChoice = sapling
  11752.             else
  11753.                 if count > secondCount then
  11754.                     secondCount = count
  11755.                     secondChoice = sapling
  11756.                 end
  11757.             end
  11758.         end
  11759.         return saplings, firstChoice, secondChoice -- table, "oak_sapling", "dark_oak_sapling"
  11760.     end
  11761.    
  11762.     function lib.createIsland(sapling, count, exit)
  11763.         -- place 4 dirt with saplings on all 4 unless jungle
  11764.         -- sapling count/type already checked
  11765.         T:forward(2) -- assume starting outside planting area
  11766.         for i = 1, 4 do
  11767.             T:go("R1F1")
  11768.             T:place("dirt", "down", false, "", true)
  11769.         end
  11770.         T:up(1)
  11771.         if count >= 4 then
  11772.             for i = 1, 4 do
  11773.                 T:go("R1F1")
  11774.                 T:place(sapling, "down", false, "", true)
  11775.             end
  11776.         else
  11777.             if sapling:find("dark") == nil and sapling ~= "" then
  11778.                 T:place(sapling, "down", false, "", true)
  11779.             end
  11780.         end
  11781.         if exit == "forward" then
  11782.             T:go("F1D1")
  11783.         elseif exit == "right" then
  11784.             T:go("R1F2D1")
  11785.         elseif exit == "left" then
  11786.             T:go("L1F1D1")
  11787.         elseif exit == "back" then
  11788.             T:go("R2F2D1")
  11789.         end
  11790.     end
  11791.    
  11792.     function lib.createSingle(sapling, exit)
  11793.         -- place single dirt with sapling on top
  11794.         -- sapling count/type already checked
  11795.         T:place("dirt", "down", false, "", true)
  11796.         T:up(1)
  11797.         if not T:place(sapling, "down", false, "", true) then -- try specific sapling
  11798.             T:place("sapling", "down", false, "", true)     -- any available sapling
  11799.         end
  11800.         if exit == "forward" then
  11801.             T:go("F1D1")
  11802.         elseif exit == "right" then
  11803.             T:go("R1F1D1")
  11804.         elseif exit == "left" then
  11805.             T:go("L1F1D1")
  11806.         elseif exit == "back" then
  11807.             T:go("R2F1D1")
  11808.         end
  11809.     end
  11810.    
  11811.     function lib.getMangroveSupplies()
  11812.         local turtleSlot, turtleCount = network.getItemFromNetwork("barrel", "minecraft:dirt", 169)
  11813.         if turtleCount < 169 then
  11814.             turtleSlot, turtleCount = network.getItemFromNetwork("chest", "minecraft:dirt", 169 - turtleCount)
  11815.             if turtleCount < 169 then   -- ask player for saplings
  11816.                 T:checkInventoryForItem({"dirt"}, {169 - turtleCount})
  11817.             end
  11818.         end
  11819.         turtleSlot, turtleCount = network.getItemFromNetwork("barrel", "minecraft:mangrove_propagule", 25)
  11820.         if turtleCount == 0 then    -- ask player for saplings
  11821.             T:checkInventoryForItem({"mangrove_propagule"}, {25}, true, "Mangrove propagules required")
  11822.         end
  11823.     end
  11824.    
  11825.     function lib.plantMangrove()
  11826.         T:go("L1F6 R1F1 U1")
  11827.         createFloorCeiling({width = 13, length = 13, up = false, down = true,
  11828.                             height = 0, subChoice = 0, useBlockType = "minecraft:dirt",
  11829.                             inventory = T:getInventory()})
  11830.        
  11831.         --T:go("U1F5 R1F5 L1")          -- lower left of planting area, facing Back
  11832.         T:go("U1F4 R1F4 L1")            -- lower left of planting area, facing Back
  11833.         for x = 1, 5 do
  11834.             for i = 1, 5 do
  11835.                 T:place("propagule", "down")    -- left 1
  11836.                 if i < 5 then
  11837.                     T:forward(1)
  11838.                 end
  11839.             end
  11840.             if x % 2 == 1 then
  11841.                 if x < 5 then
  11842.                     T:go("R1F1R1")
  11843.                 end
  11844.             else
  11845.                 T:go("L1F1L1")
  11846.             end
  11847.         end
  11848.         T:go("L1F2 L1F2 D2U2 F7R2 D2")
  11849.         --[[
  11850.         T:place("propagule", "down")    -- left 1
  11851.         T:forward(1)
  11852.         T:place("propagule", "down")    -- left 2
  11853.         T:forward(1)
  11854.         T:place("propagule", "down")    -- left 3/top 1
  11855.  
  11856.         T:go("R1F1")
  11857.         T:place("propagule", "down")    -- top 2
  11858.         T:forward(1)
  11859.         T:place("propagule", "down")    -- top 3/right 1
  11860.         T:go("R1F1")
  11861.        
  11862.         T:place("propagule", "down")    -- right 2
  11863.         T:forward(1)
  11864.         T:place("propagule", "down")    -- right 3/bottom 1
  11865.         T:go("R1F1")
  11866.         T:place("propagule", "down")    -- bottom 2]]
  11867.        
  11868.         --T:go("R1F1R2 D2U2 F7R2 D2")
  11869.     end
  11870.    
  11871.     function lib.plantSingle(R)
  11872.         local sapling, count = lib.checkSaplings(firstChoice, secondChoice)
  11873.         if count >= 1 then
  11874.             if R.networkFarm then
  11875.                 T:go("U1L1 F3R1 F4") -- outside first area
  11876.             else
  11877.                 T:go("U1F4") -- outside first area
  11878.             end
  11879.             for i = 1, 3 do -- column 1/4
  11880.                 lib.createSingle(sapling, "forward")
  11881.                 T:forward(1)
  11882.             end
  11883.             for i = 1, 2 do
  11884.                 lib.createSingle(sapling, "right") -- place 4th dirt/saling and exit to right
  11885.                 T:forward(1)
  11886.             end
  11887.             for i = 1, 2 do -- column 2/4
  11888.                 lib.createSingle(sapling, "forward")
  11889.                 T:forward(1)
  11890.             end
  11891.             for i = 1, 2 do
  11892.                 lib.createSingle(sapling, "left") -- place 4th dirt/saling and exit to right
  11893.                 T:forward(1)
  11894.             end
  11895.             for i = 1, 2 do -- column 3/4
  11896.                 lib.createSingle(sapling, "forward")
  11897.                 T:forward(1)
  11898.             end
  11899.             for i = 1, 2 do
  11900.                 lib.createSingle(sapling, "right") -- place 4th dirt/saling and exit to right
  11901.                 T:forward(1)
  11902.             end
  11903.             for i = 1, 3 do -- column 4/4
  11904.                 lib.createSingle(sapling, "forward")
  11905.                 T:forward(1)
  11906.             end
  11907.             if R.networkFarm then
  11908.                 T:go("R1F3 R1F2L1") -- in-between 2 trees
  11909.                 if R.auto then
  11910.                     harvestTreeFarm(R)
  11911.                 else
  11912.                     T:go("L1F4R2D1")
  11913.                 end
  11914.             else
  11915.                 T:go("R1F6 L1F2 R2D1")
  11916.             end
  11917.         else
  11918.             return "No saplings to plant"
  11919.         end
  11920.        
  11921.         return ""
  11922.     end
  11923.    
  11924.     function lib.plantDouble(R)
  11925.         -- assume placed 4 blocks from start
  11926.         local sapling, count = lib.checkSaplings(saplings, firstChoice, secondChoice)
  11927.        
  11928.         if count >= 4 then
  11929.             if R.networkFarm then
  11930.                 T:go("U1L1 F3R1 F3") -- outside first area
  11931.             else
  11932.                 T:go("U1F3") -- outside first area
  11933.             end
  11934.             lib.createIsland(sapling, count, "forward")
  11935.             sapling, count = lib.checkSaplings(firstChoice, secondChoice)
  11936.             T:go("F2")
  11937.             lib.createIsland(sapling, count, "right")
  11938.             sapling, count = lib.checkSaplings(firstChoice, secondChoice)
  11939.             T:go("F2")
  11940.             lib.createIsland(sapling, count,  "right")
  11941.             sapling, count = lib.checkSaplings(firstChoice, secondChoice)
  11942.             T:go("F2")
  11943.             lib.createIsland(sapling, count, "forward")
  11944.            
  11945.             if R.networkFarm then
  11946.                 T:go("R1F4 R1F1 L1")            -- on left side of double tree
  11947.                 if R.auto then
  11948.                     harvestTreeFarm(R)
  11949.                 else
  11950.                     T:go("L1F4 L1F1 L1D1")  -- back to start
  11951.                 end
  11952.             else
  11953.                 T:go("R1D1 F6L1 F3R2") -- outside first area
  11954.             end
  11955.         else
  11956.             return "Insufficient saplings to plant"
  11957.         end
  11958.         return ""
  11959.     end
  11960.    
  11961.     if R.networkFarm then
  11962.         local message = network.loadStorageLists()  -- initialises or creates lists of where an item can be found: GLOBAL LISTS!
  11963.         if message ~= nil then return {message} end
  11964.         lib.emptyInventory(R)
  11965.     end
  11966.     if R.subChoice == 3 then    -- mangrove, R.networkFarm only
  11967.         lib.getMangroveSupplies()
  11968.         lib.plantMangrove()
  11969.     else
  11970.         local saplings, firstChoice, secondChoice = lib.getSaplingInventory(R)
  11971.         if firstChoice ~= "" then
  11972.             print("first sapling: "..firstChoice .. " ("..saplings[firstChoice]..")")
  11973.         end
  11974.         if secondChoice ~= "" then
  11975.             print("second sapling: "..secondChoice .. " ("..saplings[secondChoice]..")")
  11976.         end
  11977.    
  11978.         -- check type/size of farm
  11979.         local message = ""
  11980.         if R.subChoice == 1 then            -- 16 single trees
  11981.             message = lib.plantSingle(R)    -- always ""
  11982.         elseif R.subChoice == 2 then        -- 4 double trees (4 saplings)
  11983.             message = lib.plantDouble(R)    -- "" or error about numbers
  11984.         end
  11985.     end
  11986.     if R.networkFarm then
  11987.         lib.emptyInventory(R)
  11988.     end
  11989.     return {message}
  11990. end
  11991.  
  11992. local function quickMine(R) -- 17
  11993.     --[[
  11994.     mine valuable blocks from specified area
  11995.     if R.up set or lava/water found, ceiling is formed
  11996.     if R.down is set, or lava/water found layer is kept 1 block high
  11997.     R.subChoice
  11998.     1 At mine area start, on the floor
  11999.     2 At mine area start, on the ceiling
  12000.     3 On floor, start 1 block ahead
  12001.     4 On ceiling, start 1 block ahead
  12002.     5 On floor diagonally to left"
  12003.     ]]
  12004.     local lib = {}
  12005.    
  12006.     function lib.refuel(direction)
  12007.         if T:getWater(direction)  then
  12008.             T:refuel(1000, false)
  12009.         end
  12010.     end
  12011.    
  12012.     function lib.mine(R)
  12013.         -- starts on ceiling
  12014.         local isValuable, blockType
  12015.         for i = 1, R.length do
  12016.             local fillUp = R.up
  12017.             local fillDown = R.down
  12018.             isValuable, blockType = T:isValuable("down")
  12019.             if isValuable then
  12020.                 T:dig("down")
  12021.             elseif blockType:find("water") ~= nil then
  12022.                 fillDown = true
  12023.             elseif blockType:find("lava") ~= nil then
  12024.                 lib.refuel("down")
  12025.                 fillDown = true
  12026.             end
  12027.             isValuable, blockType = T:isValuable("up")
  12028.             if isValuable then
  12029.                 T:dig("up")
  12030.             elseif blockType:find("water") ~= nil then
  12031.                 fillUp = true
  12032.             elseif blockType:find("lava") ~= nil then
  12033.                 lib.refuel("up")
  12034.                 fillUp = true
  12035.             end
  12036.             --if not turtle.detectUp() and fillUp then
  12037.             if fillUp then
  12038.                 T:fillVoid("up")
  12039.             end
  12040.             --if not turtle.detectDown() and fillDown then
  12041.             if fillDown then
  12042.                 T:fillVoid("down")
  12043.             end
  12044.             if i < R.length then
  12045.                 T:forward(1)
  12046.             end
  12047.         end
  12048.     end
  12049.    
  12050.     local outbound = true
  12051.    
  12052.     if R.subChoice == 1 then
  12053.         T:up(1)
  12054.     elseif R.subChoice == 3 then
  12055.         T:go("U1F1")
  12056.     elseif R.subChoice == 4 then
  12057.         T:go("F1")
  12058.     elseif R.subChoice == 5 then
  12059.         T:go("U1R1 F1L1 F1")
  12060.     end
  12061.    
  12062.     for w = 1, R.width do
  12063.         lib.mine(R)
  12064.         if w < R.width then
  12065.             if outbound then
  12066.                 T:go("R1F1R1")
  12067.             else
  12068.                 T:go("L1F1L1")
  12069.             end
  12070.             outbound = not outbound
  12071.         end
  12072.         if T:getFirstEmptySlot() == 0 then
  12073.             T:dumpRefuse("forward", 1)
  12074.         end
  12075.     end
  12076.     if outbound then
  12077.         T:go("L1F"..R.width - 1 .."L1F"..R.length - 1)
  12078.     else
  12079.         T:go("R1F"..R.width - 1 .."R1")
  12080.     end
  12081.    
  12082.     return {}
  12083. end
  12084.  
  12085. local function quickMineCorridor(R) -- 16
  12086.     --[[
  12087.     R.subChoice
  12088.     1: At corridor start, on the floor
  12089.     2: At corridor start, on the ceiling
  12090.     3: On floor, start 1 block ahead
  12091.     4: On ceiling, start 1 block ahead
  12092.     ]]
  12093.     if R.subChoice == 2 then
  12094.         T:down(1)
  12095.     elseif R.subChoice == 3 then
  12096.         T:forward(1)
  12097.     elseif R.subChoice == 4 then
  12098.         T:go("D1F1")
  12099.     end
  12100.     local width = R.width - 1
  12101.     local length = R.length - 1
  12102.     R.silent = true
  12103.     R.length = length
  12104.     createCorridor(R, true) -- put floor and ceiling for R.length, place torch at start
  12105.     T:turnRight(1)
  12106.     R.length = width
  12107.     createCorridor(R, true)
  12108.     T:turnRight(1)
  12109.     R.length = length
  12110.     createCorridor(R, true)
  12111.     T:turnRight(1)
  12112.     R.length = width
  12113.     createCorridor(R, true)
  12114.     T:turnRight(1)
  12115.    
  12116.     return {}
  12117. end
  12118.  
  12119. local function repairWall(startAt, height, width, replaceWith)
  12120.     -- go up to startAt
  12121.    
  12122.     -- if width = 1
  12123.    
  12124.         -- for h = startAt, height, 1 do
  12125.        
  12126.             -- replace block with replaceWith ("" = any)
  12127.            
  12128.             -- move up
  12129.            
  12130.         --end
  12131.        
  12132.         -- move back to beginning
  12133.        
  12134.     -- else
  12135.    
  12136.         -- remain = height % 2
  12137.        
  12138.         -- for w = 1, width - remain do
  12139.        
  12140.             -- for h = startAt, height, 1 do
  12141.            
  12142.                 -- replace block with replaceWith ("" = any)
  12143.                
  12144.                 -- move up
  12145.                
  12146.             --end
  12147.            
  12148.             -- move to the right 1 block
  12149.            
  12150.             -- for i = height, startAt, -1 do
  12151.            
  12152.                 -- replace block with replaceWith ("" = any)
  12153.                
  12154.                 -- move down
  12155.                
  12156.             --end
  12157.            
  12158.         -- end
  12159.        
  12160.     -- end
  12161.    
  12162. end
  12163.  
  12164. local function getTaskItemsList()
  12165.     -- list of items required for each task
  12166.     local text = {}
  12167.     --MINING
  12168.     text[11] = {"1 ladder for each level","levels / 4 torch (optional)","levels * 4 stone"}             -- ladder to bedrock
  12169.     text[12] = {"2 stairs for each level", "6 * levels stone", "1 chest"}                               -- stairs up/down
  12170.     text[13] = {"24 torch (optional)", "1 bucket (optional)", "64 stone", "1 chest"}                    -- mine at this level
  12171.     text[14] = {"levels * 4 stone","water_bucket"}                                                      -- safe drop to water block
  12172.     text[15] = {"1 soul sand", "3 water bucket", "levels * 4 stone", "1-9 buckets (optional)",
  12173.                 "2 signs if no ladder present"}                                                         -- single column bubble lift
  12174.     text[16] = {"1 bucket (optional)", "64 stone"}                                                      -- quick corridor
  12175.     text[17] = {"1 bucket (optional)", "64 stone"}                                                      -- quick mine
  12176.     text[18] = {"1 bucket (optional)"}                                                                  -- mine to bedrock
  12177.     text[19] = {"1 UNUSED diamond sword (optional)"}                                                    -- rob mineshaft
  12178.  
  12179.     -- FORESTRY
  12180.     text[21] = {"1 chest (optional)"}                                                                   -- Fell Tree
  12181.     text[22] = {"320 stone", "4 polished stone"}                                                        -- Create treefarm
  12182.     text[23] = {"min 4 saplings", "16 dirt"}                                                            -- plant treefarm
  12183.     text[24] = {"No items required"}                                                                    -- Harvest treefarm
  12184.     text[25] = {"width * length * 2 walls or fences", "torches (optional)", "4 barrels (optional)"}     -- fence or wall a forest
  12185.     text[26] = {"1 chest", " any saplings"}                                                             -- harvest and replant walled rectangle of natural forest
  12186.                
  12187.     -- FARMING         
  12188.     text[31] = {"64 stone","128 dirt (optional)", "4 water buckets","4 chests / barrels","1 sapling"}   -- Create modular crop farm
  12189.     text[32] = {"64 stone","128 dirt (optional)", "4 water buckets","5 chests/ barrels","1 sapling"}    -- extend modular farm
  12190.     text[33] = {"No items required"}                                                                    -- Manual harvest and auto setup
  12191.     text[34] = {"Walls or Fences", "Torches (optional)"}                                                -- Build a wall or fence
  12192.     text[35] = text[34]                                                                                 -- Wall or fence enclosed area
  12193.  
  12194.     -- OBSIDIAN
  12195.     text[41] = {"stone to cover area of obsidian"}                                                      -- Harvest obsidian
  12196.     text[42] = {"2 x height + width obsidian", "4 stone"}                                               -- build Nether portal
  12197.     text[43] = {"No items required"}                                                                    -- demolish Nether portal
  12198.     text[44] = {"length * 4 stone/netherrack, bucket"}                                                  -- Netherite stripping
  12199.     text[45] = {"84 stone"}                                                                             -- undermine dragon towers
  12200.     text[46] = {"No items required"}                                                                    -- deactivate dragon tower
  12201.     text[47] = {"448 stone, 145 ladders, 1 obsidian, 1 water bucket"}                                   -- dragon water trap
  12202.     text[48] = {"height * 4 stone","height * ladders", "1 trapdoor"}                                    -- build end portal platform
  12203.                
  12204.     --CANALS BRIDGES WALKWAYS
  12205.     text[51] = {"dirt or stone * length","torch (optional)"}                                            -- single path
  12206.     text[52] = {"dirt or stone 2 * length","torch (optional)"}                                          -- covered path / tunnel
  12207.     text[53] = {"2 * length stone or dirt","2 water buckets","torches (optional)"}                      -- water canal
  12208.     text[54] = {"0.5 * length Packed or blue ice","length * slabs","stone","torches (optional)"}        -- Ice canal
  12209.     text[55] = {"width * length stone"}                                                                 -- platform
  12210.     text[56] = {"width + 1 * length + 1 stone"}                                                         -- sinking platform
  12211.     text[57] = {"height * 10 stone","height * 4 + 2 soul sand, 2 water buckets"}                        -- Boat bubble lift
  12212.    
  12213.     -- MOB FARM
  12214.     text[61] = {"512 stone","1 slab"}                                                                   -- 9x9 cube round spawner
  12215.     text[62] = {"640 stone","37 slab","8 powered rail","64 rail","2 redstone torch",
  12216.                 "1 hopper minecart","1 stone button"}                                                   -- 9x9 cube round blaze spawner
  12217.     text[63] = {"2 water buckets","If using bubble lift","2 fence","2 signs","1 slab",
  12218.                 "1 soul sand (or dirt as placeholder)"}                                                 -- flood spawner chamber   
  12219.     text[64] = {"128 stone","2 water buckets","1 soul sand"}                                            --Build bubble tower kill zone
  12220.     text[65] = {"1-2 computer","2 sticky or 4 normal pistons","6 network cable + 2 wired modems",
  12221.                 "23 polished stone + 7 slabs","2 glass","1 stone pressure plate","1 dispenser",
  12222.                 "1 hopper + 2 chest", "9 redstone", "1 lava bucket"}                                    --Computercraft mob grinder
  12223.     text[66] = {"1856 stone, diorite etc (inc polished)","1 chest","10 empty buckets",
  12224.                 "2 water buckets","192 fence","8 signs","3 ladder","2 soul sand"}                       -- build endermen observation tower
  12225.     text[67] = {"24 blocks of any type"}                                                                -- Cover Trial spawners to prevent mobs escaping
  12226.    
  12227.     -- AREA CARVING
  12228.     text[71] = {"width * length dirt"}                                                                  -- Clear field
  12229.     text[72] = {"No items required"}                                                                    -- Clear rectangle width, length
  12230.     text[73] = {"No items required"}                                                                    -- Clear wall height, length
  12231.     text[74] = {"No items required"}                                                                    -- Clear rectangle perimeter only width, length
  12232.     text[75] = {"No items required"}                                                                    -- Clear structure floor/walls/ceiling
  12233.     text[76] = {"No items required"}                                                                    -- clear solid
  12234.     text[77] = {"No items required"}                                                                    -- Dig a trench
  12235.     text[78] = {"No items required"}                                                                    -- carve mountain
  12236.     text[79] = {"width * height Any material floor / ceiling"}                                          -- floor or ceiling
  12237.     text[710] = {"Blocks to add 'floor as you go'"}                                                     -- Direct control of movement
  12238.                
  12239.     -- LAVA WATER
  12240.     text[81] = {"Enough sand or gravel to fill volume"}                                                 -- sand based utilities
  12241.     text[82] = {"length * depth stone"}                                                                 -- vertical wall from surface
  12242.     text[83] = {"Blocks to add 'floor as you go'"}                                                      -- player driven path construction
  12243.     text[84] = {"width * length + extra stone"}                                                         -- water clearing by repeated block deletion
  12244.     text[85] = text[84]                                                                                 -- sinking platform
  12245.     text[86] = {"Full inventories of sand or stone"}                                                    -- Ocean monument utilities
  12246.     text[87] = {"ladder to height","stone, dirt netherrack 4 X height"}                                 -- ladder to water/lava
  12247.     text[88] = {"No items required"}                                                                    -- clear water plants
  12248.     text[89] = {"water buckets / buckets"}                                                              -- convert flowing water to source
  12249.     text[810] = {"6 buckets","slabs to cover area"}                                                     -- create sloping water
  12250.                
  12251.     -- BUILDING and MINECART
  12252.     text[91] = {"length * height stone"}                                                                -- Build a wall
  12253.     text[92] = {"length * height * 4 stone"}                                                            -- build a walled area
  12254.     text[93] = {"length * width * height stairs", "length * slabs"}                                     -- build a gable roof
  12255.     text[94] = {"length * width * height stairs", "length * slabs"}                                     -- Build a pitched roof
  12256.     text[95] = {"1 block of choice","1 redstone torch"}                                                 -- place redstone torch under current block
  12257.     text[96] = {"1 block of choice","1 redstone torch"}                                                 -- place redstone torch on upward slope
  12258.     text[97] = {"height x block of choice","height/3 x redstone torch"}                                 -- build downward slope
  12259.     text[98] = text[97]                                                                                 -- build upward slope
  12260.     return text
  12261. end
  12262.  
  12263. local function getTaskHelp(menuLevel, menuItem, noMenu, getInteger)
  12264.     -- display help about selected task
  12265.     -- terminal size = 39 x 13
  12266. --[[This line of text = 39 characters]]
  12267.     noMenu = noMenu or false
  12268.     getInteger = getInteger or false
  12269.     info = {}
  12270.     info.main = {}
  12271.     info.sub = {}
  12272.     table.insert(info.main,
  12273.     [[                ~lightGray~MINING:~white~
  12274. Can be used in over-world or nether.~brown~
  12275. Ladders and ~lightGray~stairs up/down    
  12276. Create a pre-formatted 33 x 33 blocks  
  12277. mine at chosen level.
  12278. ~blue~Bubble lift and ~cyan~safe drop to water.~magenta~  
  12279. Faster version of ~white~33x33 mine pattern  
  12280. using ~magenta~corridor and ~pink~rectangle functions.~brown~
  12281. Mine bottom layer to bedrock (not worth
  12282. the fuel and time)
  12283. ~brown~Strip resources from abandoned mines.
  12284. ]])
  12285.     table.insert(info.main,
  12286.     [[               ~lime~FORESTRY:~brown~
  12287. Fell Tree can be used in Nether as well
  12288. for cutting any size tree / branches~lime~
  12289. Create a simple 16 single tree farm
  12290. or 4 double size (Dark oak or any type)
  12291. suitable for turtle harvesting.
  12292. ~brown~ fence or wall a rectangle keeping
  12293. to contours.
  12294. ~green~Forest can be harvested and replanted.
  12295. (Must be walled off)
  12296.  
  12297. ]])
  12298.     table.insert(info.main,
  12299.     [[                ~green~FARMING:~yellow~
  12300. Farm modules can be built to fixed size
  12301. and placed next to each other in linear
  12302. or rectangular pattern.~lime~
  12303. Whole farm is managed by a dedicated
  12304. turtle, which must be equipped with a
  12305. diamond hoe as well as pickaxe.
  12306. Fuel obtained from a tree placed in the
  12307. corner of each module.
  12308. Double chests store produce and seeds.
  12309. ]])
  12310.     table.insert(info.main,
  12311.     [[               ~red~OBSIDIAN:~lightGray~
  12312. The turtle can extract obsidian from
  12313. lava areas safely.~pink~
  12314. Nether portals can be built or removed
  12315. without needing diamond pickaxes.~orange~
  12316. End World dragon towers can be
  12317. undermined ready for deactivating.
  12318. End world towers can have the crystals
  12319. destroyed.~red~
  12320. Water trapped dragon attacked from above~brown~
  12321. Minecart end portal stations built.
  12322. ]])
  12323.     table.insert(info.main,
  12324.     [[        ~brown~PATHS, BRIDGES, CANALS:~white~
  12325. Can be used in Nether and End.~lightGray~
  12326. Build pathways over air, water or lava
  12327. Optional roofed pathway for Nether use.
  12328. Tunnel through rock and place a floor
  12329. at the same time.~blue~
  12330. Build a water canal with towpath.~cyan~
  12331. Build a super-fast ice canal.~brown~
  12332. Platform for use over air, water, lava
  12333. Sinking version is removed and replaced
  12334. 1 block lower each time
  12335. ]])
  12336.     table.insert(info.main,
  12337.     [[             ~lightGray~SPAWNER FARMS:~brown~
  12338. Tools to create mob farms round
  12339. existing spawners. ~red~
  12340. Special version for Blaze farms
  12341. uses rail collection.~cyan~
  12342. Choice of bubble lift mob dropper
  12343. or ~brown~simple deep trench.~gray~
  12344. Enderman observation tower can be
  12345. built >128 above ground: is ~red~expensive.~gray~
  12346. Re-spawns new mobs when used.~red~
  12347. Suggest build only the base.
  12348. ]])
  12349.     table.insert(info.main,
  12350.     [[    ~orange~AREA CLEARING AND REFORMING:~lime~
  12351. Tools to clear a field including trees~magenta~
  12352. Clear rectangles.~pink~
  12353. Clear single walls.~brown~
  12354. Clear hollow and~orange~solid structures.~brown~
  12355. Dig a trench.~gray~
  12356. Carve away side of a mountain.~lightBlue~
  12357. Place or replace floors and ceilings
  12358.  
  12359.  
  12360.  
  12361. ]])
  12362.     table.insert(info.main,
  12363.     [[      ~blue~WATER AND LAVA TOOLS:~white~
  12364. Used to drain ocean monuments and
  12365. shipwrecks. Can also be used to make
  12366. underwater base. ~blue~Water is cleared using
  12367. sand dropping and recycling~cyan~
  12368. Destructive draining uses solid block
  12369. placing and recycling.~green~
  12370. Water plants can be removed without
  12371. damaging structures.~lightBlue~
  12372. Tools to manipulate water areas:
  12373. (convert to source, sloping water)
  12374. ]])
  12375.     table.insert(info.main,
  12376.     [[         ~cyan~BUILDING, MINECART TOOLS:~lightGray~
  12377. Build simple walls, buildings and ~gray~roofs
  12378.  
  12379. ~orange~Used to build diagonal uphill slope~brown~
  12380. and downhill slope for placing
  12381. 45 degree rail tracks.~red~
  12382. Placing Redstone torches under powered
  12383. rails when above ground level (viaduct)
  12384.  
  12385.  
  12386.  
  12387. ]])
  12388.     table.insert(info.main,
  12389.     [[           ~yellow~MEASURING TOOLS:
  12390. Used to measure
  12391.     ~red~Height
  12392.     ~purple~Depth
  12393.     ~magenta~Length~pink~
  12394.    
  12395.     Greatest depth of water
  12396.  
  12397.  
  12398.  
  12399.  
  12400. ]])
  12401.     table.insert(info.main,
  12402.     [[           ~yellow~NETWORK TOOLS:
  12403. Still on ToDo List!
  12404.  
  12405.  
  12406.  
  12407.    
  12408.  
  12409.  
  12410.  
  12411.  
  12412.  
  12413. ]])
  12414.  
  12415.     info.sub[11] =
  12416. [[~yellow~Place me on the ground at ~red~^~yellow~
  12417. The ~brown~ladder ~yellow~will start at this level
  12418. and go up or down.
  12419.  
  12420. ~lightGray~| | | | | |
  12421. ~lightGray~| | |*| | | * = Ladder support block
  12422. | | |~brown~L~lightGray~| | | ~brown~L = Ladder
  12423. ~lightGray~| | |~red~^~lightGray~| | | ~red~^ = Turtle
  12424. ~lightGray~| | | | | |
  12425. | | | | | |
  12426.  
  12427. ]]-- Ladder up/down
  12428.     info.sub[12] =
  12429. [[~lightGray~Place me on the ground at ~red~^
  12430.  
  12431. ~lightGray~| | | | | | | | * = Solid block
  12432. ~lightGray~| |*|*|*|*|*| | ~cyan~^ = Stairs to back
  12433. ~lightGray~| |*|-|~cyan~>~lightGray~|-|*| | ~cyan~> = Stairs to right
  12434. ~lightGray~| |*|~cyan~^|~lightGray~*|~cyan~V~lightGray~|*| | ~cyan~V = Stairs to front
  12435. ~lightGray~| |*|-|~cyan~<~lightGray~|-|*| | ~cyan~< = Stairs to left
  12436. ~lightGray~| |*|~red~^~lightGray~|*|*|*| | - = Corner tread
  12437. ~lightGray~| | | | | | | | ~red~^ = Turtle
  12438.  
  12439. ~lightGray~Going ~blue~down~lightGray~ -> digs down then stairs up
  12440. ]] -- Stairs up/down
  12441.         info.sub[13] =
  12442. [[~yellow~Press F3 to check Y level.
  12443.  
  12444. ~lightGray~| |~red~^~lightGray~| | | ~red~^ = Turtle behind ladder
  12445. ~lightGray~| |~gray~s~lightGray~| | | ~gray~s = 1 block space
  12446. ~lightGray~| |*| | | * = Ladder support block
  12447. ~lightGray~| |~brown~L~lightGray~| | | ~brown~L = Ladder
  12448. ~lightGray~| |~brown~s~lightGray~| | | ~brown~s~gray~ = 2 spaces (inc. ladder)
  12449. ~lightGray~| |~red~V~lightGray~| | | ~red~V = Turtle ahead of ladder~yellow~
  12450.  
  12451. ]]
  12452.     if bedrock == 0 then    --pre 1.18
  12453.         info.sub[13] = info.sub[13].."Place at Y = 5, 8, 11 ~red~(11 nether)" -- Create mine at this level
  12454.     else
  12455.         info.sub[13] = info.sub[13].."Place at Y = -59, -56, -53 (11 nether)" -- Create mine at this level
  12456.     end
  12457.     info.sub[14] =
  12458. [[~yellow~Turtle goes ~blue~DOWN ~yellow~to chosen level
  12459. enclosing all sides of the column.
  12460. Water placed at bottom. Returns here.
  12461. If next to a ladder, place as below:
  12462.  
  12463. ~lightGray~| | | | | |
  12464. ~lightGray~| | |*| | | * = Ladder support block
  12465. | |~red~^~brown~|L|~red~^~lightGray~| | ~brown~L = ladder
  12466. ~lightGray~| | | | | | ~red~^ = Turtle facing forward
  12467. ~lightGray~| | | | | |
  12468.            
  12469. ]] -- safe drop
  12470.     info.sub[15] =
  12471. [[~magenta~Direction of travel = UP!
  12472. ~yellow~Place me on the ground. I will build a
  12473. 3 x 1 ~blue~water ~yellow~source and a single column
  12474. bubble lift to the chosen height.
  12475. If near a ladder, place left or right:
  12476.  
  12477. ~lightGray~| | | | | |
  12478. ~lightGray~| | |*| | | * = Ladder support block
  12479. | |~blue~^~brown~|L|~blue~^~lightGray~| | ~brown~L = ladder
  12480. ~lightGray~| | | | | | ~blue~^ = Turtle facing forward
  12481. ~lightGray~| | | | | | ~blue~^ = Bubble column above
  12482.  
  12483. ]] -- single column bubble lift
  12484.     info.sub[16] =
  12485. [[~yellow~Place me as below:
  12486.  1. On ~blue~floor   ~yellow~(feet height)
  12487.  2. On ~lime~ceiling ~yellow~(eye height)
  12488.  
  12489. ~lightGray~B|B|B|B|B|B|B    ~yellow~W I D T H
  12490. ~lightGray~B| | | | | |B               ~orange~L
  12491. ~lightGray~B| |B|B|B| |B               ~orange~E
  12492. ~lightGray~B| |B|B|B| |B               ~orange~N
  12493. ~lightGray~B| |B|B|B| |B               ~orange~G
  12494. ~lightGray~B|~red~^~lightGray~| | | | |B  ~red~^~lightGray~ = Turtle   ~orange~T
  12495. ~lightGray~B|~cyan~P~lightGray~|B|B|B|B|B  ~cyan~P~lightGray~ = Player   ~orange~H
  12496. ]] -- quick corridor system
  12497.     info.sub[17] =
  12498. [[~yellow~~yellow~Place me as below:
  12499.  1. On ~blue~floor   ~yellow~(feet height)
  12500.  2. On ~lime~ceiling ~yellow~(eye height)
  12501. ~lightGray~B|B|B|B|B|B|B
  12502. B| | | | | |B
  12503. B| |B|B|B| |B
  12504. B| |B|B|B| |B
  12505. B| |~lime~^~lightGray~|B|B| |B
  12506. B|~red~^~lightGray~|~magenta~^~lightGray~| | | |B ~white~^~lightGray~ = turtle
  12507. B|~cyan~P~lightGray~|B|B|B|B|B ~cyan~P~lightGray~ = Player
  12508. ~yellow~Note colour of ~white~^~yellow~ matches next screen->
  12509. ]] -- quick mine
  12510.         info.sub[18] =
  12511. [[~yellow~Place me level -59 / 5 on the floor to
  12512. expose bedrock ~red~(slow and inefficient)
  12513.  
  12514. ~lightGray~| | | |B| |B|     ~yellow~W I D T H
  12515. ~lightGray~|B| | | | | |B               ~orange~L
  12516. ~lightGray~| | | |B|B| |                ~orange~E
  12517. ~lightGray~| | |B| | |B|                ~orange~N
  12518. ~lightGray~|B| | | |B| |                ~orange~G
  12519. ~lightGray~| | | | | | |B               ~orange~T
  12520. ~lightGray~|~red~^~lightGray~| | | |B| |B  ~red~^~lightGray~ = Turtle   ~orange~H
  12521. ]] -- mine all blocks to bedrock pre 1.12
  12522.         info.sub[19] =
  12523. [[~yellow~Place me on the end wall of a disused
  12524. mine in the centre block, 1 block above
  12525. the floor.
  12526. Provide a ~cyan~diamond ~yellow~sword for
  12527. harvesting string from spider webs~lightGray~
  12528.  
  12529. ~gray~-------   - = Ceiling
  12530. ~lightGray~| | | |
  12531. | |~red~T~lightGray~| |  ~red~T = Turtle (facing wall)
  12532. ~lightGray~| | | |
  12533. ~gray~-------  - = Floor
  12534. ]] -- salvage mineshaft
  12535.     info.sub[21] =
  12536. [[~yellow~Place me as below.
  12537. ~brown~Chest ~yellow~ONLY required if ~blue~0 ~yellow~fuel
  12538.  
  12539. Plan view:
  12540.  
  12541. ~green~   | | | |
  12542. ~green~   | |~lime~T~green~| |  ~lime~T = Tree
  12543. ~green~   | |~red~^~green~| |  ~red~^ = Turtle
  12544. ~green~   | | | |
  12545.  
  12546.  
  12547.  
  12548. ]] -- Fell Tree
  12549.     info.sub[22] =
  12550. [[~yellow~New, extend left/back=~red~^ ~orange~Extend Right=^
  12551. ~lightGray~|*|~blue~ | | | | | | ~brown~|D| |D| ~blue~| | ~lightGray~|*|
  12552. ~lightGray~|*|~blue~ | | ~brown~|D|D|~blue~ | | | | | | | ~lightGray~|*|
  12553. ~lightGray~|*|~blue~ | | ~brown~|D|D|~blue~ | ~brown~|D| |D| ~blue~| | ~lightGray~|*|
  12554. ~lightGray~|*|~blue~ | | | | | | | | | | | | ~lightGray~|*|
  12555. ~lightGray~|*|~blue~ | | | | | | | | | | | | ~lightGray~|*|
  12556. ~lightGray~|*|~blue~ | | | | | | | | | | | | ~lightGray~|*|
  12557. ~lightGray~|~red~^~lightGray~|*|*|*|*|*|*~magenta~|M|~lightGray~*|*|*|*|*|*|~orange~^|
  12558. ~lime~4 ~brown~double trees~green~ or 16 ~brown~single trees
  12559. ~brown~D = Dirt, ~magenta~M = Modem
  12560. ~red~^~yellow~ = Turtle ~red~(new/left/back) ~orange~^ on right
  12561. ]] -- Create treefarm
  12562.     info.sub[23] =
  12563. [[~yellow~Place me on ground as below ~lightGray~^ ~magenta~^
  12564. ~lime~|*|~blue~ | | | | | | | ~green~or |*|~blue~ | | ~brown~|D| |D| |
  12565. ~lime~|*|~blue~ | | ~brown~|D|D|~blue~ | | ~green~or |*|~blue~ | | | | | | |
  12566. ~lime~|*|~blue~ | | ~brown~|D|D|~blue~ | | ~green~or |*|~blue~ | | ~brown~|D| |D| |
  12567. ~lime~|*|~blue~ | | | | | | | ~green~or |*|~blue~ | | | | | | |
  12568. ~lime~|*|~blue~ | | | | | | | ~green~or |*|~blue~ | | | | | | |
  12569. ~lime~|*|*|*|*|~lightGray~^~lime~|*| ~magenta~|^| ~green~or |*|*|*|*|~lightGray~^~green~|*|*~magenta~|^|
  12570. ~lime~ 4 ~brown~double trees~green~   or  16 ~brown~single trees
  12571. ~brown~D = dirt
  12572. ~lightGray~^ = Turtle on marker (Legacy storage)
  12573. ~magenta~^ = Turtle on Modem (Network storage)
  12574. ]] -- Plant treefarm / Harvest treefarm
  12575.     info.sub[24] = info.sub[23]
  12576.     info.sub[25] =
  12577. [[~yellow~Place me at ~red~^
  12578.  
  12579. ~lightGray~|F|F|F|F|F|F|F| F = Fence or Wall
  12580. ~lightGray~|F|~brown~B~lime~| | | |~brown~B~lightGray~|F| ~brown~B = Barrel (corners)
  12581. ~lightGray~|F|~lime~ | | | | ~lightGray~|F|
  12582. ~lightGray~|F|~lime~ | | | |~cyan~T~lightGray~|F| ~cyan~T = Tree
  12583. ~lightGray~|F|~lime~ |~cyan~T~lime~| | | ~lightGray~|F|
  12584. ~lightGray~|F|~lime~ | | | | ~lightGray~|F|
  12585. ~lightGray~|F|~brown~B~lime~| |~cyan~T~lime~| |~brown~B~lightGray~|F|
  12586. ~lightGray~|~red~^~lightGray~|F|F|F|F|F|F| ~red~^ = Turtle
  12587. ~yellow~Fence/Wall follows land contours
  12588. ]] --
  12589.     info.sub[26] =
  12590. [[~yellow~A rectangular walled area of forest:
  12591. Place me at ~red~^
  12592.  
  12593. ~lightGray~|F|F|F|F|F|F|F| F = Fence or wall
  12594. ~lightGray~|F|~brown~B~lime~| | | |~brown~B~lightGray~|F| ~brown~B = Barrel (corners)
  12595. ~lightGray~|F|~lime~ | | | | ~lightGray~|F|
  12596. ~lightGray~|F|~lime~ | | | |~cyan~T~lightGray~|F| ~cyan~T = Tree
  12597. ~lightGray~|F|~lime~ |~cyan~T~lime~| | | ~lightGray~|F|
  12598. ~lightGray~|F|~lime~ | | | | ~lightGray~|F|
  12599. ~lightGray~|F|~red~^~lime~| |~cyan~T~lime~| |~brown~B~lightGray~|F| ~red~^ = Turtle ~brown~(on Barrel)
  12600. ~lightGray~|F|F|F|F|F|F|F|
  12601.  
  12602. ]] -- manage walled forest
  12603.     info.sub[31] =
  12604. [[~yellow~Place me on the ground as below ~red~^
  12605.  
  12606. ~lightGray~|*|~green~ | | |   ~lightGray~|*| ~green~| | ~lightGray~* = Wall
  12607. ~lightGray~|*|~green~ | | |   ~lightGray~|*| ~green~| |
  12608. ~lightGray~|*|~green~ | | |   ~lightGray~|*| ~green~| |
  12609. ~lightGray~|*|~green~ | | |   ~lightGray~|*| ~green~| |
  12610. ~lightGray~|*|~green~ | | |   ~lightGray~|*| ~green~| | ~magenta~M = Modem
  12611. ~brown~|B|B~green~| | |~lightGray~or |*| ~green~| | ~brown~B = Barrel or Chest
  12612. ~brown~|B|~blue~W~green~| | |~lightGray~or ~magenta~|M|~blue~W~green~| | ~blue~W = Water
  12613. ~lime~|~red~^~lime~|~brown~B|B|~lightGray~*|or ~brown~|~red~^~brown~|~magenta~M|~lightGray~*| ~red~^ = Turtle
  12614. ~brown~Legacy   ~lightGray~or ~magenta~Network ~yellow~Size = 12 x 12
  12615. ]] -- Create modular crop farm
  12616.     info.sub[32] =
  12617. [[~yellow~Place on any ~brown~T ~blue~T ~magenta~T~yellow~ facing < > ^ v
  12618.  
  12619. ~lightGray~|*|~green~ | | |   ~lightGray~|*| ~green~| | ~lightGray~* = Wall
  12620. ~lightGray~|*|~green~ | | |   ~lightGray~|*| ~green~| |
  12621. ~lightGray~|*|~green~ | | |   ~lightGray~|*| ~green~| |
  12622. ~lightGray~|*|~green~ | | |   ~lightGray~|*| ~green~| | ~magenta~M = Modem
  12623. ~lightGray~|*|~green~ | | |   ~lightGray~|*| ~green~| | ~brown~B = Barrel or Chest
  12624. ~brown~|B|B~green~| | |~lightGray~or |*| ~green~| | ~blue~W = Water
  12625. ~brown~|B~blue~|T| ~green~| |~lightGray~or ~magenta~|M|~blue~T~green~| | ~orange~S = Sapling or Tree
  12626. ~orange~|S|~brown~T|B|~lightGray~*|or ~brown~|B~magenta~|T|~lightGray~*| ~brown~T ~lightGray~= Turtle (~blue~T ~magenta~T~lightGray~)
  12627. ~brown~Legacy   ~lightGray~or ~magenta~Network ~yellow~Size = 12 x 12
  12628. ]] -- Extend farm
  12629.     info.sub[33] =
  12630. [[~yellow~Place as below ~blue~T~yellow~ facing < > ^ v
  12631.  
  12632. ~lightGray~|*|~green~ | | |   ~lightGray~|*| ~green~| | ~lightGray~* = Wall
  12633. ~lightGray~|*|~green~ | | |   ~lightGray~|*| ~green~| |
  12634. ~lightGray~|*|~green~ | | |   ~lightGray~|*| ~green~| |
  12635. ~lightGray~|*|~green~ | | |   ~lightGray~|*| ~green~| | ~magenta~M = Modem
  12636. ~lightGray~|*|~green~ | | |   ~lightGray~|*| ~green~| | ~brown~B = Barrel or Chest
  12637. ~brown~|B|B~green~| | |~lightGray~or |*| ~green~| | C = Crop (first)
  12638. ~brown~|B|~blue~T~green~|C| |~lightGray~or ~magenta~|M|~blue~T~green~|C| ~orange~S = Sapling or Tree
  12639. ~orange~|S|~brown~B|B|~lightGray~*|or ~magenta~|M|~brown~B|~lightGray~*| ~blue~T = Turtle on Water
  12640. ~yellow~Options to harvest or manage to follow.
  12641. ]] -- Manual harvest and auto setup
  12642.     info.sub[34] =
  12643. [[~yellow~Build a ~brown~fence ~yellow~or ~lightGray~wall ~yellow~to chosen length.
  12644.  
  12645. Turtle goes ~orange~BACKWARDS ~yellow~when started.
  12646.  
  12647. Start: length = 6, Turtle facing right
  12648.  
  12649. ~lightGray~| |~red~>~lightGray~| | | | | | | ~red~> = Turtle
  12650.  
  12651. Finish:
  12652.  
  12653. ~lightGray~| |F|F|F|F|F|F| | F = Fence or Wall
  12654. ]] -- build fence or wall
  12655.     info.sub[35] =
  12656. [[~yellow~Build a ~brown~fence ~yellow~or ~lightGray~wall ~yellow~
  12657. rectangular area.
  12658.  
  12659. ~lightGray~| | | | | | | | ~brown~F = Fence or Wall
  12660. ~lightGray~| ~brown~|F|F|F|F|F~lightGray~| |
  12661. ~lightGray~| ~brown~|F| | | |F~lightGray~| |
  12662. ~lightGray~| ~brown~|F| | | |F~lightGray~| |
  12663. ~lightGray~| ~brown~|F| | | |F~lightGray~| |
  12664. ~lightGray~| |~red~^~brown~|F|F|F|F| | ~red~^ = Turtle
  12665. ~lightGray~| | | | | | | |
  12666.  
  12667. ]] -- build rectangular fence or wall
  12668.     info.sub[41] =
  12669. [[~yellow~Place me on any block on the left
  12670. side facing the obsidian field.
  12671.  
  12672. ~gray~|O|O|O|O|O| O = Obsidian
  12673. |O|O|O|O|O|
  12674. |O|O|O|O|O|
  12675. |O|O|O|O|O|
  12676. |~red~^~gray~|O|O|O|O| ~red~^ = Turtle
  12677. |~red~^~gray~| | | | |
  12678.  
  12679. ]] -- Harvest obsidian
  12680.     info.sub[42] =
  12681. [[~yellow~Place me on the ground as below ~red~> ~pink~^
  12682. ~yellow~Start ~red~> ~yellow~(facing right) or ~pink~^ ~yellow~ahead
  12683.  
  12684. ~lightGray~| |~red~>~lightGray~|O|O|*| | |    |*|O|O|*| face view            
  12685.    ~pink~^~lightGray~               |O|~purple~+ +~lightGray~|O|
  12686.                    |O|~purple~+ +~lightGray~|O|
  12687. ~yellow~Result (plan)      ~lightGray~|O|~purple~+ +~lightGray~|O|
  12688. ~lightGray~| |O|O|O|O| | |    |*|O|O|*|
  12689.                    ~green~--------- ground
  12690. ~yellow~width=4, ~orange~height=5 ~red~(frame size)
  12691. ]] -- build Nether portal
  12692.     info.sub[43] = info.sub[42] -- Demolish Nether portal
  12693.     info.sub[44] =
  12694. [[~yellow~Press F3+G for chunk boundaries
  12695. Netherite stripmine plan view
  12696.                
  12697. ~lightGray~|*|*|*~lime~N~lightGray~*|*|*|
  12698. ~lightGray~|*|*|*~lime~|~lightGray~*|*|*| ~lime~-| = Chunk Boundaries
  12699. ~lightGray~|*|~red~^~lightGray~|*~lime~|~lightGray~*|~red~^~lightGray~|*|  ~red~^ = Turtle
  12700. ~lime~W*-*-*~blue~+~lime~*-*-*E  ~blue~+ = Boundaries junction
  12701. ~lightGray~|*|*|*~lime~|~lightGray~*|*|*|
  12702. ~lightGray~|*|*|*~lime~|~lightGray~*|*|*|  ~yellow~North of WE
  12703. ~lightGray~|*|*|*~lime~S~lightGray~*|*|*|  ~yellow~1 block from NS
  12704. Most generate close to chunk boundaries
  12705. ]] -- Netherite stripping
  12706.     info.sub[45] =
  12707. [[~yellow~Place me on the ground ~red~<~yellow~ facing ~red~West.
  12708.  
  12709. ~lime~x
  12710.                  ~lightGray~N
  12711. ~lime~a~lightGray~   -1        | | | |  
  12712. ~lime~x~lightGray~    0       ~red~W~lightGray~| |~red~<~lightGray~| |E  ~red~< = Turtle
  12713. ~lime~i~lightGray~    1        | | | |  
  12714. ~lime~s~lightGray~                S
  12715.     ~green~z axis    ~lightGray~-1 0 1
  12716.  
  12717. Centre of the dragon arena ~lime~X = 0, ~green~Z = 0
  12718. ~yellow~                           facing ~red~West
  12719. ]] -- Find dragon tower centres
  12720.     info.sub[46] =
  12721. [[~yellow~Place turtle in ceiling facing endstone
  12722.  
  12723. Plan view    Side view
  12724. ~gray~             |*|*|*|*|*|
  12725.     |*|      |*|*|*|*|*|
  12726.   |*|*|*|    |*|*|*|*|*|
  12727. |*|*|*|*|*|  |*|*|*|*|*|  * ~lightGray~= Obsidian
  12728. ~gray~|*|*|~yellow~E~gray~|*|*|  |*|*|*|*|*|  ~yellow~E ~lightGray~= Endstone
  12729. ~gray~  |*|~red~^~gray~|*|    |*|*|~red~T~gray~|*|*|~red~ ^T ~lightGray~= Turtle
  12730. ~gray~    |*|      |*|*| |*|*|
  12731.              |*|*| |*|*|
  12732. ]] -- deactivate dragon tower
  12733.     info.sub[47] =
  12734. [[~yellow~Place turtle on the ground at 100,49,0
  12735.  
  12736. |*|*|*|*|*|*|*|*|*|   ~lightGray~= Dragon Island
  12737.  
  12738.  
  12739. ~yellow~    Facing WEST
  12740. ~gray~    |*|*|*|*|*|     * ~lightGray~= Obsidian plate
  12741. ~gray~    |*|*|*|*|*|
  12742. ~yellow~  S ~gray~|*|*|~red~T~gray~|*|*| ~yellow~N   ~red~T ~lightGray~= Turtle
  12743. ~gray~    |*|*|*|*|*|
  12744.     |*|*|*|*|*|
  12745. ~yellow~         E]] -- build dragon water trap
  12746.     info.sub[48] =
  12747. [[~yellow~Place turtle under end world portal
  12748.  
  12749. ~gray~        |B|     B ~lightGray~= Bedrock
  12750. ~gray~      |B|B|B|
  12751.         ~purple~|P|     P ~lightGray~= Portal
  12752. ~gray~      |B|B|B|
  12753.         |B|
  12754.          ~green~|      
  12755.          ~green~|      ~lime~Height measured first     
  12756.          ~green~|      
  12757. ~red~         T      T ~lightGray~= Turtle 
  12758. ~yellow~  |*|*|*|*|*|*| ~red~Inventory AFTER height
  12759. ]] -- build end portal minecart
  12760.    
  12761.     info.sub[491] =
  12762. [[~yellow~Place turtle <64 blocks from shulker
  12763. Directly below or 1 block in front
  12764. ~lightGray~|*|*|*|*|*|*|*|
  12765.      ~gray~> ~lime~S ~gray~<       > T < = kill positions
  12766.        ~gray~T
  12767.  
  12768.  
  12769.  
  12770.      ~red~> T <       Attacks ahead / above
  12771. ~lightGray~|*|*|*|*|*|*|*|
  12772.  
  12773. ~red~> T <~lightGray~ = Turtle   ~lime~S~lightGray~ = Shulker]] -- Shulker is above
  12774.  
  12775.     info.sub[492] =
  12776. [[~yellow~Place turtle <64 blocks from shulker
  12777. Same level, or 1 block above/below
  12778. Plan view:
  12779.    ~red~>~lightGray~ | | | | | |~lime~S~lightGray~|
  12780.  
  12781. ~yellow~Side view:
  12782.    ~red~T~lightGray~ | | | | | |~gray~T~lightGray~|
  12783.    ~red~T~lightGray~ | | | | |~gray~T~lightGray~|~lime~S~lightGray~|   ~gray~T = kill positions
  12784.    ~red~T~lightGray~ | | | | | |~gray~T~lightGray~|
  12785.    
  12786.  ~red~> T~lightGray~ = Turtle   ~lime~S~lightGray~ = Shulker]] -- Shulker is ahead
  12787.    
  12788.     info.sub[493] =
  12789. [[~yellow~Place turtle <64 blocks from shulker
  12790. Directly above or 1 block in front
  12791. ~lightGray~|*|*|*|*|*|*|*|
  12792.      ~red~> T <       Attacks ahead / below
  12793.  
  12794.  
  12795.  
  12796.        ~gray~T
  12797.      ~gray~> ~lime~S ~gray~<       > T < = kill positions
  12798. ~lightGray~|*|*|*|*|*|*|*|
  12799.  
  12800. ~red~> T <~lightGray~ = Turtle   ~lime~S~lightGray~ = Shulker]] -- Shulker is below
  12801.  
  12802.     info.sub[494] =
  12803. [[~yellow~Place turtle in entry pit centre wall
  12804.  
  12805. ~lightGray~|*|~orange~+~lightGray~| | |~red~+~lightGray~| | |~lime~S~lightGray~|*| ~lime~S~lightGray~ = Shulker
  12806. ~lightGray~|*|~orange~+~lightGray~| |~lime~S|~red~+~lightGray~| | |~blue~+~lightGray~|*| T = turtle places
  12807. ~lightGray~|*|~orange~+~lightGray~| | |~red~+~lightGray~| | |~blue~+~lightGray~|*|
  12808. ~lightGray~|*|~orange~+~lightGray~| | |~red~+~lightGray~| | |~blue~+~lightGray~|*|     ~yellow~Plan view
  12809. ~lightGray~|*|~orange~+|+|+|~red~+|~blue~+|+|+~lightGray~|*|    |*|*|*|*|*|
  12810. ~lightGray~|*|*|*|~orange~+|~red~+|~blue~+~lightGray~|*|*|*|    |*| |~red~T~lightGray~| |*|
  12811. ~lightGray~|*|*|*|~orange~T|~red~T|~blue~T~lightGray~|*|*|*|    |*|~orange~T~lightGray~| |~blue~T~lightGray~|*|
  12812. ~lightGray~|*|*|*| | | |*|*|*|    |*| |~red~T~lightGray~| |*|
  12813.                        |*|*|*|*|*|]] -- Climb tower wall
  12814.  
  12815.     info.sub[495] =
  12816. [[~yellow~Place turtle in corner against ceiling
  12817. Face ~red~direction ~yellow~shown. It will exit up,
  12818. clear the roof above and return
  12819.  
  12820. ~lightGray~|*|*|*|*|*|*|*|*|*|*|  ~yellow~Clears 15x15
  12821. ~lightGray~|*|~red~<~lightGray~| | | | | | |~red~^~lightGray~|*|  ~yellow~area regardless
  12822. ~lightGray~|*| | | | | | | | |*|  ~yellow~of actual size
  12823.  
  12824. ~lightGray~|*| | | | | | | | |*|
  12825. ~lightGray~|*|~red~V~lightGray~| | | | | | |~red~>~lightGray~|*|
  12826. ~lightGray~|*|*|*|*|*|*|*|*|*|*|]] -- Clear roof above
  12827.  
  12828.     info.sub[496] =
  12829. [[~yellow~Place turtle in corner against ceiling
  12830. Face ~red~direction ~yellow~shown. It will exit out,
  12831. clear 4 outside walls and return.
  12832.  
  12833. ~lightGray~|*|*|*|*|*|*|*|*|*|*|  ~yellow~Size calculated
  12834. ~lightGray~|*|~red~<~lightGray~| | | | | | |~red~^~lightGray~|*|  ~yellow~automatically.
  12835. ~lightGray~|*| | | | | | | | |*|
  12836.  
  12837. ~lightGray~|*| | | | | | | | |*|
  12838. ~lightGray~|*|~red~V~lightGray~| | | | | | |~red~>~lightGray~|*|
  12839. ~lightGray~|*|*|*|*|*|*|*|*|*|*|]] -- Clear outside wall
  12840.  
  12841.     info.sub[51] =
  12842. [[~yellow~Place me on the ground as below ~red~^
  12843.  
  12844. ~yellow~Start:~blue~  |-|-|-|    - = air/water
  12845. ~orange~        |-|-|-|    - = lava
  12846. ~blue~        |-|-|-|
  12847. ~lightGray~        |*|~red~^~lightGray~|*|    ~red~^ = Turtle
  12848.  
  12849. ~yellow~Result:~blue~ |-|~lightGray~*|~blue~-|    - = air/water
  12850. ~orange~        |-|~lightGray~*|~orange~-|    - = lava
  12851. ~blue~        |-|~lightGray~*|~blue~-|
  12852. ~lightGray~        |*|*|*|    * = Solid block
  12853. ]] -- Single path
  12854.     info.sub[52] =
  12855. [[~yellow~Place ~red~^ T~yellow~ at start of path or tunnel
  12856.  
  12857. ~yellow~Plan view    Face view
  12858.  
  12859. ~lightGray~|*|~green~*~lightGray~|*|      *|*|*|*|*   ~green~* = new block
  12860. ~lightGray~|*|~green~*~lightGray~|*|      *|*|~green~*~lightGray~|*|*
  12861. ~lightGray~|*|~green~*~lightGray~|*|      *|*| |*|*
  12862. ~lightGray~|*|~green~*~lightGray~|*|      *|*|~red~T~lightGray~|*|*   ~red~^ T = Turtle
  12863.    ~red~^~lightGray~         ~green~- - * - -   ground
  12864.  
  12865. ~green~Floor + ceiling ~yellow~placed for your safety!
  12866. ]] -- Covered walkway / tunnel
  12867.     info.sub[53] =
  12868. [[~yellow~plan view
  12869.  
  12870. ~lightGray~|*|~blue~-|-~lightGray~|*|
  12871. ~lightGray~|*|~blue~-|-~lightGray~|*|    ~yellow~Cross section view
  12872. ~lightGray~|*|~blue~-|-~lightGray~|*|     ~lime~1 ~orange~2 ~brown~3 ~green~4  ~lightGray~= on ground
  12873. ~lightGray~|~lime~^~lightGray~|~blue~^~lightGray~|~cyan~^~lightGray~|~green~^~lightGray~|    |*|~blue~5~lightGray~|~cyan~6~lightGray~|*| ~lightGray~= in water
  12874.  
  12875. ~yellow~New canal       ~lime~1 ~orange~2 ~brown~3 ~green~4 ~yellow~ground level-63
  12876. ~yellow~Extend existing ~lime~1     ~green~4 ~yellow~ground level-63
  12877. ~yellow~Extend existing   ~blue~5 ~cyan~6   ~yellow~water  level-62
  12878. ]] -- new/existing canal
  12879.     info.sub[54] =
  12880. [[~lime~New ~yellow~ice canal or ~lightBlue~convert ~yellow~existing water
  12881.  
  12882. ~lightGray~ |*| | |*|    * = Slab
  12883. ~lightGray~ |~orange~T~lightGray~|~blue~I~lightGray~| |*|    ~blue~I = Ice (packed or blue)
  12884. ~lightGray~ |*| | |*|   | | = Air (empty block)
  12885. ~lightGray~ |*|~blue~I~lightGray~| |~orange~T~lightGray~|   ~orange~T = Torch (optional)
  12886. ~lime~  1 2 3 4    New ice canal
  12887. ~lightBlue~  5 6 7 8    Convert water canal
  12888.  
  12889. ~red~All turtles placed at ground level!
  12890. ~lime~1-4 ~lightGray~for ~lime~new ~lightGray~canal ~lightBlue~5-8 ~lightGray~to ~lightBlue~convert ~lightGray~water
  12891. ]] -- ice canal
  12892.     info.sub[55] =
  12893. [[~yellow~Place ~red~^~yellow~ any level air, water or lava.~lightGray~
  12894.  
  12895. | | | | | | | |
  12896. | |*|*|*|*|*| |  * = Block
  12897. | |*|*|*|*|*| |
  12898. | |*|*|*|*|*| |
  12899. | |*|*|*|*|*| |
  12900. | |~red~^~lightGray~|*|*|*|*| |  ~red~^ = Turtle~lightGray~
  12901. | | | | | | | |
  12902.  
  12903. Blocks placed under the turtle
  12904. ]] -- Platform
  12905.     info.sub[56] =
  12906. [[~yellow~Place ~red~^~yellow~ above water.~lightGray~
  12907. Existing platform replaced below
  12908.  
  12909. | |*|*|*|*|*| |  * = Block
  12910. | |*|*|*|*|*| |
  12911. | |*|*|*|*|*| |
  12912. | |*|*|*|*|*| |
  12913. | |~red~^~lightGray~|*|*|*|*| |  ~red~^ = Turtle~lightGray~
  12914. | | | | | | | |
  12915.  
  12916. Blocks placed under the turtle
  12917. ]] -- Sinking platform
  12918.     info.sub[57] =
  12919. [[~yellow~Boat Lift (Ice or Water)
  12920. Place turtle left side. ~blue~Source~yellow~ to right
  12921.  
  12922. Start~lightGray~        |*|~blue~ | ~lightGray~|*| ~yellow~Finish
  12923. ~lightGray~             |*|~blue~ | ~lightGray~|*|
  12924.              |*|~cyan~S|S|~lightGray~*| ~cyan~S ~lightGray~= Soul sand
  12925. |*|*|*|*|    |*|~brown~S|S~lightGray~|*| ~brown~S = ~lightGray~Sand + gate
  12926. |~red~^~lightGray~|~blue~W|W~lightGray~|*|    |*~blue~|W|W|~lightGray~*| ~red~^ ~lightGray~= Turtle
  12927. |*|~blue~W|W~lightGray~|*|    |*|~blue~W|W~lightGray~|*|
  12928. |*|~blue~ | ~lightGray~|*|    |*|~blue~ | ~lightGray~|*| ~yellow~Ice canal needs
  12929. ~lightGray~|*|~blue~ | ~lightGray~|*|    |*|~blue~ | ~lightGray~|*| ~yellow~2x2 water source
  12930. ]] -- boat bubble lift
  12931.     info.sub[61] =
  12932. [[~red~NOT ~yellow~for Blaze spawners!
  12933. Plan view          Side view~lightGray~
  12934.     T       T      T = Outside dungeon
  12935.  |*|*|*|*|*|*|*|   |*|~brown~1~lightGray~|*|*|*|~yellow~2~lightGray~|*| Top
  12936. T|*| | | | | |*|T  |*| | | | | |*|
  12937.  |*| | |~magenta~5~lightGray~| | |*|   |*| | |~magenta~5~lightGray~| | |*|
  12938.  |*| |~magenta~5~lightGray~|~orange~S~lightGray~|~magenta~5~lightGray~| |*|   |*|~lime~3~lightGray~|~magenta~5|~orange~S~lightGray~|~magenta~5|~green~4~lightGray~|*| Base
  12939.  |*| | |~magenta~5~lightGray~| | |*|
  12940. T|*| | | | | |*|T  * = Dungeon Wall
  12941.  |*|*|*|*|*|*|*|   ~orange~S = Spawner
  12942. ~lightGray~    T       T      T = Turtle ~brown~1 ~yellow~2 ~lime~3 ~green~4
  12943. ~magenta~5~lightGray~ = On top or adjacent (0 chests only)
  12944. ]] -- 9x9 cube round spawner
  12945.     info.sub[62] =
  12946. [[~yellow~Place turtle as indicated:
  12947. Plan view          Side view
  12948.  
  12949. ~lightGray~|*|*|*|*|*|*|*|*|  ~lime~> T = Continue build
  12950. ~lightGray~|*| | | | | | | |  ~red~< T = New build
  12951. ~lightGray~|*| | | | | | | |
  12952. |*| | | |~purple~S~lightGray~| | | |~lime~>~red~<          T
  12953. ~lightGray~|*| | | | | | | |   |*| | | |~purple~S~lightGray~| | | |~red~T
  12954. ~lightGray~|*| | | | | | | |   |*| | | | | | | |
  12955. ~lightGray~|*| | | | | | | |   |*| | | | | | | |
  12956. ~lightGray~|*|*|*|*|*|*|*|*|   |*| | | | | | | |~lime~T
  12957.                                      ~red~T]] -- 9x9 cube round blaze spawner
  12958.     info.sub[63] =
  12959. [[~yellow~ Plan view (truncated)    Side view
  12960.  
  12961. ~lightGray~ |*| | | | | | | | | |*|  |*| | | | | |
  12962. ~red~>~lightGray~|*| | | | |~purple~S~lightGray~| | | | |*|~red~<~lightGray~ |*| | | | |~purple~S~lightGray~|
  12963.  |*| | | | | | | | | |*|  |*| | | | | |
  12964.  |*| | | | | | | | | |*|  |*| | | | | |
  12965.  |*| | | | | | | | | |*|  |*| | | | | |
  12966.  |*| | | | | | | | | |*|  |*| | | | | |
  12967.  |*|*|*|*|*|*|*|*|*|*|*| ~red~T~lightGray~|*|*|*|*|*|*|
  12968.             ~red~^
  12969. ~purple~S ~lightGray~= Spawner ~red~<> ^ T ~lightGray~= Turtle
  12970. ]] -- Flood spawner chamber
  12971.     info.sub[64] =
  12972. [[~yellow~Bubble lift: Plan view at start~lightGray~
  12973.  |*| | | | | | | | | |*|
  12974.  |*| | | | |~brown~F~lightGray~| | | | |*|
  12975.  |*|*|*|*|*|~brown~F~lightGray~|*|*|*|*|*|  ~brown~F ~lightGray~= Fence
  12976.            ~brown~|~red~^~brown~|~lightGray~= Turtle on ~brown~Soul Sand
  12977.            
  12978. ~yellow~Plan view completed~lightGray~         
  12979.  |*| | | | |~brown~F~lightGray~| | | | |*|
  12980.  |*|*|*|*|*|~brown~F~lightGray~|*|*|*|*|*|  ~brown~F ~lightGray~= Fence
  12981. ~yellow~kill zone~lightGray~|*|~blue~S~lightGray~|*|~yellow~kill zone ~lightGray~Left / Right
  12982.            ~lightGray~|*|        ~blue~S ~lightGray~= Bubble lift
  12983. ]] -- Build bubble tower kill zone
  12984.     info.sub[65] =
  12985. [[~yellow~Computercraft mob grinder
  12986.  
  12987. Plan view~lightGray~ |*~blue~| | | | | | | | | |~lightGray~*|
  12988.           |*~blue~| | | | | | | | | |~lightGray~*|
  12989.           |*|*|*|*|*~blue~| |~lightGray~*|*|*|*|*| Front
  12990. ~red~T~yellow~urtle facing out    ~red~V~lightGray~              
  12991.                          
  12992. ~yellow~Side view ~lightGray~|*|*|*|*|*|*|*|*|*|*|*|
  12993.           |*|*|*|*|*|*|*|*|*|*|*|
  12994.           |*|*|*|*|*|*|*|*|*|*|*|
  12995. ~red~E~yellow~xit hole ~lightGray~|*|*|*|*|*|~red~T~lightGray~|*|*|*|*|*|
  12996.           ~gray~|*|*|*|*|*|*|*|*|*|*|*| Base
  12997. ]] -- Computercraft mob grinder
  12998.     info.sub[66] =
  12999. [[This is a 3 stage process:
  13000. 1.New tower lower base: place me on
  13001. flat ground in open plain.
  13002.  
  13003. 2.Upper base: place me in front of
  13004. the existing chest at tower base.
  13005.  
  13006. 3.Main tower, only needed to respawn
  13007. mobs. Expensive, not recommended.
  13008. Place me in front of chest in ground.
  13009. ]] -- Build endermen observation tower
  13010.     info.sub[67] =
  13011. [[Turtle moves toward spawner
  13012. Place at spawner level.
  13013.  
  13014. Can be behind a wall for protection
  13015. of player.
  13016.  
  13017. Embeds in wall with attack working
  13018.  
  13019. Needs 24 blocks
  13020.  
  13021. ]] -- Build 3*3*3 wall around trial spawner
  13022.  
  13023.     info.sub[71] =
  13024. [[~yellow~Clear field
  13025.  
  13026. ~lightGray~| | | | | |  Remove ~lime~trees ~lightGray~and ~pink~flowers
  13027. ~lightGray~| | | | | |  Fill ~gray~holes
  13028. ~lightGray~| | | | | |  Remove blocks > ground
  13029. | | | | | |
  13030. | | | | | |
  13031. |~red~^~lightGray~| | | | |  ~red~^ ~lightGray~= Turtle position
  13032.  
  13033. ~yellow~Optional use ~brown~dirt ~yellow~as surface
  13034.  
  13035. ]] -- Clear field
  13036.     info.sub[72] =
  13037. [[~yellow~Clear rectangle
  13038.  
  13039. ~lightGray~| | | | | |  Remove all blocks
  13040. ~lightGray~| | | | | |  Optional dig ~lime~up
  13041. ~lightGray~| | | | | |  Optional dig ~blue~down
  13042. ~lightGray~| | | | | |
  13043. | | | | | |
  13044. |~red~^~lightGray~| | | | |  ~red~^ ~lightGray~= Turtle position
  13045.  ~red~^
  13046.  
  13047. ~yellow~Can be used to clear 3 layers at a time
  13048.  
  13049. ]] -- Clear rectangle
  13050.  
  13051.     info.sub[73] =
  13052. [[~yellow~Clear wall
  13053. Plan view         Side view
  13054.                   ~gray~T ~lightBlue~T
  13055. ~red~>~lightGray~|~orange~>~lightGray~|*|*|*|*|*|    ~cyan~T~lightGray~|~blue~T~lightGray~|*|*|*|*|*| Top
  13056. ~lightGray~                   |*|*|*|*|*|*|
  13057.                    |*|*|*|*|*|*|
  13058.                    |*|*|*|*|*|*|
  13059.                    |*|*|*|*|*|*|
  13060.                   ~lime~T~lightGray~|~green~T~lightGray~|*|*|*|*|*| Base
  13061.                  
  13062. T = Turtle top / bottom/ inside / out
  13063. ~yellow~Bottom to top or top to bottom
  13064. ]] -- Clear wall
  13065.     info.sub[74] =
  13066. [[~yellow~Clear rectangle ~red~perimeter only
  13067.  
  13068. ~yellow~Plan view
  13069. ~lightGray~| | | | | |  Remove all blocks
  13070. ~lightGray~| |*|*|*| |  Optional dig ~lime~up
  13071. ~lightGray~| |*|*|*| |  Optional dig ~blue~down
  13072. ~lightGray~| |*|*|*| |
  13073. | |*|*|*| |
  13074. |~red~^~lightGray~| | | | |  ~red~^ ~lightGray~= Turtle position
  13075. ~red~ ^
  13076. ~yellow~Can be used to clear 3 layers at a time
  13077.  
  13078. ]] -- Clear rectangle (perimeter only)
  13079.     info.sub[75] =
  13080. [[~yellow~Demolish cube structure
  13081. Plan view        Side view
  13082.                ~gray~T ~lightBlue~T
  13083. ~lightGray~|*|*|*|*|*|*|  ~cyan~T~lightGray~|~blue~T~lightGray~|*|*|*|*|*|
  13084. |*|*|*|*|*|*|   |*|*|*|*|*|*|
  13085. |*|*|*|*|*|*|   |*|*|*|*|*|*|
  13086. |*|*|*|*|*|*|   |*|*|*|*|*|*|
  13087. |*|*|*|*|*|*|   |*|*|*|*|*|*|
  13088. |~lime~^~lightGray~|*|*|*|*|*|  ~lime~T~lightGray~|~green~T~lightGray~|*|*|*|*|*|
  13089.  ~green~^
  13090. ~lightGray~^ T = Turtle (top/base in/out)
  13091. ]] -- Clear structure floor/walls/ceiling hollow and solid
  13092.     info.sub[76] = info.sub[75]
  13093.     info.sub[77] =
  13094. [[~yellow~Dig a trench (Plan view)
  13095.  
  13096. ~lightGray~      |~red~>~lightGray~| | | | | | |
  13097.  
  13098. | |    ~red~>~lightGray~ = Turtle
  13099. | |    
  13100. | |
  13101. | |
  13102. | |    
  13103. |~red~^~lightGray~|    ~red~^~lightGray~ = Turtle
  13104. ]] -- Dig a trench
  13105.     info.sub[78] =
  13106. [[~yellow~Mountain carving     ~red~T ^ = Turtle
  13107. ~yellow~Side view            Plan view
  13108.  
  13109. ~lightGray~        |*|              |*|*|      ~yellow~L
  13110. ~gray~      |*~lightGray~|*|            ~gray~|*~lightGray~|*|*|*|    ~yellow~E
  13111. ~gray~      |*~lightGray~|*|*|*|      ~gray~|*|*~lightGray~|*|*|*|*|  ~yellow~N
  13112. ~gray~      |*~lightGray~|*|*|*|      ~gray~|*|*~lightGray~|*|*|*|*|  ~yellow~G
  13113. ~gray~      |*~lightGray~|*|*|*|       ~red~^~gray~|*~lightGray~|*|*|*|    ~yellow~T
  13114. ~gray~    |*|*~lightGray~|*|*|*|*|        |*|*|      ~yellow~H
  13115.    ~red~T~gray~|*|*~lightGray~|*|*|*|*|
  13116.     ~yellow~<--> no. of rows <-->
  13117. ]] -- Carve mountain
  13118.     info.sub[79] =
  13119. [[~yellow~Place / Replace floor or ceiling
  13120.  
  13121. Plan view      Side view
  13122. ~lightGray~| | | | | |    |*|*|*|*|*|
  13123. | | | | | |     T          Ceiling
  13124. | | | | | |
  13125. | | | | | |
  13126. | | | | | |     T          Floor
  13127. |~red~^~lightGray~| | | | |    |*|*|*|*|*|
  13128.  
  13129. ~red~^ ~lightGray~T = Turtle position
  13130.  
  13131. ]] -- (Re)place floor or ceiling
  13132.     info.sub[710] =
  13133. [[~yellow~Place me anywhere!
  13134. Menu or direct command interface.
  13135.  
  13136. ~lightGray~Commands:
  13137.  
  13138. direction + ~blue~number ~yellow~eg ~white~f2 ~yellow~= forward ~blue~2
  13139.  
  13140. ~yellow~f = forward  ~orange~b = backward
  13141. ~lime~l = left     ~red~r = right
  13142. ~lightGray~u = up       ~cyan~d = down
  13143.  
  13144. ]] -- Direct control
  13145.     info.sub[81] =
  13146. [[~blue~1.Drop sand or gravel wall
  13147.   ~yellow~Place me on water/lava surface
  13148.  
  13149. ~lightBlue~2.Fill area with sand
  13150.   ~yellow~Place  on left corner of area
  13151.  
  13152. ~yellow~3.Clear ~blue~(1) sand wall
  13153.   ~yellow~Place me on the surface of sand.
  13154.  
  13155. ~yellow~4.Clear ~lightBlue~(2) sand filled area
  13156.   ~yellow~Place on left corner of sand field
  13157. ]] -- Sand based utilities
  13158.     info.sub[82] =
  13159. [[~yellow~Turtle position ~red~> T ~gray~(~red~V ~gray~to enclose)
  13160. ~yellow~Plan view                      ~gray~Start:~red~V
  13161. ~blue~|~red~>~blue~| | | | | | ~yellow~to ~lightGray~|*|*|*|*|*|~red~V~lightGray~|~gray~ enclose
  13162.                             ~gray~*  area
  13163. ~yellow~Side view
  13164.  ~red~T
  13165. ~blue~| | | | | | | ~yellow~to ~lightGray~|*|*|*|*|*|*|
  13166. ~blue~| | | | | | |    ~lightGray~|*|*|*|*|*|*|
  13167. ~blue~| | | | | | |    ~lightGray~|*|*|*|*|*|*|
  13168. ~yellow~|S|~blue~ | | |~yellow~S|S|    |S|~lightGray~*|*|*|~yellow~S|S|
  13169. ~yellow~|S|S|S|S|S|S|    |S|S|S|S|S|S|
  13170. ]] -- build wall from water or lava surface downwards  
  13171.     info.sub[83] =
  13172. [[~yellow~Build a walled rectangle
  13173.  
  13174.  
  13175. ~yellow~L ~lightGray~|*|*|*|*|*|*|
  13176. ~yellow~e ~lightGray~|*| | | | |*|
  13177. ~yellow~n ~lightGray~|*| | | | |*|
  13178. ~yellow~g ~lightGray~|*| | | | |*|
  13179. ~yellow~t ~lightGray~|*|*|*|*|*|*|
  13180. ~yellow~h  ~red~^
  13181. ~yellow~   W i d t h      ~red~^ ~lightGray~= Turtle
  13182.  
  13183.  
  13184. ]] -- Create enclosed area
  13185.     info.sub[84] =
  13186. [[~yellow~Clear volume of water
  13187.  
  13188. Plan view
  13189. ~lightGray~|*|*|*|*|*|*|*|*|  * = Stone
  13190. |*|*~blue~| | |~lightGray~*|*~blue~| |~lightGray~*|  ~blue~| ~lightGray~= Water
  13191. |*~blue~| | | | | | |~lightGray~*|
  13192. |*~blue~| | | | | | |~lightGray~*|
  13193. |*~blue~| | | | | | |~lightGray~*|
  13194. |*~blue~| | | | | |~lightGray~*|*|
  13195. |*|~red~^~lightGray~|*~blue~| | | | |~lightGray~*|  ~red~^ ~lightGray~= Turtle
  13196. |*|*|*|*|*|*|*|*|  ~yellow~Width~blue~: ~yellow~6, ~orange~length~blue~:~orange~ 6
  13197.  
  13198. ]] -- Clear volume of water
  13199.     info.sub[85] = info.sub[56] -- Sinking platform
  13200.     info.sub[861] =
  13201. [[~yellow~Turtle placement  ~red~V~lime~ < ^ > ~yellow~over monument
  13202.  
  13203. ~red~******    ******  * ~lightGray~= Avoid this area
  13204. ~red~******    ******  V do not face front
  13205. ******    ******  ~lime~< ^ > ~lightGray~Ideal facing~lightGray~
  13206. ******    ******
  13207. ~green~******~lime~++++~green~******  ~lime~+ ~lightGray~= Ideal position~green~
  13208. ******~lime~++++~green~******  ~green~* ~lightGray~= Good position
  13209. ~green~****************
  13210. ****************  Or any corner < 12
  13211. ****************  blocks from edge
  13212. ]]-- Ocean monument 85 subChoice 1: 4 corner pillars
  13213.  
  13214.     info.sub[862] =
  13215. [[~yellow~Turtle positions ~red~> ^ < V
  13216.  
  13217. ~lightGray~|*|~red~>~lightGray~|*|*|*|*|*|~red~<~lightGray~|*|  ~red~NOT ~lightGray~corner blocks!
  13218. |~red~V~lightGray~|             |~red~V~lightGray~|
  13219. |*|             |*|
  13220. |*|             |*|
  13221. |*|             |*|
  13222. |*|             |*|
  13223. |*|             |*|  ~yellow~Fill inventory~lightGray~
  13224. |~red~^~lightGray~|             |~red~^~lightGray~|  ~yellow~with stone.~lightGray~
  13225. |*|~red~>~lightGray~|*|*|*|*|*|~red~<~lightGray~|*|  ~yellow~Add when asked
  13226. ]] -- Ocean monument build retaining walls
  13227.  
  13228.     info.sub[863] =
  13229. [[~yellow~Turtle ~red~> < ~yellow~on side walls only
  13230.  
  13231. ~lightGray~|*|*|*|*|*|*|*|*|*|*|
  13232. ~lightGray~|~red~>~cyan~| *****   ***** |~red~<~lightGray~|
  13233. ~lightGray~|~red~>~cyan~| *****   ***** |~red~<~lightGray~|
  13234. ~lightGray~|~red~>~cyan~| *****   ***** |~red~<~lightGray~|
  13235. ~lightGray~|~red~>~cyan~| *****+++***** |~red~<~lightGray~|
  13236. ~lightGray~|~red~>~cyan~| *****+++***** |~red~<~lightGray~|
  13237. ~lightGray~|~red~>~cyan~| ************* |~red~<~lightGray~|
  13238. ~lightGray~|~red~>~cyan~| ************* |~red~<~lightGray~|
  13239. |*|*|*|*|*|*|*|*|*|*|
  13240. ]] -- clear plants before sand draining
  13241.     info.sub[864] = info.sub[863]
  13242.     info.sub[865] = info.sub[863]
  13243.     info.sub[866] = info.sub[56]
  13244.     info.sub[867] =
  13245. [[~yellow~Turtle positions ~red~> ^ < V
  13246. ~green~|*|*|*|*|*~brown~|*|*|*|~red~V~brown~|*|
  13247. ~green~|~red~>~green~|- - - - ~brown~- - - -|*|
  13248. ~green~|*|- - - - ~brown~- - - -|*| ~lightGray~1 Turtle removes
  13249. ~green~|*|- - - - ~brown~- - - -|*| ~lightGray~1 coloured area
  13250. ~green~|*|- - - - ~brown~- - - -|*|
  13251. ~orange~|*|- - - - ~lime~- - - -|*| ~lightGray~6 chests / area
  13252. ~orange~|*|- - - - ~lime~- - - -|*| ~lightGray~Follow turtle
  13253. ~orange~|*|- - - - ~lime~- - - -|*|
  13254. ~orange~|*|- - - - ~lime~- - - -|~red~<~lime~| ~lightGray~30,000 fuel each!
  13255. ~orange~|*|~red~^~orange~|*|*|*~lime~|*|*|*|*|*| ~lightGray~3,000 stone each!
  13256. ]] -- Ocean monument drain and remove 1 of 4 quarters
  13257.     info.sub[87] =
  13258. [[~yellow~Ladder to water / lava: Plan view
  13259.  
  13260.  ~blue~- ~red~- ~blue~- ~red~- ~blue~- ~red~- ~blue~-    ~blue~- ~red~- ~blue~- ~red~- ~blue~- ~red~- ~blue~-
  13261.  ~red~- ~blue~- ~red~- ~blue~- ~red~- ~blue~- ~red~-    ~red~- ~blue~- ~red~-~lightGray~|*|- ~blue~- ~red~-
  13262.  ~blue~- ~red~- ~blue~- ~red~- ~blue~- ~red~- ~blue~-    ~blue~- ~red~-~lightGray~|*|~brown~L~lightGray~|*|~red~- ~blue~- ~~lightGray~
  13263. |*|*|*|~red~^~lightGray~|*|*|*|  |*|*|*|*|*|*|*|
  13264. |*|*|*|*|*|*|*|  |*|*|*|*|*|*|*|
  13265. |*|*|*|*|*|*|*|  |*|*|*|*|*|*|*|
  13266.  
  13267. ~red~^ ~lightGray~= Turtle facing water / lava
  13268. ~brown~L ~lightGray~= Ladder
  13269.  
  13270. ]] -- Ladder to water/lava 
  13271.     info.sub[88] =
  13272. [[~yellow~Place ~red~T~yellow~urtle at water edge.
  13273. Returns max ~blue~d~yellow~epth. ~yellow~Water ~green~p~yellow~lants~yellow~ removed
  13274.  
  13275.    ~red~T                       T
  13276. ~lightGray~|*|*| ~blue~| | | | | | | | | | ~lightGray~|*|
  13277. |*|*| ~blue~| | | | | | | | | ~lightGray~|*|*|
  13278. |*|*| ~blue~| | | | | | | ~lightGray~|*|*|*|*|
  13279. |*|*|*| ~blue~| | |~green~p~blue~| | ~lightGray~|*|*|*|*|*|
  13280. |*|*|*| ~blue~| | |~green~p~lightGray~|*|*|*|*|*|*|*|
  13281. |*|*|*|*~blue~|d~lightGray~|*|*|*|*|*|*|*|*|*|
  13282. |*|*|*|*|*|*|*|*|*|*|*|*|*|*|
  13283. ]] -- Clear water plants from enclosed area
  13284.     info.sub[89] =
  13285. [[Place me on the left corner of the top
  13286. of retaining wall facing water
  13287. ]] -- Convert all water to source
  13288.     info.sub[810] =
  13289. [[Place me on the left corner of the top
  13290. of retaining wall facing water.
  13291. The source blocks are placed ahead to
  13292. selected length
  13293. ]] -- Create sloping water
  13294.  
  13295. -- Building and railway
  13296.     info.sub[91] =
  13297. [[~yellow~Build a wall
  13298.  
  13299. Plan view        Side view
  13300.  
  13301. ~red~>~lightGray~| | | | | | |   |*|*|*|*|*|*|
  13302.                  |*|*|*|*|*|*|
  13303.                  |*|*|*|*|*|*|
  13304.                  |*|*|*|*|*|*|
  13305.                 ~red~T~lightGray~|~red~T~lightGray~|*|*|*|*|*|
  13306.  
  13307. ~red~> T~lightGray~ = Turtle
  13308.                  
  13309. ]] -- Build a wall
  13310.     info.sub[92] =
  13311. [[~yellow~Build a walled rectangle / house
  13312.  
  13313. Plan view         Side view
  13314.  
  13315. ~yellow~L ~lightGray~|*|*|*|*|*|*|   |*|*|*|*|*|*|
  13316. ~yellow~e ~lightGray~|*| | | | |*|   |*|*|*|*|*|*|
  13317. ~yellow~n ~lightGray~|*| | | | |*|   |*|*|*|*|*|*|
  13318. ~yellow~g ~lightGray~|*| | | | |*|   |*|*|*|*|*|*|
  13319. ~yellow~t ~lightGray~|~red~^~lightGray~|*|*|*|*|*|  ~red~T~lightGray~|~red~T~lightGray~|*|*|*|*|*|
  13320. ~yellow~h  ~red~^
  13321. ~yellow~   W i d t h      ~red~^ T ~lightGray~= Turtle
  13322.  
  13323. ]] -- Build a rectangular structure
  13324.     info.sub[93] =
  13325. [[~yellow~Build a gable roof
  13326. Gable built on right side of turtle
  13327. Plan view       End view (width)
  13328.                      ~gray~+      gable top
  13329. ~yellow~L ~lightGray~|*|*|*|*|*|      ~gray~+ + +    gable end
  13330. ~yellow~e ~lightGray~|*| | | |*|    ~red~T~gray~ + + + +  gable end
  13331. ~yellow~n ~lightGray~|*| | | |*|   |*|*|*|*|*| top of wall
  13332. ~yellow~g ~lightGray~|*| | | |*|   |*|*|*|*|*|
  13333. ~yellow~t ~lightGray~|*| | | |*|   |*|*|*|*|*|
  13334. ~yellow~h ~lightGray~|~red~^~lightGray~|*|*|*|*|   |*|*|*|*|*|
  13335. ~yellow~  W i d t h     W i d t h   ~red~^T ~lightGray~= Turtle
  13336. ]] -- Build a gable end roof
  13337.     info.sub[94] =
  13338. [[~yellow~Build a pitched roof
  13339. Width ~red~MUST ~yellow~be ~red~<= ~yellow~Length eg ~red~4~yellow~ x 6
  13340. Plan view        End view (width)
  13341.  
  13342.                   ~red~T ~lightGray~on top of building
  13343. ~yellow~L ~lightGray~|*|*|*|*|      |*|*|*|*|
  13344. ~yellow~e ~lightGray~|*| | |*|      |*|*|*|*|
  13345. ~yellow~n ~lightGray~|*| | |*|      |*|*|*|*|
  13346. ~yellow~g ~lightGray~|*| | |*|      |*|*|*|*|
  13347. ~yellow~t ~lightGray~|*| | |*|
  13348. ~yellow~h ~lightGray~|~red~^~lightGray~|*|*|*|
  13349. ~yellow~  W i d t h     ~red~^ T ~lightGray~= Turtle
  13350. ]]-- Build a pitched roof
  13351.     info.sub[95] =
  13352. [[~yellow~Place me on suspended railway stone
  13353. Redstone torch will go below me
  13354.  
  13355. ~lightGray~_____
  13356. ~lightGray~|*|*|\                           ~red~>~lightGray~|*|
  13357.     |*|~red~<                       ~lightGray~/|*|
  13358.       ~lightGray~|*|______~red~T~lightGray~_________    /~lightGray~|*|~red~!
  13359.        ~red~!~lightGray~|*|*|*|*|*|*|*|*|\ /|*|
  13360.                ~red~!        ~lightGray~|*|*|
  13361.  
  13362. ~red~T < > ~lightGray~= Turtle ~red~! ~lightGray~= Redstone Torch
  13363. On block or above rail, face up slope
  13364. ]] -- Place redstone torch under block
  13365.     info.sub[96] = info.sub[95]
  13366.     info.sub[97] =
  13367. [[~yellow~Place me on last block before up/down
  13368.  
  13369. Build down            Build up~lightGray~
  13370.  
  13371. _____~red~T~lightGray~                        ___
  13372. |*|*|*|_                     _|*|
  13373.       |*|_                 _|*|
  13374.         |*|_             _|*|
  13375.           |*|__________~red~T~lightGray~|*|
  13376.             |*|*|*|*|*|*|      
  13377. ~red~T~lightGray~ = Turtle on block, not above rail
  13378. ]] -- build down
  13379.     info.sub[98] =  info.sub[97]
  13380.  
  13381. --Measurement tools
  13382.     info.sub[101] =
  13383. [[~yellow~Place me on floor.~lightGray~
  13384.             Measured Height:
  13385. |~lightBlue~*~lightGray~|*|*|*|   ~lightBlue~7. Overhead obstruction
  13386.             ~cyan~7. ~red~NOT ~cyan~detect() ~gray~7. Change~lightGray~
  13387.   |*|*|*|
  13388.   |*|*|*|
  13389.   |~lime~S~lightGray~|*|*|   ~lime~4. Specific block found~lightGray~
  13390.   |*|*|*|
  13391.   |*|*|*|
  13392.   |*|*|*|
  13393.  ~red~T~lightGray~|*|*|~red~T~lightGray~|   ~red~T~lightGray~ = Turtle
  13394. ]] -- measure height
  13395.     info.sub[102] =
  13396. [[~yellow~Depth measurement
  13397. Place me on the floor above pit / edge
  13398.  
  13399.     ~red~T~lightGray~
  13400. 1|*|*|
  13401. 2|*|*|
  13402. 3|*|*|
  13403. 4|*|*|         Measured depth:
  13404. 5|*|~lime~S~lightGray~|         ~lime~5. Specific block found
  13405. ~lightGray~6              ~cyan~6. ~red~NOT ~cyan~turtle.detect()~lightGray~
  13406. 7|*|*|~lightBlue~*~lightGray~|*| |   ~lightBlue~6. Obstruction below
  13407. ]] -- measure depth
  13408.     info.sub[103] =
  13409. [[~yellow~Length measurement
  13410.  
  13411. ~lightGray~1 2 3 4 ~lime~5~lightGray~ 6 ~lime~7 ~cyan~8 ~lightBlue~9~lightGray~ 10
  13412.                       ~cyan~8. No block up
  13413. ~lightGray~*|*|*|*|*|*|~lime~S~lightGray~| |*|*|  ~lime~7. Search block
  13414. ~red~T~lightGray~                |*|  ~lightBlue~9. Obstruction
  13415. ~lightGray~*|*|*|*|~lime~S~lightGray~|*|*| |*|*|  ~lime~5. Search block
  13416.                       ~cyan~8. No block down
  13417.  
  13418. ~red~T ~lightGray~= Turtle
  13419.  
  13420. ]] -- measure length
  13421.     info.sub[104] = info.sub[88] -- measure deepest section of water
  13422.     info.sub[105] =
  13423. [[~yellow~Place turtle anywhere:
  13424.  
  13425.  1. make a ~blue~borehole~yellow~ to chosen level.
  13426.  
  13427.  2. Write a report called:
  13428.  
  13429.  3. ~lime~borehole~blue~X~lime~.txt ~yellow~( ~blue~X ~yellow~= computer ID )
  13430.  
  13431.  4. ~orange~Return home
  13432.  
  13433. ]] -- Borehole: Analyse blocks below
  13434.     info.sub[110] =
  13435. [[~yellow~How many turtles?:
  13436.  
  13437. ~yellow~1.~lightGray~ Alternative positions for ~lime~2 ~lightGray~turtles
  13438. ~lightGray~| |~lime~1~lightGray~|~lime~2~lightGray~| |  |~lime~1~lightGray~| | |~lime~2~lightGray~| | | | | | ground 2
  13439. ~lightGray~|*|~blue~W|W~lightGray~|*|  |*|~blue~W|W~lightGray~|*| |*|~blue~1~lightGray~|~blue~2~lightGray~|*| water  2
  13440.  
  13441. ~yellow~2.~lightGray~ Alternative positions for ~orange~4 ~lightGray~turtles
  13442. ~lightGray~|~orange~1~lightGray~|~orange~2~lightGray~|~orange~3~lightGray~|~orange~4~lightGray~|                      ground 4
  13443. ~lightGray~|*|~blue~W|W~lightGray~|*|                      water  0
  13444.  
  13445. ~yellow~Enter 1 to use ~lime~2 ~yellow~turtles or 2 to use ~orange~4]]  -- initial menu for water canal
  13446.     local line = menu.clear()
  13447.     if menuLevel == 1 then -- general help
  13448.         line = menu.colourText(line, info.main[menuItem], true, true)
  13449.         term.write("Enter to continue ")
  13450.         read()
  13451.         return ""
  13452.     else -- item specific help
  13453.         if info.sub[menuItem] ~= nil then -- help file exists
  13454.             line = menu.colourText(line, info.sub[menuItem], true, true)
  13455.             if noMenu then
  13456.                
  13457.  
  13458.                 if getInteger then
  13459.                     return line -- ready for further input
  13460.                 else
  13461.                     term.setCursorPos(1, 13)
  13462.                     menu.enterToContinue()
  13463.                 end
  13464.             else
  13465.                 term.setCursorPos(1, 13)
  13466.                 term.write("Enter=exit, Any key + Enter=more ")
  13467.                 return read()
  13468.             end
  13469.         end
  13470.     end
  13471. end
  13472.  
  13473. local function getTaskOptions()
  13474.     local options = {}
  13475.     options.main =
  13476.     {
  13477.         "Mining (includes Nether)",
  13478.         "Forestry",
  13479.         "Farming",
  13480.         "Obsidian, Nether & End",
  13481.         "Canal, bridge and walkway",
  13482.         "Spawner farm tools",
  13483.         "Area shaping and clearing",
  13484.         "Lava and Water",
  13485.         "Building and minecart",
  13486.         "Measuring tools",
  13487.         "Network Tools"
  13488.     }
  13489.     table.insert(options,
  13490.     {
  13491.         "Ladder up or down",
  13492.         "Stairs up or down",
  13493.         "Create mine at this level",
  13494.         "Safe drop to water block",
  13495.         "Single column bubble lift",
  13496.         "QuickMine corridor system",
  13497.         "QuickMine rectangle",
  13498.         "Mine bedrock level",
  13499.         "Rob disused mineshaft"
  13500.     })
  13501.     table.insert(options,
  13502.     {
  13503.         "Fell Tree",
  13504.         "Create extend or convert tree farm",
  13505.         "Plant tree farm",
  13506.         "Harvest tree farm",
  13507.         "Fence or wall a forest",
  13508.         "Harvest and replant forest"
  13509.     })
  13510.     table.insert(options,
  13511.     {
  13512.         "Create modular crop farm",
  13513.         "Extend modular crop farm",
  13514.         "Manage farm:plant,harvest,convert",
  13515.         "Build a wall or fence",
  13516.         "Fence or wall an enclosure"
  13517.     }) 
  13518.     table.insert(options,                  
  13519.     {
  13520.         "Dig obsidian field",
  13521.         "Build Nether Portal",
  13522.         "Demolish Nether Portal",
  13523.         "Stripmine Netherite",
  13524.         "Undermine Dragon Towers",
  13525.         "Deactivate Dragon Tower",
  13526.         "Build dragon water trap",
  13527.         "Build portal ladder & platform",
  13528.         "Shulker harvester"
  13529.     })
  13530.     table.insert(options,
  13531.     {
  13532.         "Simple path on air, water or lava",
  13533.         "Covered path or tunnel",
  13534.         "Water canal (mulitple options)",
  13535.         "Ice canal (multiple options)",
  13536.         "Platform",
  13537.         "Sinking platform",
  13538.         "Boat bubble lift",
  13539.     })
  13540.     table.insert(options,
  13541.     {
  13542.         "Cube around spawner (NOT blaze)",
  13543.         "Cube around Blaze spawner",
  13544.         "Flood mob farm floor",
  13545.         "Create mob bubble lift",
  13546.         "Computercraft mob grinder",
  13547.         "Build Endermen observation tower",
  13548.         "Surround trial spawner"
  13549.     })
  13550.     table.insert(options,
  13551.     {
  13552.         "Clear field (inc trees)",
  13553.         "Clear a rectangle (+ u/d opt)",
  13554.         "Clear single wall up/down",
  13555.         "Clear rectangular wall section",
  13556.         "Clear hollow structure up/down",
  13557.         "Clear solid structure up/down",
  13558.         "Dig a trench",
  13559.         "Carve mountain side",
  13560.         "Place a floor or ceiling",
  13561.         "Direct control of movement"
  13562.     })
  13563.     table.insert(options,
  13564.     {
  13565.         "Sand based utilities",
  13566.         "Vertical wall from surface",
  13567.         "Create enclosed area",
  13568.         "Clear volume of water",
  13569.         "Sinking platform",
  13570.         "Ocean monument utilities",
  13571.         "Ladder down to water/lava",
  13572.         "Clear water plants",
  13573.         "Convert all water to source",
  13574.         "Create sloping water"
  13575.     })
  13576.     table.insert(options,
  13577.     {
  13578.         "Build a wall",
  13579.         "Build a walled area / house",
  13580.         "Build a gable end roof",
  13581.         "Build a pitched roof",
  13582.         "Place Redstone:torch level track",
  13583.         "Place Redstone:torch upward track",
  13584.         "Build downward track",
  13585.         "Build upward track"
  13586.     })
  13587.     table.insert(options,
  13588.     {
  13589.         "Measure height",
  13590.         "Measure depth",
  13591.         "Measure length",
  13592.         "Measure greatest depth",
  13593.         "Borehole: Analyse blocks below"
  13594.     })
  13595.     table.insert(options,
  13596.     {
  13597.         "Craft an item",
  13598.         "Mystical Agriculture Essence tools"
  13599.     })
  13600.    
  13601.     return options
  13602. end
  13603.  
  13604. local function getTaskColours()
  13605.     local options = {}
  13606.     options.main =
  13607.     {
  13608.         colors.lightGray,   -- Mining (includes Nether)
  13609.         colors.lime,        -- Forestry
  13610.         colors.green,       -- Farming
  13611.         colors.red,         -- Obsidian, Nether & End Portal
  13612.         colors.brown,       -- Canal, bridge and walkway
  13613.         colors.lightGray,   -- Mob farm tools
  13614.         colors.orange,      -- Area shaping and clearing
  13615.         colors.blue,        -- Lava and Water
  13616.         colors.cyan,        -- Railway
  13617.         colors.purple,      -- Measuring tools
  13618.         colors.pink         -- Network tools
  13619.     }
  13620.     table.insert(options,
  13621.     {
  13622.         colors.brown,       -- Ladder up or down
  13623.         colors.lightGray,   -- Stairs up or down
  13624.         colors.lightGray,   -- Create mine at this level
  13625.         colors.blue,        -- Safe drop to water block
  13626.         colors.blue,        -- Single column bubble lift
  13627.         colors.magenta,     -- QuickMine corridor system
  13628.         colors.pink,        -- QuickMine rectangle
  13629.         colors.gray,        -- Mine bedrock level
  13630.         colors.brown        -- Rob disused mineshaft
  13631.     })
  13632.     table.insert(options,
  13633.     {
  13634.         colors.brown,       -- Fell Tree
  13635.         colors.lightGray,   -- Create tree farm
  13636.         colors.lime,        -- Plant tree farm
  13637.         colors.green,       -- Harvest tree farm
  13638.         colors.brown,       -- Fence or wall an enclosure
  13639.         colors.lime         -- Harvest and replant forest
  13640.     })
  13641.     table.insert(options,
  13642.     {
  13643.         colors.yellow,      -- Create modular crop farm
  13644.         colors.green,       -- Extend modular crop farm
  13645.         colors.lime,        -- Manage modular crop farm
  13646.         colors.brown,       -- Build wall or fence
  13647.         colors.brown        -- Build rectangle wall or fence
  13648.     }) 
  13649.     table.insert(options,                  
  13650.     {
  13651.         colors.lightGray,   -- Dig obsidian field
  13652.         colors.purple,      -- Build Nether Portal
  13653.         colors.gray,        -- Demolish Nether Portal
  13654.         colors.red,         -- Netherite stripping
  13655.         colors.orange,      -- Undermine Dragon Towers
  13656.         colors.orange,      -- Deactivate Dragon Tower
  13657.         colors.blue,        -- Build dragon water trap
  13658.         colors.lightGray,   -- Build portal minecart station
  13659.         colors.red          -- Shulker harvesting
  13660.     })
  13661.     table.insert(options,
  13662.     {
  13663.         colors.lightGray,   -- Continuous path
  13664.         colors.gray,        -- Covered walkway / tunnel
  13665.         colors.blue,        -- Water canal
  13666.         colors.lightBlue,   -- Ice canal (4 options)
  13667.         colors.brown,       -- Platform
  13668.         colors.blue,        -- Sinking platform for oceans
  13669.         colors.cyan         -- Boat bubble lift
  13670.     })
  13671.     table.insert(options,
  13672.     {
  13673.         colors.brown,       -- Cube around spawner (NOT blaze)
  13674.         colors.red,         -- Cube around Blaze spawner
  13675.         colors.cyan,        -- Flood mob farm floor
  13676.         colors.blue,        -- Create mob bubble lift
  13677.         colors.brown,       -- Dig mob drop trench
  13678.         colors.gray,        -- Build Endermen observation tower
  13679.         colors.orange       -- Surround trial spawner
  13680.     })
  13681.     table.insert(options,
  13682.     {
  13683.         colors.lime,        -- Clear field (inc trees)
  13684.         colors.magenta,     -- Clear a rectangle (+ u/d opt)
  13685.         colors.pink,        -- Clear single wall up/down
  13686.         colors.purple,      -- Clear rectangular wall section
  13687.         colors.brown,       -- Clear hollow structure up/down
  13688.         colors.orange,      -- Clear solid structure up/down
  13689.         colors.brown,       -- Dig a trench
  13690.         colors.gray,        -- Carve mountain side
  13691.         colors.lightBlue,   -- Place a floor or ceiling
  13692.         colors.red          -- Direct control of movement
  13693.     })
  13694.     table.insert(options,
  13695.     {
  13696.         colors.yellow,      -- Sand based utilities
  13697.         colors.blue,        -- Vertical wall from surface
  13698.         colors.blue,        -- Create enclosed area
  13699.         colors.cyan,        -- Clear volume of water
  13700.         colors.lightGray,   -- Sinking platform
  13701.         colors.orange,      -- Ocean monument utilities
  13702.         colors.brown,       -- Ladder down to water/lava
  13703.         colors.green,       -- Clear water plants
  13704.         colors.lightBlue,   -- Convert all water to source
  13705.         colors.blue         -- Create sloping water
  13706.     })
  13707.     table.insert(options,
  13708.     {
  13709.         colors.yellow,      -- build a wall
  13710.         colors.orange,      -- build rectangular structure
  13711.         colors.lightGray,   -- gable end roof
  13712.         colors.gray,        -- pitched roof
  13713.         colors.red,         -- Place Redstone:torch level track
  13714.         colors.red,         -- Place Redstone:torch upward track
  13715.         colors.orange,      -- Build downward track
  13716.         colors.brown        -- Build upward track
  13717.     })
  13718.     table.insert(options,
  13719.     {
  13720.         colors.red,         -- Measure height
  13721.         colors.purple,      -- Measure depth
  13722.         colors.magenta,     -- Measure length
  13723.         colors.pink,        -- Measure greatest depth
  13724.         colors.lightBlue    -- Borehole: Analyse blocks below
  13725.     })
  13726.     table.insert(options,
  13727.     {
  13728.         colors.red,         -- Craft item
  13729.         colors.magenta,     -- MysticalAgriculture
  13730.     })
  13731.     return options
  13732. end
  13733.  
  13734. local function chooseTask(R)
  13735. --[[
  13736.     called from main()
  13737.     R = chooseTask(R)
  13738.     coloured menu: 
  13739.     local pp =
  13740.     {
  13741.         prompt = colors.green,
  13742.         itemColours = {colors.white, colors.red, ... one for each item},
  13743.         menuPrompt = colors.yellow
  13744.     }
  13745.     choice, modifier = menu.menu or menu.new("Choose your option", options, pp)
  13746.     ]]
  13747.     local lib = {}
  13748.    
  13749.     function lib.getHelp(choice, itemsRequired, pp)
  13750.         local listOfItems = itemsRequired[choice]
  13751.         local result = getTaskHelp(2, choice)
  13752.         if result ~= "" then -- any key + enter = more
  13753.             lib.showItems(listOfItems)
  13754.         end
  13755.         T:clear()
  13756.         pp.itemColours = {colors.green, colors.yellow, colors.orange, colors.red}
  13757.         local options ={"Continue with selected task", "return to menu", "Return to main menu","Quit application"}
  13758.         local option = menu.menu("Choose your option", options, pp, "Type number + Enter ")
  13759.         if option == 2 then
  13760.             choice = -2 -- return to subMenu
  13761.         elseif option == 3 then
  13762.             choice = 0 -- return to main menu
  13763.         elseif option == 4 then
  13764.             choice = -1 -- quit
  13765.         -- else return original value of choice
  13766.         end
  13767.        
  13768.         return choice -- if R.choice reset to 0, then re-starts, -1 == quit
  13769.     end
  13770.    
  13771.     function lib.showItems(listOfItems)
  13772.         T:clear()
  13773.         if listOfItems ~= nil then
  13774.             if next(listOfItems) ~= nil then
  13775.                 menu.colourPrint("Items required:", colors.lime)
  13776.                 for k,v in pairs(listOfItems) do
  13777.                     menu.colourPrint("  "..v, colors.lightGray)
  13778.                 end
  13779.             else
  13780.                 menu.colourPrint("No items required", colors.lime)
  13781.             end
  13782.         else
  13783.             menu.colourPrint("No items required", colors.lime)
  13784.         end
  13785.         menu.colourWrite("Enter to continue", colors.yellow, colors.black, nil, true)
  13786.         read()
  13787.     end
  13788.    
  13789.     function lib.getItemList(subChoice, itemsRequired)
  13790.         local listOfItems = itemsRequired[subChoice]
  13791.         lib.showItems(listOfItems)
  13792.     end
  13793.    
  13794.     function lib.getChoice(choice, subChoice)
  13795.         if subChoice > 9 then -- need more than 9 sub-options
  13796.             choice = subChoice + (choice * 100) -- 10 = 10 + 1000 = 1100
  13797.         else -- eg main option 1 (mining), sub-option 1 (create mine) = 1 + 10 = 11
  13798.             choice = subChoice + (choice * 10) -- 9 = 9 + 9 * 10 = 99, 1 + 10*10 = 101
  13799.         end
  13800.         return choice
  13801.     end
  13802.    
  13803.     local modifier = ""
  13804.     local subChoice = 0
  13805.     local prompt = "Choose an option"
  13806.     local mainPrompt = "Choose option (+h = help eg 3h):"
  13807.     local subPrompt = "Choose (+h = help +i = items eg 3h):"
  13808.     local options = getTaskOptions()
  13809.     local menuColours = getTaskColours()
  13810.     local itemsRequired = getTaskItemsList()
  13811.     local menuState = 0 -- use main menu
  13812.     -- local pp = utils.getPrettyPrint(colors.cyan, colors.magenta) -- specify menu title and prompt colours
  13813.     --local pp = utils.getPrettyPrint()-- uses default colours
  13814.    
  13815.     while menuState >= 0 do -- menuState has to be -1 to exit loop
  13816.         --[[
  13817.         menu returns choice=nil if user enters just q or h
  13818.         menu returns choice=2, modifier = "h" if 2h entered
  13819.         Mining (includes Nether)        white
  13820.         Forestry                        lime
  13821.         Farming                         lime
  13822.         Obsidian, Nether & End Portal   brown
  13823.         Canal, bridge and walkway       pink   
  13824.         Mob farm tools                  gray
  13825.         Area shaping and clearing       purple 
  13826.         Lava and Water                  blue
  13827.         Railway                         magenta
  13828.         Measuring tools                 cyan       
  13829.         ]]
  13830.        
  13831.         if menuState == 0 then
  13832.             pp.itemColours = menuColours.main
  13833.             local mainChoice, modifier = menu.menu(mainPrompt, options.main, pp) -- open main menu options
  13834.             T:saveToLog("chooseTask(): mainChoice = "..tostring(mainChoice)..", modifier = "..tostring(modifier))
  13835.             if modifier == "q" then
  13836.                 R.choice = -1
  13837.                 return R -- quit application
  13838.             elseif modifier == "h" then -- get help on main menu items
  13839.                 getTaskHelp(1, mainChoice) -- "" or any return string for more
  13840.                 R.choice = 0
  13841.                 return R -- quit task system so it re-starts as help menu chosen R.choice = 0
  13842.             else
  13843.                 menuState = 1 -- next level so use sub-menu
  13844.                 R.choice = mainChoice -- 1-11
  13845.             end
  13846.         else
  13847.             pp.prompt = colors.yellow
  13848.             pp.itemColours = menuColours[R.choice]
  13849.             subChoice, modifier = menu.menu(subPrompt, options[R.choice], pp, "Back = 'q' or number + Enter: ") -- open submenu options
  13850.             T:saveToLog("chooseTask(): subChoice = "..tostring(subChoice)..", modifier = "..tostring(modifier))
  13851.             if modifier == "q" then
  13852.                 R.choice = 0    -- return to root menu. change this to -1 if quit application preferred
  13853.                 return R -- quit task system so it re-starts first menu
  13854.             elseif modifier == "i" then -- get list of items
  13855.                 local index = lib.getChoice(R.choice, subChoice)
  13856.                 lib.getItemList(index, itemsRequired) -- list of items shown
  13857.                 subChoice = nil -- allow menu to continue
  13858.             elseif modifier == "h" then -- get help on main menu items
  13859.                 local index = lib.getChoice(R.choice, subChoice)
  13860.                 index = lib.getHelp(index, itemsRequired, pp)
  13861.                 --R = lib.getHelp(R, itemsRequired, pp) -- if R.choice reset to 0 then will continue
  13862.                 if index == -2 then -- redraw sub-menu
  13863.                     subChoice = nil -- allow menu to continue
  13864.                 elseif index == -1 then -- if R.choice reset to 0 then will continue
  13865.                     R.choice = -1
  13866.                     return R
  13867.                 elseif index == 0 then
  13868.                     R.choice = 0
  13869.                     menuState = 0 -- if R.choice reset to 0 then back to main menu
  13870.                 end
  13871.             end
  13872.             if subChoice ~= nil then
  13873.                 R.choice = lib.getChoice(R.choice, subChoice)
  13874.                 return R -- exit loop
  13875.             end
  13876.         end
  13877.     end
  13878.     return R
  13879. end
  13880.  
  13881. local function getTask(R)
  13882.     --[[
  13883.     Called from main(): R = getTask(R)
  13884.     local R =
  13885.         {
  13886.             choice = 0,
  13887.             subChoice = 0,
  13888.             size = 0,
  13889.             width = 0,
  13890.             length = 0,
  13891.             height = 0,
  13892.             depth = 0,
  13893.             up = false,
  13894.             down = false,
  13895.             silent = false,
  13896.             data = {},
  13897.             torchInterval = 0,
  13898.             useBlockType = "",
  13899.             auto = false,
  13900.             side = "",
  13901.             direction = "",
  13902.             mysticalAgriculture = false
  13903.         }
  13904.     ]]
  13905.     local lib = {}
  13906.    
  13907.     function lib.isAutoHelp(choice, noAutoHelp)
  13908.         for _, v in ipairs(noAutoHelp) do
  13909.             if choice == v then
  13910.                 return true
  13911.             end
  13912.         end
  13913.         return false
  13914.     end
  13915.    
  13916.     function lib.isMysticalAgriculture(R)
  13917.         if menu.getBoolean("Mystical Agriculture modded? (y/n)", nil, colors.brown) then
  13918.             R.mysticalAgriculture = true
  13919.         end
  13920.         return R
  13921.     end
  13922.    
  13923.     local prompt = "Choose an option"
  13924.     local noAutoHelp = {23, 24, 47, 49, 53, 54, 81, 85}
  13925.     if not lib.isAutoHelp(R.choice, noAutoHelp) then -- exclude Water canal, ice canal, sand utilities, monument utilities, dragon trap
  13926.         getTaskHelp(2, R.choice, true)
  13927.         menu.clear()
  13928.     end
  13929. -- 01. MINING
  13930.     if R.choice >= 11 and  R.choice <= 13 then  -- createLadder, createStaircase, createMine)
  13931.         --getTaskHelp(2, R.choice, true)
  13932.         --menu.clear()
  13933.     elseif R.choice == 14 then  -- safe drop to water
  13934.         local currentLevel = menu.getInteger("Current level (F3->Y coord)? ", bedrock + 5, ceiling, nil, colors.lightGray)
  13935.         local destLevel = menu.getInteger("Go down to level? ("..currentLevel - 2 .." to "..bedrock + 5 ..")", bedrock + 5 , currentLevel - 2, nil, colors.blue, nil, bedrock + 5)
  13936.         R.height    = math.abs(destLevel - currentLevel)
  13937.     elseif R.choice == 15 then  -- bubble lift
  13938.         R.currentLevel = menu.getInteger("Current level (F3->Y coord)? ", bedrock + 5 , ceiling, nil, colors.blue, nil, bedrock + 5)
  13939.         local destLevel = menu.getInteger("Go up to level? ("..R.currentLevel + 2 .." to "..ceiling ..") ", R.currentLevel + 2, ceiling, nil, colors.lightGray)
  13940.         --[[if currentLevel <= bedrock + 5 then
  13941.             if menu.getBoolean("Confirm close to bedrock (y / n) ", nil, colors.yellow, colors.black) then
  13942.                 R.data = "bedrock"
  13943.             end
  13944.         end]]
  13945.         R.height    = math.abs(destLevel - R.currentLevel)
  13946.     elseif R.choice == 16 then -- create mining corridor system default: square 17 x 17
  13947.         local choices = {"At corridor start, on the floor",
  13948.                          "At corridor start, on the ceiling",
  13949.                          "On floor, move forward to start",
  13950.                          "On ceiling, move forward to start"}
  13951.         pp.itemColours = {colors.lime, colors.green, colors.magenta, colors.pink}
  13952.         local userChoice, modifier = menu.menu("Starting position?", choices, pp, "Type number + Enter ") -- 1 to 4
  13953.         if modifier == "q" then -- quit chosen
  13954.             R.data = "quit"
  13955.             return R
  13956.         end
  13957.         R.width     = menu.getInteger("Width (2-64 default 17) ", 2, 64, nil, colors.yellow, nil, 17)
  13958.         R.length    = menu.getInteger("Length (2-64 default 17) ", 2, 64, nil, colors.orange, nil, 17)
  13959.         R.torchInterval = 9 -- 8 spaces between torches
  13960.         if mcMajorVersion >= 1.18 then
  13961.             R.torchInterval = 17 -- 16 spaces between torches
  13962.         end
  13963.         R.subChoice = userChoice
  13964.     elseif R.choice == 17 then -- clear a rectangle, fill empty spaces above
  13965.         local choices = {"At mine area start, on the floor",
  13966.                          "At mine area start, on the ceiling",
  13967.                          "On floor, move forward to start",
  13968.                          "On ceiling, move forward to start",
  13969.                          "On floor diagonally to left"} -- just finished corridor
  13970.         pp.itemColours = {colors.lime, colors.green, colors.magenta, colors.pink, colors.red}
  13971.         local userChoice, modifier = menu.menu("Starting position?", choices, pp, "Type number + Enter ") -- 1 to 4
  13972.         if modifier == "q" then -- quit chosen
  13973.             R.data = "quit"
  13974.             return R
  13975.         end
  13976.         R.width     = menu.getInteger("Width (2-64 default 15) ", 2, 64, nil, colors.yellow, nil, 15)
  13977.         R.length    = menu.getInteger("Length (2-64 default 15) ", 2, 64, nil, colors.orange, nil, 15)
  13978.         R.subChoice = userChoice
  13979.     elseif R.choice == 18 then -- Mine bedrock area
  13980.         --getTaskHelp(2, R.choice, true)
  13981.         --menu.clear()
  13982.         R.width     = menu.getInteger("Width (2-64 default 17) ", 2, 64, nil, colors.yellow, nil,  17)
  13983.         R.length    = menu.getInteger("Length (2-64 default 17) ", 2, 64, nil, colors.orange, nil,  17)
  13984.         if menu.getBoolean("Leave bedrock exposed? (y/n) ", nil, colors.yellow) then
  13985.             R.data = "leaveExposed"
  13986.         end
  13987.     elseif R.choice == 19 then -- salvage mine shaft
  13988.         R.torchInterval = menu.getInteger("Torch spacing? (0-64)", 0, 64, nil, colors.red)
  13989. -- 02. FORESTRY
  13990.     elseif R.choice == 22 then --Create treefarm
  13991.         R.width = 15
  13992.         R.length = 15
  13993.         pp.itemColours = {colors.lightGray, colors.green, colors.orange, colors.yellow, colors.gray, colors.lightGray}
  13994.         choices = {"New tree farm. Starting here", "On left side of existing farm <-",
  13995.                     "On right side of existing farm ->", "On back left of existing farm ^",
  13996.                     "Convert legacy with storage", "Convert Legacy without storage"}
  13997.         local choice = menu.menu(prompt, choices, pp, "Type number + Enter ")
  13998.         if choice == 1 then
  13999.             R.data = "new"
  14000.         elseif choice == 2 then
  14001.             R.data = "left"
  14002.         elseif choice == 3 then
  14003.             R.data = "right"
  14004.         elseif choice == 4 then
  14005.             R.data = "back"
  14006.         elseif choice == 5 then
  14007.             R.data = "convertStorage"
  14008.         elseif choice == 6 then
  14009.             R.data = "convert"
  14010.         end
  14011.         R.networkFarm = true
  14012.         --R.networkFarm =  menu.getBoolean({"Automate this farm with network","storage and modem? (y/n)"}, nil, {colors.yellow, colors.orange}, colors.black)
  14013.         if R.data ~= "convert" and R.data ~= "convertStorage" then
  14014.             R.up = menu.getBoolean({"Any blocks/trees above current level","in a 15 x 15 block area (y/n) "}, nil, {colors.yellow, colors.orange}, colors.black)
  14015.         end
  14016.     elseif R.choice == 23 then -- plant treefarm
  14017.         R = utils.assessTreeFarm(R) -- sets network and sapling type
  14018.         if R.message ~= "" then
  14019.             return {R.message}  -- location error
  14020.         end
  14021.         if R.networkFarm then
  14022.             if R.logType == "mangrove" then
  14023.                 R.subChoice = 3 -- mangrove
  14024.             elseif R.logType == "spruce" or R.logType == "dark_oak" then
  14025.                 R.subChoice = 2 -- double trees
  14026.             else
  14027.                 R.subChoice = 1 -- single trees
  14028.             end
  14029.         else
  14030.             local choices   = {"16 x oak, birch, acacia, cherry", "4 double spruce, dark oak, jungle"}
  14031.             pp.itemColours = {colors.lime, colors.green}
  14032.             R.subChoice = menu.menu(prompt, choices, pp, "Type number + Enter ")
  14033.         end
  14034.     elseif R.choice == 24 then -- Harvest treefarm
  14035.         R = utils.assessTreeFarm(R) -- sets network and sapling type
  14036.         if R.message ~= "" then
  14037.             return {R.message}  -- location error
  14038.         end
  14039.     elseif R.choice == 25 then -- Build wall or fence
  14040.         R.width     = menu.getInteger("Width of the area (1-64)", 1, 64, nil, colors.yellow)
  14041.         R.length    = menu.getInteger("Length of the area (1-64)", 1, 64, nil, colors.orange)
  14042.         R.torchInterval = menu.getInteger("Torch spacing? (0-64)", 0, 64, nil, colors.red)
  14043.         if menu.getBoolean("Storage barrels in corners? (y/n)", nil, colors.brown) then
  14044.             R.data = "barrel"
  14045.         end
  14046.     elseif R.choice == 26 then -- Harvest and replant natural forest
  14047.  
  14048.        
  14049. -- 03. FARMING
  14050.     elseif R.choice == 31 then -- new crop farm
  14051.         --R = lib.isMysticalAgriculture(R)
  14052.         local choices   = {"Use local storage", "Use modem network storage"}
  14053.         pp.itemColours = {colors.lime, colors.green}
  14054.         local choice = menu.menu(prompt, choices, pp, "Type number + Enter ")
  14055.         if choice == 2 then
  14056.             R.networkFarm = true
  14057.         end
  14058.     elseif R.choice == 32 then -- extend crop farm
  14059.         R = utils.assessFarm(R)-- sets R.networkFarm and R.mysticalAgriculture
  14060.         local choices   = {"Add a farm at the back", "Add a farm to the right"}
  14061.         pp.itemColours = {colors.lime, colors.green}
  14062.         R.subChoice = menu.menu(prompt, choices, pp, "Type number + Enter ")
  14063.     elseif R.choice == 33 then -- manage crop farm
  14064.         R = utils.checkFarmPosition(R)
  14065.         --R = utils.assessFarm(R)-- sets R.networkFarm and R.mysticalAgriculture
  14066.     elseif R.choice == 34 then -- fence or wall
  14067.         R.length    = menu.getInteger("Length of wall / fence (1-256)", 1, 256, nil, colors.orange)
  14068.         R.torchInterval = menu.getInteger("Torch spacing? (0-64)", 0, 64, nil, colors.red)
  14069.     elseif R.choice == 35 then -- rectangle fence or wall
  14070.         R.width     = menu.getInteger("Width of the area (1-64)", 1, 64, nil, colors.yellow)
  14071.         R.length    = menu.getInteger("Length of the area (1-64)", 1, 64, nil, colors.orange)
  14072.         R.torchInterval = menu.getInteger("Torch spacing? (0-64)", 0, 64, nil, colors.red)
  14073.    
  14074. -- 04. OBSIDIAN
  14075.     elseif R.choice == 41 then  -- Harvest obsidian
  14076.         R.width     = menu.getInteger("Width of the area (1-64) ", 1, 64, nil, colors.yellow)
  14077.         R.length    = menu.getInteger("Length of the area (1-64) ", 1, 64, nil, colors.orange)
  14078.     elseif R.choice == 42 or R.choice == 43 then -- build Nether portal / demolish Nether portal
  14079.         local choices = {"Facing portal: forward, turn right",
  14080.                          "Aligned with portal: start ahead"}
  14081.         pp.itemColours = {colors.pink, colors.red}
  14082.         local userChoice, modifier = menu.menu("Starting position?", choices, pp, "Type number + Enter ") -- 1 to 2
  14083.         R.subChoice = userChoice
  14084.         if modifier == "q" then -- quit chosen
  14085.             R.data = "quit"
  14086.             return R
  14087.         end
  14088.         R.length    = menu.getInteger("Portal width", 1, 64, nil, colors.yellow, nil, 4)
  14089.         R.height    = menu.getInteger("Portal height (inc. buried base)", 1, 64, nil, colors.lightGray, nil, 5)
  14090.         R.width     = menu.getInteger("Thickness (Usually 1) Enter = 1)", 1, 64, nil, colors.red, nil, 1)
  14091.         if menu.getBoolean("Base below surface (y/n)", nil, colors.orange) then
  14092.             R.data = "bury"
  14093.         end
  14094.     elseif R.choice == 44 then -- Netherite stripping
  14095.         R.width     = 1
  14096.         while R.length == 0 or R.length % 16 > 0 do
  14097.             R.length = menu.getInteger("Strip length (divisible by 16) ", 16, 512, 1, colors.orange)
  14098.             if R.length % 16 > 0 then
  14099.                 menu.colourText(1, "~red~Must be multiple of 16")
  14100.             end
  14101.         end
  14102.         R.torchInterval = 16
  14103.         R.data = "seal"
  14104.     elseif R.choice == 45 then -- undermine dragon towers
  14105.  
  14106.     elseif R.choice == 46 then -- deactivate dragon tower
  14107.    
  14108.     elseif R.choice == 47 then -- build dragon water trap
  14109.         local text =
  14110. [[~yellow~You ~red~MUST ~yellow~have already constructed a
  14111. bridge / tunnel into the dragon island
  14112. to allow for safe construction of the
  14113. dragon trap.
  14114.  
  14115. If not use toolkit:
  14116.   ~brown~5.~gray~2 Covered path or tunnel~yellow~ length 100
  14117.  
  14118. start centre of obsidian platform:
  14119.   100,49,0 facing west]]
  14120.         menu.colourText(1, text)
  14121.         if menu.getBoolean("Bridge already made? (y/n)", 12, colors.orange) then
  14122.             getTaskHelp(2, R.choice, true)
  14123.             menu.clear()
  14124.         else
  14125.             R.data = "quit"
  14126.             return R
  14127.         end
  14128.     elseif R.choice == 48 then -- build end portal minecart station
  14129.     elseif R.choice == 49 then -- Shulker harvesting
  14130.         local items =
  14131.         {
  14132.             "Shulker is above",
  14133.             "Shulker is ahead",
  14134.             "Shulker is below",
  14135.             "Climb tower wall",
  14136.             "Clear roof above",
  14137.             "Clear outside walls"
  14138.         }
  14139.         pp.itemColours = {colors.lightGray, colors.brown, colors.blue, colors.yellow, colors.orange, colors.lightGray}
  14140.         R.subChoice = menu.menu("Which utility? ", items, pp)
  14141.         getTaskHelp(2, R.choice * 10 + R.subChoice, true)  -- eg 490 + 1 = 491
  14142.         menu.clear()   
  14143.        
  14144. -- 05. CANAL BRIDGE
  14145.     elseif R.choice == 51 then  --single path
  14146.         if menu.getBoolean("Shorten an existing path? (y/n)", nil, colors.brown) then
  14147.             R.data = "reduce"
  14148.         end
  14149.         if R.data == "reduce" then
  14150.             R.length = menu.getInteger("Reduce by? 0 = here to end", 0, 1024, nil, colors.orange, nil, 64)
  14151.         else
  14152.             R.length = menu.getInteger("Path length? 0 = continuous ", 0, 1024, nil, colors.orange, nil, 64)
  14153.             R.torchInterval = menu.getInteger("Torch spacing? (0-64)", 0, 64, line, colors.red, nil, 0)
  14154.         end
  14155.     elseif R.choice == 52 then  -- 2 block corridor / covered walkway
  14156.         R.length    = menu.getInteger("Corridor length?", 1, 1024, nil, colors.orange, nil, 64)
  14157.         R.torchInterval = menu.getInteger("Torch spacing? (0-64)", 0, 64, nil, colors.red)
  14158.         if menu.getBoolean("Seal off water or lava? (y/n)", nil, colors.brown) then
  14159.             R.data = "seal"
  14160.         end
  14161.     elseif R.choice == 53 then  --left/right side of new/existing canal
  14162.         local line = getTaskHelp(2, 110, true, true) -- request line no of help display if needed for getInteger
  14163.         R.data = menu.getInteger("Choose method 1 or 2", 1, 2, line + 1, colors.white)  -- 2 or 4 turtles
  14164.         menu.clear()
  14165.         line = getTaskHelp(2, R.choice, true, true) -- request line no of help display if needed for getInteger
  14166.         R.subChoice = menu.getInteger("Type position of this turtle", 1, 6, line + 1, colors.white)
  14167.         local position = "on ground"
  14168.         local side = "left"
  14169.         local colour = colors.lime
  14170.         if R.subChoice == 1 then
  14171.             R.side = "L"
  14172.             R.height = 1
  14173.             position = "on ground"
  14174.             colour = colors.lime
  14175.         elseif R.subChoice == 2 then
  14176.             R.side = "L"
  14177.             R.height = 1
  14178.             position = "above canal"
  14179.             colour = colors.orange
  14180.         elseif R.subChoice == 3 then
  14181.             R.side = "R"
  14182.             side = "right"
  14183.             R.height = 1
  14184.             position = "above canal"
  14185.             colour = colors.brown
  14186.         elseif R.subChoice == 4 then
  14187.             R.side = "R"
  14188.             side = "right"
  14189.             R.height = 1
  14190.             colour = colors.green
  14191.         elseif R.subChoice == 5 then
  14192.             R.side = "L"
  14193.             R.height = 0
  14194.             position = "on canal base"
  14195.             colour = colors.blue
  14196.         elseif R.subChoice == 6 then
  14197.             R.side = "R"
  14198.             side = "right"
  14199.             R.height = 0
  14200.             position = "on canal base"
  14201.             colour = colors.cyan
  14202.         end
  14203.         line = menu.clear()
  14204.         if R.data == 1 or (R.data == 2 and (R.subChoice == 1 or R.subChoice == 4)) then
  14205.             R.torchInterval = menu.getInteger("Torch spacing? (0-64)", 0, 64, line, colors.red, nil, 0)
  14206.         end
  14207.         -- menu.colourPrint(text, fg, bg, width)
  14208.         -- menu.colourText(row, text, reset)
  14209.         menu.colourPrint("Turtle placed "..side.." "..position, colour)
  14210.         R.length    = menu.getInteger("Canal length? 0 = continuous ", 0, 512, line + 2, colors.orange, nil, 64)
  14211.     elseif R.choice == 54 then  --ice canal 4 sections: edge+torch, edge, centre+ice, centre no ice
  14212.         local line = getTaskHelp(2, R.choice, true, true) -- request line no of help display if needed for getInteger
  14213.         R.subChoice = menu.getInteger("Type turtle position", 1, 8, line, colors.white)
  14214.         R.side = "L"
  14215.         local position = "on towpath"
  14216.         local side = "left"
  14217.         local colour = colors.lime
  14218.         line = menu.clear()
  14219.         --[[
  14220.             new canal
  14221.             1 = left towpath- move to centre left
  14222.             2 = centre left ice or hollow
  14223.             3 = centre right ice or hollow
  14224.             4 = right side - move to centre right
  14225.             convert water canal
  14226.             5 = left towpath
  14227.             6 = centre left ice or air
  14228.             7 = centre right ice or air
  14229.             8 = right towpath
  14230.         ]]
  14231.         if R.subChoice == 1 or R.subChoice == 4 or R.subChoice == 5 or R.subChoice == 8 then
  14232.             -- on towpath r/l
  14233.             R.torchInterval = menu.getInteger("Torch spacing? (0-64)", 0, 64, line, colors.red, nil, 0)
  14234.             if R.subChoice == 4 or R.subChoice == 8 then
  14235.                 side = "right"
  14236.                 R.side = "R"
  14237.             end
  14238.             if R.subChoice <= 4 then -- 1 or 4 only
  14239.                 if menu.getBoolean("Place Ice as well? (y/n)", line + 1, colors.blue) then
  14240.                     R.data = "ice"
  14241.                 end
  14242.             end
  14243.         else -- on canal centre r/l
  14244.             if R.subChoice == 3 or R.subChoice == 7 then
  14245.                 side = "right"
  14246.                 R.side = "R"
  14247.             end
  14248.             if R.subChoice == 2 or R.subChoice == 3 then -- new canal
  14249.                 R.torchInterval = menu.getInteger("Torch spacing? (0-64)", 0, 64, line, colors.red, nil, 0)
  14250.                 position = "above air/ice"
  14251.             end
  14252.             if menu.getBoolean("Place Ice as well? (y/n)", line + 1, colors.blue) then
  14253.                 R.data = "ice"
  14254.                 if R.subChoice == 6 or R.subChoice == 7 then   
  14255.                     position = "alternate air/ice"
  14256.                     colour = colors.lightBlue
  14257.                 end
  14258.             else
  14259.                 if R.subChoice == 6 or R.subChoice == 7 then   
  14260.                     position = "on air spaces"
  14261.                     colour = colors.lighGray
  14262.                     if R.subChoice == 7 then
  14263.                         side = "right"
  14264.                         R.side = "R"
  14265.                     end
  14266.                 end
  14267.             end
  14268.         end
  14269.         R.length    = menu.getInteger("Canal length? 0 = continuous ", 0, 1024, line + 2, colors.orange, nil, 64)
  14270.         menu.colourPrint("Turtle on "..side.." side "..position, colour)
  14271.         sleep(2)
  14272.     elseif R.choice == 55 then  --platform
  14273.         R.width     = menu.getInteger("Platform width", 1, 256, 1, colors.yellow)
  14274.         R.length    = menu.getInteger("Platform length", 1, 256, 2, colors.orange)
  14275.         if menu.getBoolean("Remove blocks above? (y/n)",3, colours.red) then
  14276.             R.up = true
  14277.         end
  14278.        
  14279.     elseif R.choice == 56 then  --sinking platform
  14280.         R.width     = menu.getInteger("Width (excluding retaining wall)? ", 1, 1024, nil, colors.yellow)
  14281.         R.length    = menu.getInteger("Length (excluding retaining wall)? ", 1, 1024 / R.width, nil, colors.orange)
  14282.         R.height    = menu.getInteger("Levels to go down?", 1, 1024 / R.width * R.length, nil, colors.blue)
  14283.     elseif R.choice == 57 then  --boat bubble lift
  14284.         R.height    = menu.getInteger("Levels to go up?", 1, ceiling - 2, nil, colors.lightBlue)
  14285.        
  14286. -- 06. MOB SPAWNER TOOLS
  14287.     elseif R.choice == 61 then -- create cube round mob spawner
  14288.         local text =
  14289. [[~yellow~Turtle placement options:
  14290.  
  14291. ~lime~1. On top or in front of the spawner
  14292.    ~green~(cave spider OR no chests around)
  14293. ~lightGray~2. Outside the dungeon at floor or
  14294.    ceiling level ~red~1 block from corner
  14295. ~yellow~Make sure you have access to the block
  14296. facing the spawner, as any chests
  14297. found inside will be placed outside
  14298. the new dungeon wall at this height
  14299.  
  14300. ~white~Enter to continue
  14301. ]]
  14302.         menu.colourText(nil, text, true)
  14303.         read()
  14304.         menu.clear()
  14305.         R.data = "spawner"
  14306.         if menu.getBoolean("Any chests around spawner? (y/n)",1, colours.red) then
  14307.             R.data = "chest"
  14308.         end
  14309.         local choices = {"Turtle on spawner (no chests)",
  14310.                          "Outside wall: left side, floor",
  14311.                          "Outside wall: left side, ceiling",
  14312.                          "Outside wall: right side, floor",
  14313.                          "Outside wall: right side, ceiling"}
  14314.         pp.itemColours = {colors.red, colors.lime, colors.orange, colors.green, colors.yellow}
  14315.         local userChoice, modifier = menu.new("Select turtle position", choices, pp) -- 1 to 5
  14316.        
  14317.         if modifier == "q" then -- quit chosen
  14318.             R.data = "quit"
  14319.             return R
  14320.         end
  14321.         R.subChoice     = userChoice
  14322.         if R.subChoice > 1 then -- get dungeon wall dimensions
  14323.             R.width     = menu.getInteger("Dungeon external width ", 0, 11, nil, colors.yellow, nil, 11)
  14324.             R.length    = menu.getInteger("Dungeon external length ", 0, 11, nil, colors.orange, nil, 11)
  14325.         end
  14326.        
  14327.     elseif R.choice == 62 then -- Blaze spawner
  14328.         local text =
  14329. [[~yellow~Turtle placement options:
  14330.  
  14331. ~red~1a On top of the spawner
  14332.  
  14333. ~red~1b ~orange~At the same level in direct line of
  14334. spawner.~yellow~ (can be behind a safety wall)
  14335.  
  14336. ~red~1c~lightGray~ Room below spawner (via stairs)
  14337.  
  14338. ~lime~2 Finish build: In brick floor outside
  14339.   existing cube. (facing player)
  14340.  
  14341. ~white~Enter to continue]]
  14342.         menu.colourText(nil, text, true)
  14343.         read()
  14344.         pp.itemColours = {colors.red, colors.lime}
  14345.         R.subChoice = menu.new("Choose your option", {"New Blaze spawner", "Complete part build (killzone)"}, pp, "Type number + Enter") -- 1 = new, 2= continue
  14346.         R.data = "blaze"            -- default
  14347.         if R.subChoice == 1 then    -- new blaze spawner
  14348.             text =
  14349. [[~yellow~Turtle placement options:
  14350. ~red~1. On top of the spawner
  14351. ~orange~2. In direct sight (safety wall ok)
  14352. ~lightGray~3. Room below spawner (via stairs)
  14353.    |*|*|*|*|*|   Ceiling
  14354.    |*|*|~red~T~lightGray~|*|*|   Turtle under ceiling
  14355.    |*|~brown~+~lightGray~|*|~brown~+~lightGray~|*|   Top of fence
  14356.    |*|~brown~+~lightGray~|*|~brown~+~lightGray~|*|
  14357.    |*|~brown~+~lightGray~|*|~brown~+~lightGray~|*|   Bottom of fence
  14358.    |*|*|*|*|*|   Floor level]]
  14359.             --R.subChoice = menu.new("Choose your position", {"On top of spawner", "In direct sight (safety wall ok)", "Room below spawner"}, pp, "Type number + Enter")
  14360.             --getInteger(prompt, minValue, maxValue, row, fg, bg, default)
  14361.             T:clear()
  14362.             menu.colourText(nil, text, true)
  14363.             R.subChoice = menu.getInteger("Enter position number", 1, 3, 12, colors.white)
  14364.         else
  14365.             R.data = "restart"      -- complete part build
  14366.         end
  14367.     elseif R.choice == 63 then -- flood spawner cube
  14368.         local choices = {"Bubble lift", "Computercraft mob softener"}
  14369.         pp.itemColours = {colors.lime, colors.orange}
  14370.         local userChoice = menu.menu("Flood for Mob grinder type?", choices, pp, "Type number + Enter ")
  14371.         R.subChoice = userChoice -- 1 bubble, 2 computercraft
  14372.     elseif R.choice == 64 then -- create bubble lift at mob spawner
  14373.         local choices = {"Dropzone on left", "Dropzone on right"}
  14374.         pp.itemColours = {colors.lime, colors.orange}
  14375.         local userChoice = menu.menu("Which side do you want the mobs?", choices, pp, "Type number + Enter")
  14376.         R.subChoice = userChoice -- 1 left, 2 right
  14377.     elseif R.choice == 65 then -- Computercraft mob grinder
  14378.         local choices = {"Use Sticky pistons", "Use non-sticky pistons"}
  14379.         pp.itemColours = {colors.lime, colors.orange}
  14380.         local userChoice = menu.menu("Which type of pistons?", choices, pp, "Type number + Enter ")
  14381.         R.subChoice = userChoice -- 1 sticky, 2 normal
  14382.    
  14383. -- 07. AREA CARVING
  14384.     elseif R.choice == 71 then --Clear field
  14385.         R.width     = menu.getInteger("Width of the area (1-64)", 1, 64, nil, colors.yellow)
  14386.         R.length    = menu.getInteger("Length of the area (1-64)", 1, 64, nil, colors.orange)
  14387.         if menu.getBoolean("Dirt on the surface (y/n)", nil, colors.brown) then
  14388.             R.useBlockType = "dirt"
  14389.         end
  14390.     elseif R.choice == 72 then -- Clear solid rectangle R.width, R.length
  14391.         R.width     = menu.getInteger("Rectangle width (1-256)", 1, 256, nil, colors.yellow)
  14392.         R.length    = menu.getInteger("Rectangle length (1-256)", 1, 256, nil, colors.orange)
  14393.         R.up        = menu.getBoolean("Remove blocks above?", nil, colors.yellow)
  14394.         R.down      = menu.getBoolean("Remove blocks below?", nil, colors.orange)
  14395.         if menu.getBoolean("Am I outside clearing zone (y/n)?", nil, colors.yellow) then
  14396.             T:forward(1)
  14397.         end
  14398.     elseif R.choice == 73 then -- Clear wall
  14399.         R.width     = 1
  14400.         R.length    = menu.getInteger("Length of wall (1-256) ", 1, 256, nil, colors.orange)
  14401.         R.height    = menu.getInteger("Height of wall (1-50) ", 1, 50, nil, colors.lightGray)
  14402.         pp.itemColours = {colors.lime, colors.orange}
  14403.         R.subChoice = menu.menu("Which direction?", {"Bottom -> Top", "Top -> Bottom"}, pp, "Type number + Enter ") -- open direction menu options
  14404.         if R.subChoice == 1 then
  14405.             if menu.getBoolean("Am I outside clearing zone (y/n)?", nil, colors.yellow) then
  14406.                 T:forward(1)
  14407.             end
  14408.         else
  14409.             pp.itemColours = {colors.lightBlue, colors.cyan, colors.blue, colors.gray}
  14410.             local choice = menu.menu("Exact position?", {"On top of clearing zone", "In front of clearing zone", "Inside clearing zone","Above AND outside"}, pp, "Type number + Enter ")
  14411.             if choice == 1 then
  14412.                 T:down(1)
  14413.             elseif choice == 2 then
  14414.                 T:forward(1)
  14415.             elseif choice == 4 then
  14416.                 T:go("D1F1")
  14417.             end
  14418.         end
  14419.     elseif R.choice == 74 then -- Clear rectangle perimeter only R.width, R.length
  14420.         R.width     = menu.getInteger("Perimeter width (1-256) ", 1, 256, nil, colors.yellow)
  14421.         R.length    = menu.getInteger("Perimeter length (1-256) ", 1, 256, nil, colors.orange)
  14422.         R.height    = 1
  14423.         if menu.getBoolean("Remove blocks above? (y/n)", 3, colours.red) then
  14424.             R.up = true
  14425.         end
  14426.         if menu.getBoolean("Remove blocks below? (y/n)", 4, colours.red) then
  14427.             R.down = true
  14428.         end
  14429.         if menu.getBoolean("Am I outside clearing zone (y/n)?", 5, colors.yellow) then
  14430.             T:forward(1)
  14431.         end
  14432.     elseif R.choice == 75 or R.choice == 76 then -- Clear hollow building floor/walls/ceiling OR clear solid object
  14433.         R.width     = menu.getInteger("Structure width (1-256)", 1, 256, nil, colors.yellow)
  14434.         R.length    = menu.getInteger("Structure length (1-256)", 1, 256, nil, colors.orange)
  14435.         R.height    = menu.getInteger("Depth/Height (1-256)", 1, 256, nil, colors.lightGray)
  14436.         if R.choice == 75 then -- hollow building so need to check if floors/ceilings to be removed
  14437.             R.data = {}
  14438.             R.data.ceiling = false
  14439.             R.data.floor = false
  14440.             if menu.getBoolean("Remove ceiling? (y/n)", nil, colors.yellow, colors.black) then
  14441.                 R.data.ceiling = true
  14442.             end
  14443.             if menu.getBoolean("Remove floor? (y/n)", nil, colors.orange, colors.black) then
  14444.                 R.data.floor = true
  14445.             end
  14446.         end
  14447.         pp.itemColours = {colors.lime, colors.orange}
  14448.         local direction = menu.menu("Which direction?", {"Bottom -> Top", "Top -> Bottom"}, pp, "Type number + Enter ") -- open direction menu options
  14449.         if direction == 1 then
  14450.             R.direction = "up"
  14451.             if menu.getBoolean("Am I outside clearing zone (y/n)?", nil, colors.yellow) then
  14452.                 T:forward(1)
  14453.             end
  14454.         else
  14455.             R.direction = "down"
  14456.             pp.itemColours = {colors.lightBlue, colors.cyan, colors.blue, colors.gray}
  14457.             local choice = menu.menu("Exact position?", {"On top of clearing zone", "Outside clearing zone", "Inside clearing zone","Above AND outside"}, pp, "Type number + Enter ")
  14458.             if choice == 1 then
  14459.                 T:down(1)
  14460.             elseif choice == 2 then
  14461.                 T:forward(1)
  14462.             elseif choice == 4 then
  14463.                 T:go("D1F1")
  14464.             end
  14465.         end
  14466.     elseif R.choice == 77 then  -- Dig a trench
  14467.         R.height    = menu.getInteger("Depth of the trench (1-64) ", 1, 64, nil, colors.blue)
  14468.         R.length    = menu.getInteger("Trench length? 0 = continuous ", 0, 1024, nil, colors.orange)
  14469.     elseif R.choice == 78 then  -- Carve side of mountain
  14470.         pp.itemColours = {colors.lime, colors.orange}
  14471.         R.subChoice = menu.menu("Remove blocks on which side?", {"Left", "Right"}, pp, "Type number + Enter ") -- open direction menu options
  14472.         R.width     = menu.getInteger("Remove how many vertical rows? ", 1, 1024, nil, colors.yellow)
  14473.         R.length    = menu.getInteger({"Max length of each row?"," - Prevents cutting too wide","1 to 255 default 64"}, 1, 255, nil, {colors.orange, colors.red, colors.white}, colors.black, 64)
  14474.     elseif R.choice == 79 then  -- Place a floor or ceiling
  14475.         local items = { "Replacing current floor",
  14476.                         "New floor over existing",
  14477.                         "Replacing current ceiling",
  14478.                         "New ceiling under existing"}
  14479.         pp.itemColours = {colors.lime, colors.orange, colors.green, colors.yellow}
  14480.         R.subChoice = menu.menu("Laying what?", items, pp, "Type number + Enter ")
  14481.         R.height = menu.getInteger({"Enter 0 for on-site placement",
  14482.                                     "If in deep water or above reach",
  14483.                                     "Enter approx depth/height"}, 0, 64, nil, {colors.yellow, colors.blue, colors.cyan} )
  14484.         if R.subChoice < 3 then
  14485.             R.down = true
  14486.             R.width     = menu.getInteger("Width of floor (1-64) ", 1, 64, nil, colors.yellow)
  14487.             R.length    = menu.getInteger("Length of floor (1-64) ", 1, 64, nil, colors.orange)
  14488.         else
  14489.             R.up = true
  14490.             R.width     = menu.getInteger("Width of ceiling (1-64) ", 1, 64, nil, colors.yellow)
  14491.             R.length    = menu.getInteger("Length of ceiling (1-64) ", 1, 64, nil, colors.orange)
  14492.         end
  14493.     elseif R.choice == 710 or R.choice == 83 then -- Direct movement
  14494.         local choices = {"Simple path", "Covered 2 block high path"}
  14495.         pp.itemColours = {colors.lime, colors.orange}
  14496.         local userChoice = menu.menu("Choose your path option", choices, pp, "Type number + Enter ")
  14497.         R.subChoice = userChoice -- 1 open, 2 covered
  14498.         choices = {"Command driven", "Menu driven"}
  14499.         userChoice = menu.menu("Choose your preference", choices, pp, "Type number + Enter ")
  14500.         if userChoice == 1 then
  14501.             R.data = "cmd"
  14502.         else
  14503.             R.data = "menu"
  14504.         end
  14505.        
  14506. -- 08. WATER LAVA
  14507.     elseif R.choice == 81 then -- Sand based utilities
  14508.         local line = getTaskHelp(2, R.choice, true, true) -- request line no of help display if needed for getInteger
  14509.         R.subChoice = menu.getInteger("Type number of choice", 1, 4, line, colors.white)
  14510.         T:clear()
  14511.         if R.subChoice == 1 then -- drop sand into water or lava surface until solid ground reached
  14512.             R.width   = 1
  14513.             R.length  = menu.getInteger("Length of sand wall (0=to block) ", 0, 60, nil, colors.orange)
  14514.         elseif R.subChoice == 2 then    -- clear rectangle on top of building and fill with sand
  14515.             R.width   = menu.getInteger("Width of area (<=30) ", 1, 30, nil, colors.yellow)
  14516.             R.length  = menu.getInteger("Length of of area (<=30) ", 1, 30, nil, colors.orange)
  14517.         elseif R.subChoice == 3 then    -- clear sand wall or harvest sand
  14518.             R.width   = 1
  14519.             R.length  = menu.getInteger("Length of sand (0=auto-detect)", 0, 60, nil, colors.orange)
  14520.             choices = {"Stay at end of wall", "Return home"}
  14521.             pp.itemColours = {colors.lime, colors.orange}
  14522.             userChoice = menu.menu("Choose your preference", choices, pp, "Type number + Enter ")
  14523.             if userChoice == 2 then
  14524.                 R.data = "return"
  14525.             end
  14526.         elseif R.subChoice == 4 then    -- remove sand from cube. start at top
  14527.             R.width   = menu.getInteger("Width of sand (<=30) ", 1, 30, nil, colors.yellow)
  14528.             R.length  = menu.getInteger("Length of of sand (<=30) ", 1, 30, nil, colors.orange)
  14529.             R.height  = 0
  14530.             R.data = "down" -- always starts at the top
  14531.         end
  14532.         if menu.getBoolean("Am I outside the active zone (y/n)?", nil, colors.yellow) then
  14533.             T:forward(1)
  14534.         end
  14535.     elseif R.choice == 82 then -- build wall from water or lava surface downwards
  14536.         R.width     = 1
  14537.         if menu.getBoolean("Going 90 deg. from existing? (y/n)", nil, colors.yellow, colors.black) then
  14538.             getTaskHelp(2, 852, true)
  14539.             menu.clear()   
  14540.             R.length = 56
  14541.             R.data = "withPath" -- ensures turtle will break through path
  14542.             R.length = menu.getInteger({"Wall length minus corners","(Ocean monument default 56)","Number -> Enter or Enter only (56)"},
  14543.                                         1, 64, nil, {colors.yellow, colors.orange, colors.green}, colors.black, 56)
  14544.         else
  14545.             R.length    = menu.getInteger("Length of the wall (1-60) ", 1, 60, nil, colors.yellow)
  14546.             R.height    = menu.getInteger("Fixed depth or 0 = to floor ", 0, 60, nil, colors.yellow)
  14547.         end
  14548.     -- for 83 see 710
  14549.     elseif R.choice == 84 then -- Clear area of water bounded by blocks
  14550.         R.width     = menu.getInteger("Width of water (0=autodetect) ", 0, 64, nil, colors.yellow)
  14551.         if R.width > 0 then
  14552.             R.length = menu.getInteger("Length of water", 1, 64, nil, colors.orange)
  14553.         end
  14554.         R.height    = menu.getInteger("Depth of water (0=autodetect)", 0, 64, nil, colors.blue)
  14555.     elseif R.choice == 85 then -- Sinking platform
  14556.         R.width     = menu.getInteger("Width (excluding retaining wall)? ", 1, 1024, nil, colors.yellow)
  14557.         R.length    = menu.getInteger("Length (excluding retaining wall)? ", 1, 1024 / R.width, nil, colors.orange)
  14558.         R.height    = menu.getInteger("Levels to go down?", 1, 1024 / R.width * R.length, nil, colors.blue)
  14559.     elseif R.choice == 86 then -- ocean monument utilities
  14560.         local items =
  14561.         {
  14562.             "Build 4 corner marker columns",
  14563.             "Retaining wall beween 2 columns",
  14564.             "Clear plants pre sand draining",
  14565.             "Drop sand wall",
  14566.             "Recover sand wall",
  14567.             "Sinking platform",
  14568.             "Drain and remove structure"   
  14569.         }
  14570.         pp.itemColours = {colors.lightGray, colors.brown, colors.blue, colors.yellow, colors.orange, colors.lightGray, colors.gray}
  14571.         R.subChoice = menu.menu("Which utility? ", items, pp)
  14572.         if R.subChoice ~= nil then
  14573.             getTaskHelp(2, R.choice * 10 + R.subChoice, true)
  14574.         end
  14575.         menu.clear()   
  14576.         if R.subChoice == 1 then -- Build 4 corner marker columns          
  14577.             R.useBlockType = "prismarine"
  14578.             R.data = "oceanMonumentColumns"
  14579.         elseif R.subChoice == 2 then -- Retaining wall beween 2 columns
  14580.             R.length = 56
  14581.             R.data = "withPath" -- ensures turtle will break through path
  14582.             T:clear()
  14583.             R.length = menu.getInteger({"Wall length minus corners","(Ocean monument default 56)","Number -> Enter or Enter only (56)"},
  14584.                                          1, 64, nil, {colors.yellow, colors.orange, colors.green}, colors.black, 56)
  14585.         elseif R.subChoice == 3 then -- Clear plants pre sand draining
  14586.             R.useBlockType = "prismarine"
  14587.             R.data = "clearWaterPlants"
  14588.             R.silent = true
  14589.             R.width     = menu.getInteger("water width (0=auto detect) ", 0, 64, nil, colors.yellow)
  14590.             R.length    = menu.getInteger("water length (0=auto detect) ", 0, 64, nil, colors.orange)
  14591.         elseif R.subChoice == 4 then -- Drain using sand utilities
  14592.             R.width   = 1
  14593.             R.length  = 0
  14594.         elseif R.subChoice == 5 then -- remove sand wall using sand utilities
  14595.             R.width   = 1
  14596.             R.length  = 0
  14597.         elseif R.subChoice == 6 then -- sinking platform
  14598.             R.width = menu.getInteger("Width (excluding retaining wall)? ", 1, 1024, nil, colors.yellow)
  14599.             R.length = menu.getInteger("Length (excluding retaining wall)? ", 1, 1024 / R.width, nil, colors.orange)
  14600.             R.height = menu.getInteger("Levels to go down?", 1, 1024 / R.width * R.length, nil, colors.blue)
  14601.         elseif R.subChoice == 7 then -- Drain and remove structure
  14602.             R.width = 28
  14603.             R.length = 28
  14604.             R.height = menu.getInteger("Go down how far from current", 1, 64, nil, colors.blue)
  14605.         end
  14606.     elseif R.choice == 87 then -- Ladder to water/lava 
  14607.         R.height    = menu.getInteger("est. height above (?F3) ", 1, 256, nil, colors.blue)
  14608.     elseif R.choice == 88 then -- Clear water plants from enclosed area
  14609.         R.data      = "clearWaterPlants"
  14610.         R.width     = menu.getInteger("water width (0=auto detect) ", 0, 64, nil, colors.yellow)
  14611.         R.length    = menu.getInteger("water length (0=auto detect) ", 0, 64, nil, colors.orange)
  14612.     elseif R.choice == 89 then -- convert flowing water to source
  14613.         R.width     = menu.getInteger("water width  (0=auto detect) ", 0, 64, nil, colors.yellow)
  14614.         if R.width > 0 then
  14615.             R.length = menu.getInteger("water length (0=auto detect) ", 0, 64, nil, colors.orange)
  14616.         end
  14617.         R.height    = menu.getInteger("water depth (0=auto detect) ", 0, 64, nil, colors.blue)
  14618.     elseif R.choice == 810 then -- create sloping water
  14619.         R.width     = menu.getInteger("water slope width (usually 7/8) ", 1, 8, nil, colors.yellow)
  14620.         R.length    = menu.getInteger("water slope length ", 1, 64, nil, colors.orange)
  14621.    
  14622. -- 09. BUILDING & RAILWAY
  14623.     elseif R.choice == 91 then -- Build wall
  14624.         R.width     = 1
  14625.         R.length    = menu.getInteger("Length of wall (1-256) ", 1, 256, nil, colors.orange)
  14626.         R.height    = menu.getInteger("Height of wall (1-50) ", 1, 50, nil, colors.lightGray)
  14627.         pp.itemColours = {colors.lime, colors.magenta}
  14628.         R.subChoice = menu.menu("What is my current position?", {"End of wall: Start ahead","Within the wall: start here"}, pp, "Type number + Enter")
  14629.     elseif R.choice == 92 then -- Build rectangular structure
  14630.         R.width     = menu.getInteger("Building width (1-256)", 1, 256, nil, colors.yellow)
  14631.         R.length    = menu.getInteger("Building length (1-256) ", 1, 256, nil, colors.orange)
  14632.         R.height    = menu.getInteger("Building Height(1-50) ", 1, 50, nil, colors.lightGray)
  14633.         pp.itemColours = {colors.lime, colors.magenta}
  14634.         R.subChoice = menu.menu("What is my current position?", {"Outside building: Start ahead","Within the walls: start here"}, pp, "Type number + Enter")
  14635.     elseif R.choice == 93 or R.choice == 94 then -- Build gable roof / pitched roof
  14636.         R.width     = menu.getInteger("Building width (1-256)", 1, 256, nil, colors.yellow)
  14637.         R.length    = menu.getInteger("Building length (1-256) ", 1, 256, nil, colors.orange)
  14638.     elseif R.choice == 97 then -- build downward slope
  14639.     --getInteger(prompt, minValue, maxValue, row, fg, bg, default)
  14640.         R.height    = menu.getInteger("How many blocks down (0=to ground)?", 0, 256, nil, colors.blue)
  14641.         R.depth     = menu.getInteger("How much headroom (default 2)?", 0, 5, nil, colors.cyan, colors.black, 2)
  14642.         R.down = true
  14643.     elseif R.choice == 98 then -- build upward slope
  14644.         R.height    = menu.getInteger("Go up by how many blocks?", 1, 256, nil, colors.lightGray)
  14645.         R.depth     = menu.getInteger("How much headroom (default 2)?", 0, 5, nil, colors.cyan, colors.black, 2)
  14646.         R.up = true
  14647.        
  14648. -- 10. MEASURING TOOLS
  14649.     elseif R.choice == 101 then -- measure height
  14650.         pp.itemColours = {colors.lightBlue, colors.cyan, colors.lime, colors.gray}
  14651.         R.subChoice = menu.menu("Measure using?", {"Obstruction above", "No further blocks in front", "Detect specific block in front", "Until block above changes" }, pp, "Type number + Enter ") -- open direction menu options
  14652.         if R.subChoice == 3 then
  14653.             R.data = menu.getString({"Search for? eg 'ore', 'obsidian'",">"}, false, 3, 20, nil, {colors.lime, colors.yellow})
  14654.         end
  14655.         R.size = menu.getInteger("Max distance before abort?", 1, 1024, nil, colors.red, nil, 64)
  14656.     elseif R.choice == 102 then -- measure depth
  14657.         pp.itemColours = {colors.lime, colors.orange, colors.green}
  14658.         R.subChoice = menu.menu("Measure using?", {"Water/Lava/Obstruction below", "No further blocks ahead", "Detect specific block ahead"}, pp, "Type number + Enter") -- open direction menu options
  14659.         if R.subChoice == 3 then
  14660.             R.data = menu.getString({"Search for? eg 'ore', 'obsidian'",">"}, false, 3, 20, nil, {colors.lime, colors.yellow})
  14661.         end
  14662.     elseif R.choice == 103 then -- measure length
  14663.         pp.itemColours = {colors.gray, colors.lime, colors.green, colors.blue, colors.cyan}
  14664.         R.subChoice = menu.menu("Measure using?", {"Obstruction ahead",
  14665.                                                     "No further blocks above",
  14666.                                                     "No further blocks below",
  14667.                                                     "Detect specific block above",
  14668.                                                     "Detect specific block below"}, pp, "Type number + Enter") -- open direction menu options
  14669.         if R.subChoice == 4 or R.subChoice == 5 then
  14670.             R.data = menu.getString({"Search for? eg 'ore', 'obsidian'",">"}, false, 3, 20, nil, {colors.lime, colors.yellow})
  14671.         end
  14672.         R.size = menu.getInteger("Max distance before abort?", 1, 1024, nil, colors.red, nil, 64)
  14673.     elseif R.choice == 104 then -- measure greatest depth of water body
  14674.         pp.itemColours = {colors.lime, colors.orange}
  14675.         R.subChoice = menu.menu("Measure using?", {"Player entered", "No further water below"}, pp, "Type number + Enter ") -- open direction menu options
  14676.         -- 1 = player entry, 2 = water length
  14677.         if R.subChoice == 1 then
  14678.             R.length = menu.getInteger("water length (0=auto detect) ", 0, 256)
  14679.         end
  14680.     elseif R.choice == 105 then -- drill borehole and write borhole.txt to file
  14681.         R.height = menu.getInteger("Current level (F3->Y coord)? ", bedrock + 5, ceiling, nil, colors.lightGray)
  14682.         R.depth = menu.getInteger("Go down to level? ("..R.height - 2 .." to "..bedrock + 5 ..")", bedrock + 5 ,R.height - 2, nil, colors.blue, nil, bedrock + 5)
  14683.    
  14684. -- 11 Network tools
  14685.     elseif R.choice == 111 then -- craft an item
  14686.         local modem = peripheral.find("modem")      -- find modem
  14687.         if modem == nil then
  14688.             menu.colourText(1, "~yellow~I need to be placed close to an\n"..
  14689.                   "~red~activated ~brown~computercraft:modem_full~yellow~ to use "..
  14690.                   "~magenta~networked resources ~yellow~for crafting"..
  14691.                   "\n\n~blue~(Right-click modem to activate)\n")
  14692.                 error()
  14693.         end
  14694.     elseif R.choice == 112 then -- mystical Agriculture essence tools
  14695.         if T:getBlockType("forward") ~= "minecraft:crafter" then
  14696.             menu.colourText(1, "~yellow~I need to be placed in front of a\n"..
  14697.                   "~red~minecraft:crafter ~yellow~(mc 1.21.1) as \n"..
  14698.                   "currently turtle.craft() with a\n"..
  14699.                   "~lime~mystical agriculture crystal\n~red~destroys the crystal!\n")
  14700.                 error()
  14701.         end
  14702.         pp.itemColours = {colors.green, colors.lime, colors.orange, colors.blue, colors.red}
  14703.         R.subChoice = menu.menu("Process Essences:", {"Inferium to Prudentium",
  14704.                                                     "Prudentium to Tertium",
  14705.                                                     "Tertium to Imperium",
  14706.                                                     "Imperium to Supremium",
  14707.                                                     "Supremium to Insanium"}, pp, "Type number + Enter") -- open essence menu options
  14708.         R.size = menu.getInteger("Quantity? 0 = all)", 0, 1024, nil, colors.orange, nil, 64)
  14709.     end
  14710.  
  14711.     return R
  14712. end
  14713.  
  14714. local function getTaskInventoryTo30(R)
  14715.     local retValue = {}
  14716.     local thanks = "Thank you.\n\nPress the ESC key\n\nStand Back..\n"
  14717.     --local pp = utils.getPrettyPrint()
  14718.    
  14719.     if R.choice == 0 then --Missing pickaxe
  14720.         T:checkInventoryForItem({"minecraft:diamond_pickaxe"}, {1})
  14721.         print("Diamond Pickaxe being tested...")
  14722.         T:setEquipment()
  14723.     elseif R.choice == 1 then --Missing crafting table
  14724.         T:checkInventoryForItem({"minecraft:crafting_table"}, {1}) -- 0 if not present
  14725.         print("Crafting table being tested...")
  14726.         T:setEquipment()
  14727.     elseif R.choice == 2 then --Missing chest
  14728.         T:checkInventoryForItem({"minecraft:chest"}, {1}) -- 0 if not present
  14729.         sleep(1.5)
  14730.        
  14731.     -- MINING
  14732.    
  14733.     elseif R.choice == 11 or R.choice == 12 then            -- ladder down / up / stairs down / stairs up
  14734.         local description = "Creating ladder going down"    -- default action description
  14735.         pp.itemColours = {colors.blue, colors.lightGray}
  14736.         pp.allowModifier = false -- only allow numbers to be returned
  14737.         local inAir = false
  14738.         local inNether = menu.getBoolean("Are you in the nether? (y/n) ", nil, colors.yellow, colors.black)
  14739.        
  14740.         if inNether then
  14741.             bedrock = 0
  14742.             inAir = true
  14743.         end
  14744.         local option = menu.menu("Which direction?", {"Going down","Going up"}, pp)
  14745.         if option == 1 then
  14746.             R.down = true
  14747.             R.height = menu.getInteger("Current level (F3->Y coord)? ", bedrock + 5, ceiling, nil, colors.lightGray)
  14748.             R.depth = menu.getInteger("Go down to level? ("..R.height - 2 .." to "..bedrock + 5 ..") ", bedrock + 5 , R.height - 2, nil, colors.blue)
  14749.             if R.choice == 11 then
  14750.                 if menu.getBoolean("Build a shelter at base? (y/n) ", nil, colors.yellow, colors.black) then
  14751.                     R.data = "chamber"
  14752.                 end
  14753.             end
  14754.             if R.choice == 13 then
  14755.                 description = "Creating stairs going down"
  14756.             end
  14757.         else
  14758.             R.up = true
  14759.             --R.depth = utils.getSize(true,"Current level (F3->Y coord)? ", bedrock + 5, ceiling)
  14760.             --R.height = utils.getSize(true, "Go up to level? ("..R.depth + 2 .." to "..ceiling ..")", R.depth + 2, ceiling)
  14761.             R.depth = menu.getInteger("Current level (F3->Y coord)? ", bedrock + 5, ceiling, nil, colors.blue)
  14762.             R.height = menu.getInteger("Go up to level? ("..R.depth + 2 .." to "..ceiling ..") ", R.depth + 2, ceiling, nil, colors.lightGray)
  14763.             if R.choice == 11 then
  14764.                 description = "Creating ladder going up"
  14765.             else
  14766.                 description = "Creating stairs going up"
  14767.             end
  14768.         end
  14769.         local range = math.abs(R.height - R.depth)
  14770.        
  14771.         if not inAir then
  14772.             inAir = menu.getBoolean("Are you in air (add blocks)? (y/n) ", nil, colors.yellow, colors.black)
  14773.         end
  14774.         --[[
  14775.         range examples
  14776.         -50 to -59 = -59 -(-50) = -9   down
  14777.         -59 to -50 = -50 -(-59) = 9    up
  14778.          70 to -48 = -48 -   70 = -118 down
  14779.           5 to  64 =  64 -   5  = 59   up
  14780.         ]]
  14781.         T:checkInventoryForItem({"minecraft:bucket"}, {1}, false)
  14782.         if R.choice == 11 then -- ladders
  14783.             utils.checkFuelNeeded(range * 2)
  14784.             T:checkInventoryForItem({"minecraft:ladder"}, {range})
  14785.             T:checkInventoryForItem({"minecraft:torch"}, {math.floor(range / 3)}, false)
  14786.             if inAir then
  14787.                 range = range * 4 -- more blocks needed
  14788.             end
  14789.             T:checkInventoryForItem({"stone"}, {range})
  14790.         else -- stairs
  14791.             utils.checkFuelNeeded(range * 10) -- stairs: each layer needs 10 moves
  14792.             local numStairsNeeded = range
  14793.             local data = T:getStock("stairs")
  14794.             local numStairs = data.total
  14795.             local cobbleNeeded = math.min(range * 6, 256)
  14796.             if inAir then
  14797.                 cobbleNeeded = range * 6 -- 5 blocks / layer unless in water or lava
  14798.             end
  14799.             if numStairs < numStairsNeeded then
  14800.                 cobbleNeeded = cobbleNeeded + (math.floor((2 * (numStairsNeeded - numStairs)) / 3))
  14801.             end
  14802.             T:checkInventoryForItem({"stairs"}, {numStairsNeeded}, false)
  14803.             T:checkInventoryForItem({"stone"}, {cobbleNeeded})
  14804.             T:checkInventoryForItem({"minecraft:chest"}, {1})   -- needed for crafting
  14805.         end
  14806.         menu.colourPrint(description, colors.lightBlue)
  14807.         os.sleep(2)    -- pause for 2 secs to allow time to press esc
  14808.         if R.choice == 11 then
  14809.             retValue = createLadder(R) -- "bedrock", 70, -48
  14810.         else
  14811.             retValue = createStaircase(R)
  14812.         end
  14813.     elseif R.choice == 13 then --Create Mine at this level
  14814.         utils.checkFuelNeeded(960)
  14815.         T:checkInventoryForItem({"minecraft:torch"}, {24}, false)
  14816.         T:checkInventoryForItem({"minecraft:bucket"}, {1}, false)
  14817.         T:checkInventoryForItem({"stone"}, {64})
  14818.         T:checkInventoryForItem({"minecraft:chest"}, {1})
  14819.         menu.colourPrint("CreateMine starting", colors.lightBlue)
  14820.         sleep(2)
  14821.         retValue = createMine()
  14822.     elseif R.choice == 14 then  -- safe drop to water
  14823.         utils.checkFuelNeeded(R.height * 2)
  14824.         T:checkInventoryForItem({"minecraft:water_bucket"}, {1})
  14825.         T:checkInventoryForItem({"stone"}, {R.height * 2}, false) -- estimate only partial cloaking needed
  14826.         menu.colourPrint("Creating safe drop ".. R.height.. " blocks deep", colors.lightBlue)
  14827.         menu.colourPrint("Wait for my return!", colors.yellow)
  14828.         os.sleep(2)    -- pause for 2 secs to allow time to press esc
  14829.         retValue = createSafeDrop(R)
  14830.     elseif R.choice == 15 then  -- single column bubble lift
  14831.         local needSigns = true
  14832.         utils.checkFuelNeeded(R.height * 6)
  14833.         --print(text, fg, bg, width)
  14834.         menu.colourPrint("Checking if near ladder", colors.red)
  14835.         T:turnRight(1)
  14836.         local blockType = T:getBlockType("forward")
  14837.         if blockType:find("ladder") == nil then
  14838.             T:turnLeft(2)
  14839.             blockType = T:getBlockType("forward")
  14840.             if blockType:find("ladder") ~= nil then
  14841.                 needSigns = false
  14842.             end
  14843.             T:turnRight(1)
  14844.         else
  14845.             needSigns = false
  14846.             T:turnLeft(1)
  14847.         end
  14848.         sleep(1)
  14849.         if needSigns then
  14850.             T:checkInventoryForItem({"sign"}, {2}, true, "Ladder not found")
  14851.         else
  14852.             menu.colourPrint("Ladder found, signs not required", colors.orange)
  14853.             sleep(1)
  14854.         end
  14855.         if T:getBlockType("down"):find("bedrock") == nil then -- NOT bedrock below
  14856.             T:checkInventoryForItem({"minecraft:water_bucket"}, {3})
  14857.         else
  14858.             T:checkInventoryForItem({"minecraft:water_bucket"}, {2})
  14859.         end
  14860.         T:checkInventoryForItem({"minecraft:soul_sand", "minecraft:dirt"}, {1, 1}, true, "? use dirt as soul sand placeholder")
  14861.         T:checkInventoryForItem({"stone"}, {R.height * 4 + 10}, false) -- estimate only partial cloaking needed
  14862.         local _, slots = T:getFirstEmptySlot() -- returns first empty slot, no of empty slots
  14863.         T:checkInventoryForItem({"minecraft:bucket"}, {slots}, false, "More buckets = faster build!")
  14864.         menu.colourPrint("Creating bubble lift ".. R.height.. " blocks high", colors.lime)
  14865.        
  14866.         if needSigns then -- ladder present as signs not required
  14867.             menu.colourPrint("\nSTAND CLEAR!", colors.red)
  14868.         else
  14869.             menu.colourPrint("Wait at the top via ladder...", colors.orange)
  14870.             menu.colourPrint("\nSTAND ON THE LADDER FOR SAFETY!", colors.red)
  14871.         end
  14872.         menu.colourPrint("\nEnter to continue", colors.yellow)
  14873.         read()
  14874.         retValue = createBubbleLift(R)
  14875.     elseif R.choice == 16 then --QuickMine corridor
  14876.         utils.checkFuelNeeded((R.width * 2 + R.length * 2) * 2)
  14877.         T:checkInventoryForItem({"stone"}, {R.width * 2 + R.length * 2}, false)
  14878.         T:checkInventoryForItem({"minecraft:torch"}, {math.floor((R.width * 2 + R.length * 2) / R.torchInterval)}, false)
  14879.         T:checkInventoryForItem({"minecraft:bucket"}, {1}, false)
  14880.         menu.colourPrint("QuickMine corridor: R.subChoice "..R.width.. " x "..R.length, colors.lightBlue)
  14881.         retValue = quickMineCorridor(R)
  14882.     elseif R.choice == 17 then --QuickMine
  14883.         utils.checkFuelNeeded(R.width * R.length)
  14884.         T:checkInventoryForItem({"stone"}, {64})
  14885.         T:checkInventoryForItem({"minecraft:bucket"}, {1}, false)
  14886.         menu.colourPrint("QuickMine rectangle: R.subChoice "..R.width.. " x "..R.length, colors.lightBlue)
  14887.         retValue = quickMine(R)
  14888.     elseif R.choice == 18 then --Mine bedrock area
  14889.         utils.checkFuelNeeded(R.width * R.length)
  14890.         T:checkInventoryForItem({"stone"}, {64})
  14891.         T:checkInventoryForItem({"minecraft:bucket"}, {1}, false)
  14892.         retValue = mineBedrockArea(R) -- R.data = "leaveExposed" or ""
  14893.     elseif R.choice == 19 then  -- salvage mineshaft
  14894.         local equippedRight, equippedLeft, inInventory = T:setEquipment() -- check for crafting table, sword, pickaxe, Put sword in spare slot
  14895.         if equippedLeft ~= "minecraft:diamond_pickaxe" and equippedRight ~= "minecraft:diamond_pickaxe" then
  14896.             T:checkInventoryForItem({"minecraft:diamond_pickaxe"}, {1}, true)
  14897.             equippedRight, equippedLeft, inInventory = T:setEquipment() -- check for crafting table, sword, pickaxe, Put sword in spare slot
  14898.         end
  14899.         if inInventory ~= "minecraft:diamond_sword" then
  14900.             T:checkInventoryForItem({"minecraft:diamond_sword"}, {1}, false, "To harvest spider webs\nyou need a diamond sword.") --checkInventoryForItem(self, items, quantities, required, message, name)
  14901.         end
  14902.         if T:getItemSlot("minecraft:diamond_sword") > 0 then
  14903.             inInventory = "minecraft:diamond_sword"
  14904.         end
  14905.         T:checkInventoryForItem({"minecraft:torch"}, {R.torchInterval}, false)
  14906.         if inInventory == "minecraft:diamond_sword" then
  14907.             menu.colourPrint("Clearing Mineshaft and cobwebs", colors.orange)
  14908.         else
  14909.             menu.colourPrint("Clearing Mineshaft", colors.yellow)
  14910.         end
  14911.         retValue = clearMineshaft(R, equippedRight, equippedLeft, inInventory) -- pass whether the sword is present
  14912.        
  14913.     -- FORESTRY
  14914.     elseif R.choice == 21 then  -- Fell tree
  14915.         if T:isLog("forward") then
  14916.             if turtle.getFuelLevel() < 30 then
  14917.                 T:checkInventoryForItem({"minecraft:chest"}, {1}, false,"Fuel level critical: "..turtle.getFuelLevel())
  14918.                 turtle.select(1)
  14919.                 T:dig("forward")
  14920.                 T:craft("planks", 4)
  14921.                 T:refuel()
  14922.                 T:forward(1)
  14923.                 T:up(2)
  14924.                 T:craft("chest", 1)
  14925.             else
  14926.                 T:forward(1)
  14927.             end
  14928.             menu.colourPrint("Felling tree", colors.lime)
  14929.             os.sleep(2)    -- pause for 2 secs to allow time to press esc
  14930.             T:harvestWholeTree("up")
  14931.             while turtle.down() do end
  14932.             retValue = {"Tree Harvested"}
  14933.         else
  14934.             retValue =
  14935.             {
  14936.                 "No log in front..",
  14937.                 "Move me in front of a tree!"
  14938.             }
  14939.         end
  14940.     elseif R.choice == 22 then --Create treefarm
  14941.         utils.checkFuelNeeded(900)
  14942.         if R.data == "convertStorage" or R.data == "convert" then
  14943.             T:checkInventoryForItem({"log"}, {1}, true, "Use log to indicate sapling type")
  14944.             T:checkInventoryForItem({"barrel"}, {2})
  14945.             T:checkInventoryForItem({"minecraft:hopper"}, {1})
  14946.             if R.data == "convertStorage" then
  14947.                 T:checkInventoryForItem({"chest"}, {8})
  14948.                 T:checkInventoryForItem({"wired_modem_full"}, {3})
  14949.                 T:checkInventoryForItem({"computercraft:cable"}, {24})
  14950.             else
  14951.                 T:checkInventoryForItem({"wired_modem_full"}, {2})
  14952.                 T:checkInventoryForItem({"computercraft:cable"}, {22})
  14953.             end
  14954.         else
  14955.             T:checkInventoryForItem({"stone"}, {320})
  14956.             T:checkInventoryForItem({"minecraft:water_bucket"}, {4})
  14957.             T:checkInventoryForItem({"minecraft:hopper"}, {1})
  14958.             T:checkInventoryForItem({"log"}, {1}, true, "Use log to indicate sapling type")
  14959.             R.useBlockType = T:getMostItem("", true)
  14960.             T:checkInventoryForItem({"barrel"}, {2})
  14961.             if R.data == "new" then
  14962.                 T:checkInventoryForItem({"chest"}, {8})
  14963.                 T:checkInventoryForItem({"wired_modem_full"}, {3})
  14964.                 T:checkInventoryForItem({"computercraft:cable"}, {24})
  14965.             else
  14966.                 T:checkInventoryForItem({"wired_modem_full"}, {2})
  14967.                 if R.data == "back" then
  14968.                     T:checkInventoryForItem({"computercraft:cable"}, {34})
  14969.                 else
  14970.                     T:checkInventoryForItem({"computercraft:cable"}, {22})
  14971.                 end
  14972.             end
  14973.             menu.colourPrint("Creating Tree Farm with "..R.useBlockType, colors.lime)
  14974.         end
  14975.         retValue = createTreefarm(R)
  14976.     elseif R.choice == 23 then -- Plant treefarm
  14977.         if not R.networkFarm then
  14978.             if R.subChoice == 1 then
  14979.                 utils.checkFuelNeeded(180)
  14980.                 T:checkInventoryForItem({"dirt"}, {16})
  14981.                 T:checkInventoryForItem({"sapling"}, {16}, false, "Max 16. NOT dark oak")
  14982.             elseif R.subChoice == 2 then
  14983.                 utils.checkFuelNeeded(180)
  14984.                 T:checkInventoryForItem({"dirt"}, {16})
  14985.                 T:checkInventoryForItem({"sapling"}, {16}, false, "Max 16. 4 saplings / tree")
  14986.             end
  14987.         end
  14988.         menu.colourPrint("plantTreefarm starting: size "..R.subChoice, colors.lime)
  14989.         retValue = plantTreefarm(R)
  14990.     elseif R.choice == 24 then  -- Harvest treefarm
  14991.         menu.colourPrint("Harvesting treefarm starting", colors.lime)
  14992.         retValue = harvestTreeFarm(R)
  14993.     elseif R.choice == 25 then  -- Build wall / fence
  14994.         utils.checkFuelNeeded(R.width * R.length * 2)
  14995.         local quantity = math.ceil((R.width + R.length) * 2.3)
  14996.         T:checkInventoryForItem({"wall", "fence"}, {quantity, quantity})
  14997.         if R.torchInterval > 0 then
  14998.             T:checkInventoryForItem({"minecraft:torch"}, {math.floor(quantity / R.torchInterval)}, false)
  14999.         end
  15000.         if R.data == "barrel" then
  15001.             T:checkInventoryForItem({"barrel"}, {4}, false)
  15002.         end
  15003.         R.useBlockType = T:getMostItem("", false) -- no excluded blocks, any block type
  15004.         menu.colourPrint("Creating "..R.width.." x "..R.length.." walled enclosure", colors.yellow)
  15005.         menu.colourPrint("Using: "..R.useBlockType, colors.orange)
  15006.         retValue = createWallOrFence(R)
  15007.     elseif R.choice == 26 then  -- clear natural forest
  15008.         T:checkInventoryForItem({"minecraft:chest"}, {1})
  15009.         T:checkInventoryForItem({"sapling"}, {64}, false)
  15010.         menu.colourPrint("Clearing and replanting trees", colors.lime)
  15011.         os.sleep(2)    -- pause for 2 secs to allow time to press esc
  15012.         retValue = clearAndReplantTrees()
  15013.     end
  15014.     return retValue
  15015. end
  15016.  
  15017. local function getTaskInventoryTo70(R)
  15018.     local retValue = {}
  15019.     local thanks = "Thank you.\n\nPress the ESC key\n\nStand Back..\n"
  15020.     -- FARMING
  15021.     if R.choice == 31 then  -- Create modular farm
  15022.         utils.checkFuelNeeded(300)
  15023.         T:checkInventoryForItem({"stone"}, {64})
  15024.         T:checkInventoryForItem({"dirt"}, {128}, false)
  15025.         T:checkInventoryForItem({"water_bucket"}, {4})
  15026.         if R.networkFarm then  
  15027.             T:checkInventoryForItem({"barrel"}, {1})
  15028.             T:checkInventoryForItem({"chest",}, {8})
  15029.             T:checkInventoryForItem({"ladder",}, {5})
  15030.             T:checkInventoryForItem({"wired_modem_full"}, {3})
  15031.             T:checkInventoryForItem({"computercraft:cable"}, {70})
  15032.         else    -- conventional storage farm
  15033.             T:checkInventoryForItem({"chest", "barrel"}, {5,5})
  15034.         end
  15035.         T:checkInventoryForItem({"sapling"}, {1})
  15036.         R.useBlockType = T:getMostItem("dirt", true) -- exclude dirt from count
  15037.         print(thanks)
  15038.         os.sleep(2)    -- pause for 2 secs to allow time to press esc
  15039.         print("Creating modular farm with "..R.useBlockType)
  15040.         retValue = createFarm(R, "")
  15041.     elseif R.choice == 32 then  -- Extend modular farm
  15042.         utils.checkFuelNeeded(300)
  15043.         T:checkInventoryForItem({"sapling"}, {1})
  15044.         T:checkInventoryForItem({"stone"}, {64})
  15045.         T:checkInventoryForItem({"dirt"}, {128}, false)
  15046.         T:checkInventoryForItem({"water_bucket"}, {4})
  15047.         if R.networkFarm then  
  15048.             T:checkInventoryForItem({"barrel"}, {1})
  15049.             T:checkInventoryForItem({"ladder",}, {5})
  15050.             T:checkInventoryForItem({"wired_modem_full"}, {2})
  15051.             T:checkInventoryForItem({"computercraft:cable"}, {57})
  15052.         else    -- conventional storage farm
  15053.             T:checkInventoryForItem({"chest", "barrel"}, {5,5})
  15054.         end
  15055.         R.useBlockType = T:getMostItem("dirt", true) -- exclude dirt from count
  15056.         menu.colourPrint("Checking position...\n", colors.green)
  15057.         os.sleep(2)    -- pause for 2 secs to allow time to press esc
  15058.         retValue = createFarmExtension(R) -- subChoice: 1 = ahead, 2 = to the right
  15059.     elseif R.choice == 33 then  -- manage modular farm
  15060.         -- inventory checked in function depending on option taken
  15061.         R.data = "farm"
  15062.         retValue = manageFarmSetup(R)
  15063.     elseif R.choice == 34 then  -- build single fence
  15064.         utils.checkFuelNeeded(R.length)
  15065.         T:checkInventoryForItem({"wall", "fence"}, {R.length, R.length})
  15066.         if R.torchInterval > 0 then
  15067.             T:checkInventoryForItem({"minecraft:torch"}, {math.ceil(R.length / R.torchInterval)}, false)
  15068.         end
  15069.         R.useBlockType = T:getMostItem("minecraft:torch") -- exclude torch
  15070.         menu.colourPrint("Creating "..R.length.." wall or fence", colors.green)
  15071.         menu.colourPrint("Using: "..R.useBlockType, colors.orange)
  15072.         retValue = createWallOrFence(R) -- barrels not included in R.data, R.width = 0 so only single length
  15073.     elseif R.choice == 35 then  -- build fence
  15074.         utils.checkFuelNeeded(R.width * R.length * 2)
  15075.         local quantity = math.ceil((R.width + R.length) * 2.3)
  15076.         T:checkInventoryForItem({"wall", "fence"}, {quantity, quantity})
  15077.         if R.torchInterval > 0 then
  15078.             T:checkInventoryForItem({"minecraft:torch"}, {math.floor(quantity / R.torchInterval)}, false)
  15079.         end
  15080.         R.useBlockType = T:getMostItem("minecraft:torch") -- exclude torch
  15081.         menu.colourPrint("Creating "..R.width.." x "..R.length.." walled enclosure", colors.green)
  15082.         menu.colourPrint("Using: "..R.useBlockType, colors.orange)
  15083.         retValue = createWallOrFence(R) -- barrels not included in R.data
  15084.     -- OBSIDIAN
  15085.     elseif R.choice == 41 then --harvest obsidian
  15086.         utils.checkFuelNeeded(R.width * R.length * 3)
  15087.         T:checkInventoryForItem({"stone"}, {R.width * R.length})
  15088.         T:checkInventoryForItem({"minecraft:bucket", "minecraft:lava_bucket"}, {1, 1}, false, "Get extra fuel as well!")
  15089.         menu.colourPrint("Harvesting obsidian area: size "..R.width.. " x "..R.length, colors.orange)
  15090.         sleep(2)
  15091.         retValue = harvestObsidian(R)
  15092.     elseif R.choice == 42 then --build nether portal
  15093.         utils.checkFuelNeeded(R.length * R.height * R.width)    -- length = width when facing standard = 5 high 4 length 1 width
  15094.         T:checkInventoryForItem({"minecraft:obsidian"}, {((R.length - 2 + R.height - 2) * R.width * 2)})
  15095.         T:checkInventoryForItem({"stone"}, {R.width * 4})
  15096.         R.useBlockType = T:getMostItem("obsidian", true) -- exclude obsidian from count
  15097.         menu.colourPrint("Building Nether portal", colors.orange)
  15098.         sleep(2)
  15099.         retValue = createPortal(R)
  15100.     elseif R.choice == 43 then --demolish nether portal
  15101.         utils.checkFuelNeeded(20)
  15102.         print("Demolishing Nether portal")
  15103.         retValue = demolishPortal(R)
  15104.     elseif R.choice == 44 then -- Strip mine Netherite
  15105.         utils.checkFuelNeeded(R.length * 2)
  15106.         T:checkInventoryForItem({"stone"}, {R.length * 4})
  15107.         R.useBlockType = T:getMostItem("", true) -- exclude none, use stone only/netherrack
  15108.         T:checkInventoryForItem({"cobble"}, {math.floor(R.length / 16) * 4}, true, "For marking Chunk boundaries")
  15109.         T:checkInventoryForItem({"minecraft:bucket", "minecraft:lava_bucket"}, {1, 1})
  15110.         T:checkInventoryForItem({"minecraft:torch"}, {math.floor(R.length / 16)}, false)
  15111.         retValue = createStripMine(R)
  15112.     elseif R.choice == 45 then --undermine dragon towers
  15113.         utils.checkFuelNeeded(500)
  15114.         T:checkInventoryForItem({"minecraft:cobblestone", "minecraft:cobbled_deepslate"}, {84, 84})
  15115.         print("Undermining dragon towers")
  15116.         retValue = undermineDragonTowers()
  15117.     elseif R.choice == 46 then --deactivate dragon tower
  15118.         utils.checkFuelNeeded(50)
  15119.         print("Deactivating dragon tower")
  15120.         retValue = deactivateDragonTower()
  15121.     elseif R.choice == 47 then --build dragon water trap
  15122.         utils.checkFuelNeeded(256)
  15123.         T:checkInventoryForItem({"stone"}, {356})
  15124.         T:checkInventoryForItem({"minecraft:obsidian"}, {1})
  15125.         T:checkInventoryForItem({"minecraft:ladder"}, {145})
  15126.         T:checkInventoryForItem({"minecraft:water_bucket"}, {1})
  15127.         print("Building dragon water trap")
  15128.         retValue = createDragonTrap()
  15129.     elseif R.choice == 48 then --build portal minecart station
  15130.         utils.checkFuelNeeded(200)
  15131.         menu.colourPrint("Inventory after height measurement", colors.red)
  15132.         menu.colourPrint("Enter to start measurement.", colors.lime)
  15133.         read()
  15134.         print("Building portal platform")
  15135.         retValue = createPortalPlatform()
  15136.     elseif R.choice == 49 then -- shulker harvesting
  15137.         retValue = harvestShulkers(R)
  15138.        
  15139.     -- CANAL BRIDGE
  15140.     elseif R.choice == 51 then  -- continuous path over void/water/lava
  15141.         utils.checkFuelNeeded(R.length) -- allow for R.length
  15142.         if R.data ~= "reduce" then
  15143.             T:checkInventoryForItem({"stone"}, {R.length}, false)
  15144.             if R.torchInterval > 0 then
  15145.                 T:checkInventoryForItem({"minecraft:torch"}, {math.floor(R.length/R.torchInterval)}, false)
  15146.             end
  15147.             print("Building continuous path")
  15148.         end
  15149.         retValue = utils.createPath(R) -- returns {numBlocks}
  15150.     elseif R.choice == 52 then  -- simple 2 block corridor
  15151.         utils.checkFuelNeeded(R.length)
  15152.         T:checkInventoryForItem({"stone"}, {R.length * 2}, false)
  15153.         if R.torchInterval > 0 then
  15154.             T:checkInventoryForItem({"minecraft:torch"}, {math.ceil(R.length / R.torchInterval)}, false)
  15155.         end
  15156.         print(thanks)
  15157.         os.sleep(2)    -- pause for 2 secs to allow time to press esc
  15158.         print("Building simple corridor")
  15159.         retValue = createCorridor(R)
  15160.     elseif R.choice == 53 then  -- canal management
  15161.         local torches = 0
  15162.         local length = R.length
  15163.         if length > 0 then
  15164.             utils.checkFuelNeeded(length * 4) -- allow for 1024 R.length
  15165.             if R.torchInterval > 0 then
  15166.                 torches = math.floor(length / R.torchInterval)
  15167.             end
  15168.         else
  15169.             utils.checkFuelNeeded(2048) -- allow for 1024 R.length
  15170.             length = 256
  15171.         end
  15172.         T:checkInventoryForItem({"stone"}, {length})
  15173.         R.useBlockType = T:getMostItem("", true)
  15174.         if R.data == 1 or (R.data == 2 and (R.subChoice == 2 or R.subChoice == 3 or R.subChoice == 5 or R.subChoice == 6)) then
  15175.             T:checkInventoryForItem({"minecraft:water_bucket"}, {2})
  15176.         end
  15177.         if torches > 0 then
  15178.             T:checkInventoryForItem({"minecraft:torch"}, {torches}, false)
  15179.         end
  15180.         retValue = createWaterCanal(R)
  15181.     elseif R.choice == 54 then  -- ice canal
  15182.         local default = R.length
  15183.         if R.length > 0 then
  15184.             utils.checkFuelNeeded(R.length)
  15185.         else
  15186.             default = 64
  15187.             utils.checkFuelNeeded(default * 2) -- allow for 128 min R.length
  15188.         end
  15189.         --[[
  15190.             new canal
  15191.             1 = left towpath- move to centre left
  15192.             2 = centre left ice or hollow
  15193.             3 = centre right ice or hollow
  15194.             4 = right side - move to centre right
  15195.             convert water canal
  15196.             5 = left towpath
  15197.             6 = centre left ice or air
  15198.             7 = centre right ice or air
  15199.             8 = right towpath
  15200.         ]]
  15201.         if R.subChoice <= 5 or R.subChoice == 8 then    -- towpath
  15202.             T:checkInventoryForItem({"slab"}, {default}, true, "Add slabs to length required")
  15203.         end
  15204.         if R.torchInterval > 0 then
  15205.             --checkInventoryForItem(items, quantities, required, message)
  15206.             T:checkInventoryForItem({"stone"}, {math.ceil(default / R.torchInterval)}, true, "NOT bricks!")
  15207.             R.useBlockType = T:getMostItem("", true)
  15208.             T:checkInventoryForItem({"torch"}, {math.ceil(default / R.torchInterval)}, false)
  15209.         end
  15210.         if R.data == "ice" then -- ice canal with 2 spaces above
  15211.             T:checkInventoryForItem({"minecraft:packed_ice", "minecraft:blue_ice"}, {math.ceil(R.length / 2), math.ceil(R.length / 2)}, false)
  15212.         end
  15213.         print(thanks)
  15214.         os.sleep(2)    -- pause for 2 secs to allow time to press esc
  15215.         print("Building ice canal")
  15216.         retValue = createIceCanal(R)
  15217.     elseif R.choice == 55 then -- platform
  15218.         local volume = R.width * R.length
  15219.         utils.checkFuelNeeded(volume)
  15220.         T:checkInventoryForItem({"stone", "dirt"}, {volume, volume})
  15221.         R.useBlockType = T:getMostItem("", true)
  15222.         print(thanks)
  15223.         os.sleep(2)    -- pause for 2 secs to allow time to press esc
  15224.         print("Building platform")
  15225.         retValue = createPlatform(R)
  15226.     elseif R.choice == 56 then -- sinking platform
  15227.         local volume = (R.width + 1) * (R.length + 1)
  15228.         utils.checkFuelNeeded(volume * (R.height + 1))
  15229.         T:checkInventoryForItem({"stone"}, {volume + ((R.length * 2) + (R.width * 2) * R.height) + 1})
  15230.         print(thanks)
  15231.         os.sleep(2)    -- pause for 2 secs to allow time to press esc
  15232.         print("Building sinking platform")
  15233.         retValue = createSinkingPlatform(R)
  15234.     elseif R.choice == 57 then -- boat bubble lift
  15235.         utils.checkFuelNeeded(R.height * 20)
  15236.         T:checkInventoryForItem({"minecraft:bucket","minecraft:water_bucket"}, {2, 2})
  15237.         T:checkInventoryForItem({"stone"}, {R.height * 10})
  15238.         T:checkInventoryForItem({"gate"}, {R.height * 2})
  15239.         T:checkInventoryForItem({"minecraft:soul_sand"}, {R.height * 2 + 2})
  15240.         print(thanks)
  15241.         os.sleep(2)    -- pause for 2 secs to allow time to press esc
  15242.         print("Building boat bubble lift")
  15243.         retValue = createBoatLift(R)
  15244.         --retValue = createBoatLift(R) -- R.subChoice:0=new, R.subChoice:1=extend, R.length:0=left, 1=right
  15245.        
  15246.     -- MOB SPAWNER
  15247.     elseif R.choice == 61 then  --  9x9 hollow cube cobble lined
  15248.         utils.checkFuelNeeded(600) -- allow for 600 moves
  15249.         T:checkInventoryForItem({"slab","stone"}, {1, 3}, true, "Slab can be crafted from 3 stone")
  15250.         if T:getItemSlot("stone") > 0 and T:getItemSlot("slab") == 0 then -- no slabs, but is stone
  15251.             local craftOK, message = T:craft("slab")
  15252.         end
  15253.         if R.data == "chest" then
  15254.             menu.colourText(nil, "~yellow~Chest(s) will be emptied first\nWait for me to exit and request\n~orange~around 700 stone!\n\n~lightGray~Enter to continue", true)
  15255.             read()
  15256.         else
  15257.             T:checkInventoryForItem({"stone"}, {512}, false, "Full cube uses ~700 blocks\nEstimate your requirements")
  15258.         end
  15259.         print(thanks)
  15260.         os.sleep(2)    -- pause for 2 secs to allow time to press esc
  15261.         retValue = createMobFarmCube(R) -- R.data = "spawner" or "chest": not blaze
  15262.     elseif R.choice == 62 then  -- Blaze spawner
  15263.         utils.checkFuelNeeded(2500) -- allow for 2500 moves
  15264.         if R.data == "blaze" then
  15265.             T:checkInventoryForItem({"slab"}, {122}, true)
  15266.             T:checkInventoryForItem({"stone"}, {576})
  15267.             print("You will be asked for more assets later")
  15268.             print("Enter to continue")
  15269.             read()
  15270.         end  -- else R.data == "restart"
  15271.         retValue = createMobFarmCube(R) --R.data = "blaze" or R.data = "restart"
  15272.     elseif R.choice == 63 then  --  flood mob spawner
  15273.         utils.checkFuelNeeded(60) -- allow for 60 moves
  15274.         T:checkInventoryForItem({"minecraft:water_bucket"}, {2})
  15275.         if R.subChoice == 1 then
  15276.             T:checkInventoryForItem({"fence"}, {2})
  15277.             T:checkInventoryForItem({"sign"}, {2})
  15278.             T:checkInventoryForItem({"slab"}, {1})
  15279.             T:checkInventoryForItem({"minecraft:soul_sand", "minecraft:dirt"}, {1, 1}, true)
  15280.         end
  15281.         print(thanks)
  15282.         os.sleep(2)    -- pause for 2 secs to allow time to press esc
  15283.         retValue = floodMobFarm(R)
  15284.     elseif R.choice == 64 then -- build bubble lift on top of soul sand
  15285.         utils.checkFuelNeeded(200) -- allow for 200 moves
  15286.         T:checkInventoryForItem({"slab","stone"}, {6, 3}, true, "Slabs can be crafted from 3 stone")
  15287.         if T:getItemSlot("stone") > 0 and T:getItemSlot("slab") == 0 then -- no slabs, but is stone
  15288.             local craftOK, message = T:craft("slab")
  15289.         end
  15290.         T:checkInventoryForItem({"minecraft:water_bucket"}, {2})
  15291.         T:checkInventoryForItem({"stone"}, {256})
  15292.         if T:getBlockType("down") ~= "minecraft:soul_sand" then
  15293.             T:checkInventoryForItem({"minecraft:soul_sand"}, {1})
  15294.         end
  15295.         T:checkInventoryForItem({"hopper"}, {1}, false)
  15296.         if T:getItemSlot("hopper") > 0 then
  15297.             T:checkInventoryForItem({"chest"}, {2})
  15298.         end
  15299.         print(thanks)
  15300.         os.sleep(2)    -- pause for 2 secs to allow time to press esc
  15301.         retValue = createMobBubbleLift(R)
  15302.     elseif R.choice == 65 then -- computercraft mob grinder
  15303.         utils.checkFuelNeeded(1000) -- allow for 1000 moves
  15304.         T:checkInventoryForItem({"stone"}, {256}) -- for ceiling, walls and floor of area
  15305.         R.useBlockType = T:getMostItem("", true) -- stone only, no exclusions
  15306.         print(thanks)
  15307.         os.sleep(2)    -- pause for 2 secs to allow time to press esc
  15308.         retValue = createMobGrinder(R)
  15309.     elseif R.choice == 66 then -- build endermen tower
  15310.         -- build in 3 sections, base, tower, top
  15311.         getTaskHelp(2, 66) -- compulsory help display
  15312.         read() -- pause until user ready
  15313.         local numFuel = 700
  15314.         local choices =
  15315.         {
  15316.             "New tower lower base",
  15317.             "Add tower upper base + killzone",
  15318.             "128 block tower to existing base"
  15319.         }
  15320.         pp.itemColours = {colors.lime, colors.orange, colors.green}
  15321.         local option, modifier = menu.menu("Select build stage:", choices, pp)
  15322.         if option == 1 then --lower base
  15323.             T:checkInventoryForItem({"minecraft:chest"}, {1})
  15324.             T:place("chest", "down", false)
  15325.             T:emptyInventory("up")
  15326.             utils.checkFuelNeeded(320) -- allow for 320 moves
  15327.             T:checkInventoryForItem({"stone"}, {144}) -- <3 stacks
  15328.             T:checkInventoryForItem({"minecraft:water_bucket"}, {2})
  15329.             T:checkInventoryForItem({"fence"}, {4})
  15330.             T:checkInventoryForItem({"sign"}, {4})
  15331.             T:checkInventoryForItem({"door"}, {2})
  15332.             T:checkInventoryForItem({"minecraft:soul_sand"}, {1})
  15333.         elseif option == 2 then -- upper base
  15334.             utils.checkFuelNeeded(710) -- allow for 703 moves
  15335.             T:checkInventoryForItem({"stone"}, {384}) -- 6 stacks
  15336.             T:checkInventoryForItem({"minecraft:bucket"}, {4})
  15337.             T:checkInventoryForItem({"fence"}, {15})
  15338.             T:checkInventoryForItem({"sign"}, {4})
  15339.             T:checkInventoryForItem({"ladder"}, {3})
  15340.             T:checkInventoryForItem({"minecraft:soul_sand"}, {1})
  15341.         else -- main tower
  15342.             utils.checkFuelNeeded(3000) -- allow for 3000 moves
  15343.             if T:getBlockType("down") ~= "minecraft:chest" then
  15344.                 T:checkInventoryForItem({"minecraft:chest"}, {1})
  15345.                 T:place("chest", "down", false)
  15346.             end
  15347.             T:checkInventoryForItem({"stone"}, {768}) -- 12 stacks
  15348.             T:checkInventoryForItem({"minecraft:bucket"}, {10})
  15349.             T:checkInventoryForItem({"fence"}, {64})    -- 1 stacks    
  15350.         end
  15351.         print(thanks)
  15352.         os.sleep(2)    -- pause for 2 secs to allow time to press esc
  15353.         retValue = createEnderTower(option) -- 1, 2, or 3
  15354.     elseif R.choice == 67 then -- surround Trial spawner
  15355.         utils.checkFuelNeeded(200) -- allow for 1000 moves
  15356.         T:checkInventoryForItem({"stone"}, {57}) -- for covering spawner
  15357.         R.useBlockType = T:getMostItem("", false) -- use whatever block type available
  15358.         R.height = 3
  15359.         R.width = 5
  15360.         R.length = 5
  15361.         retValue = createTrialCover(R)
  15362.     end
  15363.     return retValue
  15364. end
  15365.  
  15366. local function getTaskInventory(R)
  15367.     -- run this loop 2x per second to check if player has put anything in the inventory
  15368.     -- fuel 1 coal = 60 = 4 planks. 64 planks = 16 coal = 960 units
  15369.     local retValue = {}
  15370.     local thanks = "Thank you.\n\nPress the ESC key\n\nStand Back..\n"
  15371.    
  15372.     if R.choice < 31 then
  15373.         return getTaskInventoryTo30(R) -- Mining, Forestry
  15374.     elseif R.choice < 71 then
  15375.         return getTaskInventoryTo70(R) -- Farming, Obsidian, Canal / Bridge
  15376.     elseif R.choice > 500 and R.choice < 700 then
  15377.         return getTaskInventoryTo70(R)
  15378.     else -- Area carving, Lava/Water, Railway
  15379.         -- AREA CARVING
  15380.         if R.choice == 71 then--Clear area
  15381.             utils.checkFuelNeeded(R.width * R.length * 3)
  15382.             if R.useBlockType == "dirt" then
  15383.                 T:checkInventoryForItem({"minecraft:dirt"}, {R.width * R.length})
  15384.             end
  15385.             print(thanks)
  15386.             sleep(2)
  15387.             print("Clearing area: size "..R.width.. " x "..R.length)
  15388.             retValue = clearArea(R)
  15389.         elseif R.choice == 72 then --Clear rectangle
  15390.             -- R.choice, R.width(R.subChoice), R.length(R.width), up(R.length), down(R.height) from getTask()
  15391.             utils.checkFuelNeeded(R.width * R.length)
  15392.             print("Clearing rectangle: size "..R.width.. " x "..R.length)
  15393.             retValue = clearRectangle(R)
  15394.         elseif R.choice == 73 then --Clear wall
  15395.             utils.checkFuelNeeded(R.length * R.height)
  15396.             print("Removing wall "..R.length.." long x "..R.height.." high")
  15397.             retValue = clearWall(R)
  15398.         elseif R.choice == 74 then --Clear single R.height perimeter wall
  15399.             utils.checkFuelNeeded((R.width + R.length) * 2)
  15400.             print("Recycling wall section "..R.width.." x "..R.length)
  15401.             retValue = clearPerimeter(R)
  15402.         elseif R.choice == 75 then --Clear hollow structure
  15403.             utils.checkFuelNeeded((R.width * R.length) + ((R.width + R.length) * R.height))
  15404.             print("Recycling hollow object "..R.width.." x "..R.length.." height: "..R.height)
  15405.             retValue = clearBuilding(R)
  15406.         elseif R.choice == 76 then --Clear solid structure / extend water pool
  15407.             utils.checkFuelNeeded((R.width * R.length) + ((R.width + R.length) * R.height))
  15408.             print("Recycling solid object w:"..R.width..", l:"..R.length..", h:"..R.height)
  15409.             retValue = clearSolid(R)
  15410.         elseif R.choice == 77 then  -- Dig trench
  15411.             utils.checkFuelNeeded(R.height * R.length * 2)
  15412.             print(thanks)
  15413.             os.sleep(2)    -- pause for 2 secs to allow time to press esc
  15414.             if R.length == 0 then
  15415.                 print("Digging continuous trench "..R.height.." blocks deep")
  15416.             else
  15417.                 print("Digging trench "..R.length.." blocks long, "..R.height.." blocks deep")
  15418.             end
  15419.             retValue = digTrench(R)
  15420.         elseif R.choice == 78 then --Carve mountain
  15421.             utils.checkFuelNeeded(R.length * R.width * 10)
  15422.             print("Carving mountain side "..R.width.." x "..R.length)
  15423.             retValue = clearMountainSide(R)
  15424.         elseif R.choice == 79 then -- Place floor or Ceiling
  15425.             local blocks = R.width * R.length
  15426.             utils.checkFuelNeeded(blocks)
  15427.             T:clear()
  15428.             local inventory = T:getInventory()
  15429.             repeat
  15430.                 T:clear()
  15431.                 --colourText(row, text, reset)
  15432.                 menu.colourText(1, "~yellow~Add floor/ceiling blocks to inventory", false)
  15433.                 menu.colourText(2, "~orange~Multiple types will allow patterns", false)
  15434.                 if blocks - inventory.blockCount > 0 then
  15435.                     menu.colourText(3, "~lime~Add ~red~"..blocks - inventory.blockCount .." ~lime~blocks", false)
  15436.                 else
  15437.                     menu.colourText(3, "~lime~Minimum number of blocks recieved ~red~("..blocks..")", false)
  15438.                     menu.colourText(5, "~yellow~Add more or press ~red~'Enter'~yellow~ to start", true)
  15439.                 end
  15440.                
  15441.                 inventory = T:updateInventory() -- returns inventory or nil if key pressed
  15442.             until inventory == nil
  15443.             --T:checkInventoryForItem({"stone", "log", "planks"}, {blocks, blocks, blocks}, false)
  15444.             R.inventory = T:getInventory()
  15445.             --[[
  15446.                 {
  15447.                 inventory["minecraft:cobblestone"] = 128
  15448.                 inventory["minecraft:stone"] = 64
  15449.                 inventory.names = {minecraft:cobblestone, minecraft:stone}
  15450.                 inventory.quantities = {128, 64}
  15451.                 inventory.blockTypeCount = 2,
  15452.                 inventory.blockCount = 196,
  15453.                 inventory.useBlock = "minecraft:cobblestone"
  15454.                 inventory.mostBlock = "minecraft:cobblestone"
  15455.                 inventory.mostCount = 128
  15456.                 }
  15457.             ]]
  15458.             R.useBlockType = R.inventory.useBlock
  15459.             -- check if > 1 type of material, then use R.data == "random"
  15460.             T:clear()
  15461.             if R.inventory.blockTypeCount > 1 then
  15462.                 local choice, modifier
  15463.                 local choices = {"Random block placement", "Striped pattern"}
  15464.                 pp.itemColours = {colors.lime, colors.orange}
  15465.                 if R.inventory.blockTypeCount == 2 then
  15466.                     table.insert(choices, "Checkerboard pattern")
  15467.                     table.insert(pp.itemColours, colors.green)
  15468.                 end
  15469.                 choice, modifier = menu.menu("Layout pattern?", choices, pp, "Type number + Enter ")
  15470.                 if choice == 1 then
  15471.                     R.data = "random"
  15472.                 elseif choice == 2 then
  15473.                     R.data = "striped"
  15474.                 else
  15475.                     R.data = "checked"
  15476.                 end
  15477.             end
  15478.             retValue = createFloorCeiling(R) -- R.subChoice integer 1 to 4
  15479.         elseif R.choice == 710 then -- direct commands
  15480.             utils.checkFuelNeeded(200)
  15481.             createRectanglePath(R)
  15482.            
  15483. -- LAVA WATER
  15484.         elseif R.choice == 81 then -- Sand based utilities
  15485.             if R.subChoice == 1 then    -- Drop sand or gravel wall
  15486.                 utils.checkFuelNeeded(100)
  15487.                 T:checkInventoryForItem({"sand", "gravel"}, {1024, 1024}, false)
  15488.                 if R.length == 0 then
  15489.                     print("Building sand wall. Auto length: ")
  15490.                 else
  15491.                     print("Building sand wall. length: "..R.length)
  15492.                 end
  15493.                 retValue = createSandWall(R)
  15494.             elseif R.subChoice == 2 then    -- Fill area with sand
  15495.                 utils.checkFuelNeeded(R.length * R.width)
  15496.                 T:checkInventoryForItem({"sand"}, {1024}, false)
  15497.                 print("Filling area with sand. length: "..R.length.." width: "..R.width)
  15498.                 retValue = sandFillArea(R)
  15499.             elseif R.subChoice == 3 then -- Clear sand wall
  15500.                 utils.checkFuelNeeded(200)
  15501.                 if R.length == 0 then
  15502.                     print("Digging sand. Auto length")
  15503.                 else
  15504.                     print("Digging sand. length: "..R.length)
  15505.                 end
  15506.                 retValue = clearSandWall(R)
  15507.             elseif R.subChoice == 4 then    -- Clear sand filled area
  15508.                 utils.checkFuelNeeded(R.length * R.width * 4)
  15509.                 print("Removing sand cube. length: "..R.length.." width: "..R.width)
  15510.                 retValue = clearSandCube(R)
  15511.             end
  15512.        
  15513.         elseif R.choice == 82 then --build containing wall in water or lava
  15514.             utils.checkFuelNeeded(R.length * R.length)
  15515.             local depth = R.height
  15516.             if depth == 0 then
  15517.                 depth = 20
  15518.             end
  15519.             T:checkInventoryForItem({"stone"}, {R.length * depth}, false)
  15520.             print("Building retaining wall in lava/water. length "..R.length)
  15521.             retValue = createRetainingWall(R)          
  15522.         elseif R.choice == 83 then -- create a rectangle path in water/lava
  15523.             utils.checkFuelNeeded(200)
  15524.             createRectanglePath(R)
  15525.         elseif  R.choice == 84 then -- Delete water/lava
  15526.             if R.width == 0 then
  15527.                 utils.checkFuelNeeded(2000)
  15528.                 T:checkInventoryForItem({"stone"}, {256}, false)
  15529.                 print("Deleting water using auto-detection")
  15530.             else
  15531.                 if R.height == 0 then
  15532.                     utils.checkFuelNeeded(2000)
  15533.                     T:checkInventoryForItem({"stone"}, {256}, false)
  15534.                 else
  15535.                     utils.checkFuelNeeded(R.width * R.length * R.height)
  15536.                     T:checkInventoryForItem({"stone"}, {math.max(R.length, R.width) * 2}, false)
  15537.                 end
  15538.                 print("Deleting enclosed water "..R.width.." x "..R.length.." x ".. R.height)
  15539.             end
  15540.             retValue = utils.drainWaterLava(R)
  15541.         elseif  R.choice == 85 then -- Sinking platform
  15542.             local volume = (R.width + 1) * (R.length + 1)
  15543.             utils.checkFuelNeeded(volume * (R.height + 1))
  15544.             T:checkInventoryForItem({"stone"}, {volume + ((R.length * 2) + (R.width * 2) * R.height) + 1})
  15545.             print(thanks)
  15546.             os.sleep(2)    -- pause for 2 secs to allow time to press esc
  15547.             print("Building sinking platform")
  15548.             retValue = createSinkingPlatform(R)
  15549.         elseif R.choice == 86 then -- ocean monument utilities
  15550.             if R.subChoice == 1 then    -- Build 4 corner marker columns
  15551.                 --R.useBlockType = "prismarine", R.data = "oceanMonumentColumns"
  15552.                 T:checkInventoryForItem({"stone"}, {448})
  15553.                 retValue = oceanMonumentColumns(R)
  15554.             elseif R.subChoice == 2 then    -- Retaining wall beween 2 columns
  15555.                 T:checkInventoryForItem({"stone"}, {1024})
  15556.                 retValue = createRetainingWall(R)
  15557.             elseif R.subChoice == 3 then    -- Clear plants pre sand draining
  15558.                 retValue = clearWaterPlants(R)
  15559.             elseif R.subChoice == 4 then    -- Use sand draining
  15560.                 utils.checkFuelNeeded(100)
  15561.                 T:checkInventoryForItem({"sand", "gravel"}, {1024, 1024}, false)
  15562.                 if R.length == 0 then
  15563.                     print("Building sand wall. Auto length: ")
  15564.                 else
  15565.                     print("Building sand wall. length: "..R.length)
  15566.                 end
  15567.                 retValue = createSandWall(R)
  15568.             elseif R.subChoice == 5 then    -- remove sand wall
  15569.                 utils.checkFuelNeeded(200)
  15570.                 print("Digging sand from ocean monument")
  15571.                 retValue = clearSandWall(R)
  15572.             elseif R.subChoice == 6 then    -- Drain and remove structure
  15573.                 T:checkInventoryForItem({"stone"}, {1024})
  15574.                 retValue = utils.drainWaterLava(R)
  15575.             end
  15576.         elseif R.choice == 87 then --ladder to water/lava
  15577.             utils.checkFuelNeeded(R.height * 2)
  15578.             T:checkInventoryForItem({"minecraft:ladder"}, {R.height}, true, "Add more to be safe!")
  15579.             local cobble = R.height * 3 + 10
  15580.             T:checkInventoryForItem({"stone"}, {cobble})
  15581.             print(thanks)
  15582.             os.sleep(2)    -- pause for 2 secs to allow time to press esc
  15583.             print("Creating ladder to bedrock")
  15584.             retValue = createLadderToWater()
  15585.         elseif R.choice == 88 then --remove plants
  15586.             utils.checkFuelNeeded(R.length * R.width * 4)
  15587.             T:checkInventoryForItem({"sand", "stone"}, {64, 64})
  15588.             local width = R.width
  15589.             local length = R.length
  15590.             if width == 0 then
  15591.                 width = "auto"
  15592.             end
  15593.             if length == 0 then
  15594.                 length = "auto"
  15595.             end
  15596.             print("Removing water plants. length: "..length.." width: "..width)
  15597.             retValue = clearWaterPlants(R)
  15598.         elseif R.choice == 89 then -- convert flowing water to source
  15599.             --utils.checkFuelNeeded(R.length * R.width * 4) -- unknown as size not calculated
  15600.             T:checkInventoryForItem({"water_bucket", "bucket"}, {12, 12})
  15601.             T:checkInventoryForItem({"slab"}, {128})
  15602.             print("Converting water to source "..R.width.." x "..R.length.." x ".. R.height)
  15603.             retValue = convertWater(R)
  15604.         elseif R.choice == 810 then -- create sloping water
  15605.             utils.checkFuelNeeded(R.length * R.width * 3)
  15606.             local buckets = math.floor(R.length / 2) + 1
  15607.             T:checkInventoryForItem({"water_bucket", "bucket"}, {buckets, buckets})
  15608.             T:checkInventoryForItem({"slab"}, {R.length * R.width})
  15609.             print("Creating sloping water field "..R.width.." x "..R.length.." x ".. R.height)
  15610.             retValue = createSlopingWater(R)
  15611.         -- BUILDING & RAILWAY
  15612.         elseif R.choice == 91 then -- Build a wall
  15613.             local blocks = R.height * R.length
  15614.             utils.checkFuelNeeded(blocks)
  15615.             T:checkInventoryForItem({"stone", "planks", "bricks"}, {blocks, blocks, blocks})
  15616.             R.useBlockType = T:getMostItem()
  15617.             print("Building a wall using "..R.useBlockType)
  15618.             sleep(2)
  15619.             retValue = buildWall(R)
  15620.         elseif R.choice == 92 then -- Build a rectangular structure
  15621.             local blocks = (R.height * R.length * 2) + (R.height * R.width * 2)
  15622.             utils.checkFuelNeeded(blocks)
  15623.             T:checkInventoryForItem({"stone", "planks", "bricks"}, {blocks, blocks, blocks})
  15624.             R.useBlockType = T:getMostItem()
  15625.             print("Building a house using "..R.useBlockType)
  15626.             sleep(2)
  15627.             retValue = buildStructure(R)
  15628.         elseif R.choice == 93 or R.choice == 94 then -- Build a gable end roof / pitched roof
  15629.             local blocks = ((R.width + 2) * (R.length + 2))
  15630.             utils.checkFuelNeeded(blocks)
  15631.             if menu.getBoolean("Using stairs / planks for roof (y/n)", nil, colors.yellow) then
  15632.                 T:checkInventoryForItem({"planks", "stairs"}, {blocks, blocks})
  15633.             else
  15634.                 T:checkInventoryForItem({"stone"}, {blocks})
  15635.             end
  15636.            
  15637.             R.useBlockType = T:getMostItem()
  15638.             if R.choice == 93 then              -- Build a gableroof
  15639.                 if R.width % 2 == 1 then
  15640.                     T:checkInventoryForItem({"slab"}, {R.length + 2}, false, "Match slabs with roof blocks")
  15641.                 end
  15642.                 blocks = (R.width * 6)
  15643.                 T:checkInventoryForItem({"stone", "planks", "bricks"}, {blocks, blocks, blocks}, true, "Match gable with existing building")
  15644.             else
  15645.                 local isWidthOdd, isLengthOdd, width, length = false, false, 0, 0
  15646.                 R, isWidthOdd, isLengthOdd, width, length = utils.getRoofStats(R)
  15647.                 if isWidthOdd then
  15648.                     T:checkInventoryForItem({"slab"}, {length - 2}, false, "Match slabs with roof blocks")
  15649.                 end
  15650.             end
  15651.             print("Building a roof using "..R.useBlockType)
  15652.             sleep(2)
  15653.             if R.choice == 93 then              -- Build a gableroof
  15654.                 retValue = buildGableRoof(R)
  15655.             else                                -- Build a pitched roof
  15656.                 retValue = buildPitchedRoof(R)
  15657.             end
  15658.         elseif R.choice == 95 or R.choice == 96 then --place redstone torch level or downward slope
  15659.             utils.checkFuelNeeded(10)
  15660.             if R.choice == 95 then
  15661.                 R.data = "level"
  15662.             else
  15663.                 R.data = "up"
  15664.             end
  15665.             T:checkInventoryForItem({"stone"}, {1})
  15666.             R.useBlockType = T:getMostItem("", true)
  15667.             T:checkInventoryForItem({"minecraft:redstone_torch"}, {1})
  15668.             print("Placing redstone torch on ".. R.useBlockType)
  15669.             retValue = placeRedstoneTorch(R)
  15670.         elseif R.choice == 97 or R.choice == 98 then --build downward/upward slope
  15671.             local blocks = R.height * 2
  15672.             if R.height == 0 then
  15673.                 blocks = 64
  15674.             end
  15675.             utils.checkFuelNeeded(blocks)
  15676.             T:checkInventoryForItem({"stone"}, {blocks})
  15677.             print("Building slope")
  15678.             sleep(2)
  15679.             retValue = createRailway(R)
  15680.         elseif R.choice == 101 or R.choice == 102 or R.choice == 103 or R.choice == 104 then -- measure height/depth/length
  15681.             retValue = measure(R)
  15682.         elseif R.choice == 105 then--Borehole
  15683.             retValue = createBorehole(R)
  15684.         elseif R.choice == 111 then --craft an item
  15685.             retValue = utils.craftItem(R)
  15686.         elseif R.choice == 112 then --MysticalAgriculture essence conversion
  15687.             retValue = utils.convertEssence(R)
  15688.         end
  15689.     end
  15690.     return retValue
  15691. end
  15692.  
  15693. local function test(R)
  15694.     local lib = {}
  15695.    
  15696.     function lib.dig(direction, bypass, slot)
  15697.         direction = direction or "forward"
  15698.         slot = slot or 1
  15699.         bypass = bypass or true
  15700.        
  15701.         print("direction: "..direction)
  15702.         print("bypass: "..tostring(bypass))
  15703.         print("slot: "..slot)
  15704.        
  15705.     end
  15706.     lib.dig("down")
  15707.     read()
  15708. end
  15709.  
  15710. local function main()
  15711.     local lib = {}
  15712.    
  15713.     function lib.checkLabel()
  15714.         if os.getComputerLabel() == nil then
  15715.             os.setComputerLabel("toolkit")
  15716.             print("Computer label set to "..os.getComputerLabel())
  15717.         end
  15718.     end
  15719.    
  15720.     function lib.checkLibs(libDir, filename)
  15721.         local fileExists = false
  15722.         if fs.exists(libDir) then
  15723.             if not fs.isDir(libDir) then
  15724.                 fs.move(libDir, libDir.."Renamed")
  15725.                 fs.makeDir(libDir)
  15726.             end
  15727.         else
  15728.             fs.makeDir(libDir)
  15729.         end
  15730.         if fs.exists(fs.combine(libDir, filename)) or fs.exists(fs.combine(libDir, filename..".lua")) then
  15731.             fileExists = true
  15732.         end
  15733.         return fileExists
  15734.     end
  15735.    
  15736.     local doContinue = true
  15737.     lib.checkLabel() -- make sure turtle label is set
  15738.     --check if lib folder exists
  15739.     if not lib.checkLibs("lib", "clsTurtle") then
  15740.         -- use pastebin get to download clsTurtle to libs folder
  15741.         print("Missing clsTurtle.lua in libs directory")
  15742.         print("Attempting to obtain from Pastebin...")
  15743.         if shell.run("pastebin","get","tvfj90gK","lib/clsTurtle.lua") then
  15744.             print("clsTurtle.lua installed from Pastebin")
  15745.         else
  15746.             print("failed to install clsTurtle.lua from Pastebin")
  15747.             doContinue = false
  15748.         end
  15749.     end
  15750.     if not lib.checkLibs("lib", "menu") then
  15751.         -- use pastebin get to download menu.lua to libs folder
  15752.         print("Missing menu.lua in libs directory")
  15753.         print("Attempting to obtain from Pastebin...")
  15754.         if shell.run("pastebin","get","BhjbYsw4","lib/menu.lua") then
  15755.             print("menu.lua installed from Pastebin")
  15756.         else
  15757.             print("failed to install menu.lua from Pastebin")
  15758.             doContinue = false
  15759.         end
  15760.     end
  15761.     if doContinue then
  15762.         local result = {}
  15763.         local R =
  15764.         {
  15765.             choice = 0,
  15766.             currentLevel = 0,
  15767.             subChoice = 0,
  15768.             size = 0,
  15769.             width = 0,
  15770.             length = 0,
  15771.             height = 0,
  15772.             depth = 0,
  15773.             up = false,
  15774.             down = false,
  15775.             silent = false,
  15776.             data = {},
  15777.             torchInterval = 0,
  15778.             useBlockType = "",
  15779.             auto = false,
  15780.             side = "",
  15781.             direction = "",
  15782.             ready = false,
  15783.             networkFarm = false,
  15784.             mysticalAgriculture = false,
  15785.             logType = "",
  15786.             treeSize = "",
  15787.             message = "",
  15788.             inventory = {}
  15789.         }
  15790.         menu = require("lib.menu")
  15791.         T = require("lib.clsTurtle").new(false) -- true enables logfile to log.txt note dot NOT colon
  15792.         T:clear()
  15793.         doContinue = false  -- reset
  15794.         if args[1] ~= nil then
  15795.             if args[1]:sub(1,1) == "h" then
  15796.                 local help =
  15797. [[... = any following characters
  15798.  
  15799. tk v...     = mc/ccTweaked versions
  15800. tk log      = enable logging
  15801. tk log d... = enable logging + debug
  15802. tk find     = writes locate.txt
  15803. tk test     = runs test(R)
  15804. tk farm     = runs manageFarm(R)
  15805.  
  15806.  
  15807.  
  15808. Enter to exit]]
  15809.                 menu.colourPrint(help, colours.yellow)
  15810.                 read()
  15811.             elseif args[1] == "log" then
  15812.                 if args[2] ~= nil then
  15813.                     if args[2]:sub(1,1) == "d" then
  15814.                         dbug = true -- set dbug flag
  15815.                         menu.colourPrint("Logging and debugging enabled", colors.lime)
  15816.                     end
  15817.                 else
  15818.                     menu.colourPrint("Logging enabled", colors.lime)
  15819.                 end
  15820.                 if T:getLogExists() then
  15821.                     if menu.getBoolean("Delete existing log file? (y/n)", 3, colors.orange) then
  15822.                         T:deleteLog()
  15823.                         menu.colourPrint("Log file deleted", colors.yellow)
  15824.                     end
  15825.                 end
  15826.                 T:setUseLog(true)
  15827.                 doContinue = true
  15828.                 utils.waitForInput()
  15829.                 --if dbug then
  15830.                     --menu.colourPrint("Enter to continue...", colors.lightBlue)
  15831.                     --read()
  15832.                 --end
  15833.             elseif args[1] == "farm" then
  15834.                 R.silent = true
  15835.                 R.data = "farm"
  15836.                 R.auto = true
  15837.                 manageFarm(R)
  15838.             --elseif args[1] == "tree" then
  15839.                 --R.silent = true
  15840.                 --R.data = "treefarm"
  15841.                 --R.auto = true
  15842.                 --manageTreeFarm(R) -- use file to read status
  15843.             elseif args[1] == "find" then
  15844.                 -- missing turtle: player used 'tk find'
  15845.                 T:setUseLog(true)
  15846.                 T:setLogFileName("locate.txt")
  15847.                 T:appendLine("Booting succeeded")
  15848.                 T:appendLine("Block ahead: "..T:getBlockType("forward"))
  15849.                 T:appendLine("Block above: "..T:getBlockType("up"))
  15850.                 T:appendLine("Block below: "..T:getBlockType("down"))
  15851.             elseif args[1] == "test" then
  15852.                 test(R)
  15853.             elseif args[1]:find("v") ~= nil then
  15854.                 print("_HOST:")
  15855.                 print()
  15856.                 print(_HOST)
  15857.                 print()
  15858.                 print("Minecraft major version: "..mcMajorVersion)
  15859.                 print("Minecraft minor version: "..mcMinorVersion)
  15860.                 print("ccTweaked major version: "..ccMajorVersion)
  15861.                 print("ccTweaked minor version: "..ccMinorVersion)
  15862.                 print("tk version:              "..tkVersion)
  15863.                 print("clsTurtle version:       "..version)
  15864.                 print("\nEnter to exit")
  15865.                 read()
  15866.             end
  15867.         else
  15868.             doContinue = true
  15869.         end
  15870.         if doContinue then
  15871.             print("Minecraft major version: "..mcMajorVersion)
  15872.             print("Bedrock level: "..bedrock)
  15873. -- T:setUseLog(use, filename, delete)
  15874. --T:setUseLog(true, "log.txt", true)
  15875.             if T:getUseLog() then
  15876.                 if T:saveToLog("Started with logging enabled", true) then
  15877.                     menu.colourPrint("\nEnter to continue...", colors.lightBlue)
  15878.                     read()
  15879.                 end
  15880.             else
  15881.                 print("Logging disabled")
  15882.             end
  15883.             sleep(1)
  15884.             while R.choice == 0 do
  15885.                 R = chooseTask(R)
  15886.             end
  15887.             if R.choice > 0 then
  15888.                 R = getTask(R)
  15889.                 if R.data ~= "quit" then
  15890.                     result = getTaskInventory(R) -- table of comments
  15891.                 end
  15892.             end
  15893.         end
  15894.         T:clear()
  15895.         table.insert(result, "Thank you for using 'survival toolkit'")
  15896.         local clr = {colors.yellow, colors.orange, colors.green, colors.lightBlue}
  15897.         local count = 1
  15898.         for _, value in ipairs(result) do
  15899.             --print(value)
  15900.             --.print(text, fg, bg, width)
  15901.             menu.colourPrint(tostring(value), clr[count])
  15902.             count = count + 1
  15903.             if count > #clr then
  15904.                 count = 1
  15905.             end
  15906.         end
  15907.     else
  15908.         print("Add missing files and restart")
  15909.     end
  15910. end
  15911.  
  15912. main()
Add Comment
Please, Sign In to add comment