Inksaver

clsTurtle.lua (Turtle Class):2024/11/02

Feb 8th, 2016 (edited)
2,733
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
Lua 113.14 KB | None | 0 0
  1. version = 20241102.0800
  2. --[[
  3.     https://pastebin.com/tvfj90gK
  4.     Last edited: see version YYYYMMDD.HHMM
  5.     save as T.lua, preferably in /lib folder
  6.     as is not meant to be run directly from CraftOS command line
  7.     usage:
  8.     T = require("lib/T").new()
  9.     T:clear()
  10.     T:forward(2)
  11.     To use logging
  12.     T = require("lib.T"):new(true) -- true enables logfile to log.txt
  13.    
  14.     Computercraft started with mc version 1.7.10 and went to 1.8.9
  15.     ccTweaked started around mc 1.12 and currently at 1.21.1
  16.     mc 1.18 has new blocks and bedrock at -64, so needs to be taken into account.
  17.     _HOST = The ComputerCraft and Minecraft version of the current computer environment.
  18.     For example, ComputerCraft 1.93.0 (Minecraft 1.15.2).
  19.    
  20.     This version no longer uses the 'damage' property found in older versions.
  21. ]]
  22.  
  23. local g = 0.3 -- time for gravity to work on sand/gravel
  24. local bedrock = 0
  25. local ceiling = 255
  26. local deletesWater = false
  27. local mcMajorVersion = tonumber(_HOST:sub(_HOST:find("Minecraft") + 10, _HOST:find("\)") - 1)) -- eg 1.18 or 1.20 -> 1.18, 1.20
  28.  
  29. function string:Split(sSeparator, nMax, bRegexp, noEmpty)
  30.     --[[return a table split with sSeparator. noEmpty removes empty elements
  31.         use: tblSplit = SplitTest:Split('~',[nil], [nil], false) or tblSplit = string.Split(SplitTest, '~')]]  
  32.     assert(sSeparator ~= '','separator must not be empty string')
  33.     assert(nMax == nil or nMax >= 1, 'nMax must be >= 1 and not nil')
  34.     if noEmpty == nil then noEmpty = true end
  35.  
  36.     local aRecord = {}
  37.     local newRecord = {}
  38.  
  39.     if self:len() > 0 then
  40.         local bPlain = not bRegexp
  41.         nMax = nMax or -1
  42.  
  43.         local nField, nStart = 1, 1
  44.         local nFirst,nLast = self:find(sSeparator, nStart, bPlain)
  45.         while nFirst and nMax ~= 0 do
  46.             aRecord[nField] = self:sub(nStart, nFirst-1)
  47.             nField = nField+1
  48.             nStart = nLast+1
  49.             nFirst,nLast = self:find(sSeparator, nStart, bPlain)
  50.             nMax = nMax-1
  51.         end
  52.         aRecord[nField] = self:sub(nStart)
  53.        
  54.         if noEmpty then --split on newline preserves empty values
  55.             for i = 1, #aRecord do
  56.                 if aRecord[i] ~= "" then
  57.                     table.insert(newRecord, aRecord[i])
  58.                 end
  59.             end
  60.         else
  61.             newRecord = aRecord
  62.         end
  63.     end
  64.    
  65.     return newRecord
  66. end
  67.  
  68. if tonumber(mcMajorVersion) == nil then -- 1.18.3 NAN
  69.     mcMajorVersion = tonumber(_HOST:sub(_HOST:find("Minecraft") + 10, _HOST:find("\)") - 3)) -- eg 1.19.4 -> 1.19
  70. end
  71. if mcMajorVersion < 1.7  and mcMajorVersion >= 1.18 then -- 1.12 to 1.??
  72.     bedrock = -64
  73.     ceiling = 319
  74. end
  75. if mcMajorVersion < 1.7  and mcMajorVersion <= 1.12 then -- 1.12 to 1.??
  76.     deletesWater = true
  77. end
  78. local stone = {
  79.                 "minecraft:cobblestone",
  80.                 "minecraft:mossy_cobblestone",
  81.                 "minecraft:netherrack",
  82.                 "minecraft:blackstone",
  83.                 "minecraft:gilded_blackstone",
  84.                 "minecraft:polished_blackstone",
  85.                 "minecraft:polished_blackstone_bricks",
  86.                 "minecraft:cracked_polished_blackstone_bricks",
  87.                 "minecraft:chiseled_polished_blackstone",
  88.                 "minecraft:basalt",
  89.                 "minecraft:deepslate",
  90.                 "minecraft:cobbled_deepslate",
  91.                 "minecraft:chiseled_deepslate",
  92.                 "minecraft:polished_deepslate",
  93.                 "minecraft:deepslate_bricks",
  94.                 "minecraft:cracked_deepslate_bricks",
  95.                 "minecraft:deepslate_tiles",
  96.                 "minecraft:cracked_deepslate_tiles",
  97.                 "minecraft:tuff",
  98.                 "minecraft:tuff_bricks",
  99.                 "minecraft:polished_tuff",
  100.                 "minecraft:chiseled_tuff",
  101.                 "minecraft:chiseled_tuff_bricks",
  102.                 "minecraft:granite",
  103.                 "minecraft:diorite",
  104.                 "minecraft:andesite",
  105.                 "minecraft:end_stone",
  106.                 "minecraft:obsidian",
  107.                 "minecraft:stone",
  108.                 "minecraft:smooth_stone",
  109.                 "minecraft:stone_bricks",
  110.                 "minecraft:cracked_stone_bricks",
  111.                 "minecraft:chiseled_stone_bricks",
  112.                 "minecraft:mossy_stone_bricks",
  113.                 "minecraft:sandstone",
  114.                 "minecraft:smooth_sandstone",
  115.                 "minecraft:chiseled_sandstone",
  116.                 "minecraft:cut_sandstone",
  117.                 "minecraft:red_sandstone",
  118.                 "minecraft:smooth_red_sandstone",
  119.                 "minecraft:chiseled_red_sandstone",
  120.                 "minecraft:cut_red_sandstone",
  121.                 "minecraft:obsidian",
  122.                 "minecraft:dirt",
  123.                 "minecraft:glass",
  124.                 "minecraft:purpur_block",
  125.                 "minecraft:terracotta",
  126.                 "minecraft:white_terracotta",
  127.                 "minecraft:red_terracotta",
  128.                 "minecraft:orange_terracotta",
  129.                 "minecraft:yellow_terracotta",
  130.                 "minecraft:brown_terracotta",
  131.                 "minecraft:light_gray_terracotta"
  132.               } -- must be exact mc names!
  133.              
  134. local flowers =
  135. {
  136.     "minecraft:sapling",
  137.     "minecraft:oak_sapling",
  138.     "minecraft:spruce_sapling",
  139.     "minecraft:birch_sapling",
  140.     "minecraft:jungle_sapling",
  141.     "minecraft:acacia_sapling",
  142.     "minecraft:dark_oak_sapling",
  143.     "minecraft:leaves",
  144.     "minecraft:oak_leaves",
  145.     "minecraft:spruce_leaves",
  146.     "minecraft:birch_leaves",
  147.     "minecraft:jungle_leaves",
  148.     "minecraft:acacia_leaves",
  149.     "minecraft:dark_oak_leaves",
  150.     "minecraft:azalea_leaves",
  151.     "minecraft:flowering_azalea_leaves",
  152.     "minecraft:dandelion",
  153.     "minecraft:poppy",
  154.     "minecraft:blue_orchid",
  155.     "minecraft:allium",
  156.     "minecraft:azure_bluet",
  157.     "minecraft:red_tulip",
  158.     "minecraft:orange_tulip",
  159.     "minecraft:white_tulip",
  160.     "minecraft:pink_tulip",
  161.     "minecraft:oxeye_daisy",
  162.     "minecraft:cornflower",
  163.     "minecraft:lily_of_the_valley",
  164.     "minecraft:sunflower",
  165.     "minecraft:lilac",
  166.     "minecraft:rose_bush",
  167.     "minecraft:peony",
  168.     "minecraft:wither_rose",
  169.     "minecraft:spore_blossom",
  170.     "minecraft:fern",
  171.     "minecraft:large_fern",
  172.     "minecraft:grass",
  173.     "minecraft:tall_grass",
  174.     "minecraft:azalea",
  175.     "minecraft:flowering_azalea",
  176.     "minecraft:deadbush",
  177.     "minecraft:dead_bush",
  178.     "minecraft:seagrass",
  179.     "minecraft:tall_seagrass",
  180.     "minecraft:sea_pickle",
  181.     "minecraft:brown_mushroom",
  182.     "minecraft:red_mushroom",
  183.     "minecraft:crimson_fungus",
  184.     "minecraft:warped_fungus",
  185.     "minecraft:crimson_roots",
  186.     "minecraft:warped_roots",
  187.     "minecraft:vine",
  188.     "minecraft:vines",
  189.     "minecraft:weeping_vines",
  190.     "minecraft:twisting_vines",
  191.     "minecraft:sugar_cane",
  192.     "minecraft:kelp",
  193.     "minecraft:kelp_plant",
  194.     "minecraft:moss_block",
  195.     "minecraft:moss_carpet",
  196.     "minecraft:hanging_roots",
  197.     "minecraft:big_dripleaf",
  198.     "minecraft:small_dripleaf",
  199.     "minecraft:bamboo",
  200.     "minecraft:cactus",
  201.     "minecraft:glow_lichen",
  202.     "minecraft:waterlily",
  203.     "minecraft:lily_pad",
  204.     "minecraft:pumpkin",
  205.     "minecraft:melon",
  206.     "minecraft:melon_block",
  207.     "minecraft:cocoa",
  208.     "minecraft:double_plant",
  209.     "minecraft:sponge",
  210.     "minecraft:wet_sponge"
  211. }
  212.  
  213. local maSeeds = {
  214.     "mysticalagriculture:prosperity_seed_base",
  215.     "mysticalagriculture:soulium_seed_base",
  216.     "mysticalagriculture:air_seeds",
  217.     "mysticalagriculture:earth_seeds",
  218.     "mysticalagriculture:water_seeds",
  219.     "mysticalagriculture:fire_seeds",
  220.     "mysticalagriculture:inferium_seeds",
  221.     "mysticalagriculture:stone_seeds",
  222.     "mysticalagriculture:dirt_seeds",
  223.     "mysticalagriculture:wood_seeds",
  224.     "mysticalagriculture:ice_seeds",
  225.     "mysticalagriculture:deepslate_seeds",
  226.     "mysticalagriculture:nature_seeds",
  227.     "mysticalagriculture:dye_seeds",
  228.     "mysticalagriculture:nether_seeds",
  229.     "mysticalagriculture:coal_seeds",
  230.     "mysticalagriculture:coral_seeds",
  231.     "mysticalagriculture:honey_seeds",
  232.     "mysticalagriculture:amethyst_seeds",
  233.     "mysticalagriculture:pig_seeds",
  234.     "mysticalagriculture:chicken_seeds",
  235.     "mysticalagriculture:cow_seeds",
  236.     "mysticalagriculture:sheep_seeds",
  237.     "mysticalagriculture:squid_seeds",
  238.     "mysticalagriculture:fish_seeds",
  239.     "mysticalagriculture:slime_seeds",
  240.     "mysticalagriculture:turtle_seeds",
  241.     "mysticalagriculture:armadillo_seeds",
  242.     "mysticalagriculture:rubber_seeds",
  243.     "mysticalagriculture:silicon_seeds",
  244.     "mysticalagriculture:sulfur_seeds",
  245.     "mysticalagriculture:aluminum_seeds",
  246.     "mysticalagriculture:saltpeter_seeds",
  247.     "mysticalagriculture:apatite_seeds",
  248.     "mysticalagriculture:iron_seeds",
  249.     "mysticalagriculture:copper_seeds",
  250.     "mysticalagriculture:nether_quartz_seeds",
  251.     "mysticalagriculture:glowstone_seeds",
  252.     "mysticalagriculture:redstone_seeds",
  253.     "mysticalagriculture:obsidian_seeds",
  254.     "mysticalagriculture:prismarine_seeds",
  255.     "mysticalagriculture:zombie_seeds",
  256.     "mysticalagriculture:skeleton_seeds",
  257.     "mysticalagriculture:spider_seeds",
  258.     "mysticalagriculture:creeper_seeds",
  259.     "mysticalagriculture:rabbit_seeds",
  260.     "mysticalagriculture:tin_seeds",
  261.     "mysticalagriculture:bronze_seeds",
  262.     "mysticalagriculture:zinc_seeds",
  263.     "mysticalagriculture:brass_seeds",
  264.     "mysticalagriculture:silver_seeds",
  265.     "mysticalagriculture:lead_seeds",
  266.     "mysticalagriculture:graphite_seeds",
  267.     "mysticalagriculture:gold_seeds",
  268.     "mysticalagriculture:lapis_lazuli_seeds",
  269.     "mysticalagriculture:end_seeds",
  270.     "mysticalagriculture:experience_seeds",
  271.     "mysticalagriculture:breeze_seeds",
  272.     "mysticalagriculture:blaze_seeds",
  273.     "mysticalagriculture:ghast_seeds",
  274.     "mysticalagriculture:enderman_seeds",
  275.     "mysticalagriculture:steel_seeds",
  276.     "mysticalagriculture:nickel_seeds",
  277.     "mysticalagriculture:constantum_seeds",
  278.     "mysticalagriculture:electrum_seeds",
  279.     "mysticalagriculture:invar_seeds",
  280.     "mysticalagriculture:uranium_seeds",
  281.     "mysticalagriculture:ruby_seeds",
  282.     "mysticalagriculture:sapphire_seeds",
  283.     "mysticalagriculture:peridot_seeds",
  284.     "mysticalagriculture:soulium_seeds",
  285.     "mysticalagriculture:diamond_seeds",
  286.     "mysticalagriculture:emerald_seeds",
  287.     "mysticalagriculture:netherite_seeds",
  288.     "mysticalagriculture:wither_skeleton_seeds",
  289.     "mysticalagriculture:platinum_seeds",
  290.     "mysticalagriculture:iridium_seeds",
  291.     "mysticalagriculture:nether_star_seeds",
  292.     "mysticalagriculture:dragon_egg_seeds"
  293. }
  294.              
  295. local T = {}
  296. T.__index = T
  297.  
  298. setmetatable(T,
  299. {
  300.     __call = function (cls, ...)
  301.     return cls.new(...)
  302.     end,
  303. })
  304. -- if you want to pass arguments at construction...
  305. function T.new(useLog) --note dot, NOT colon, list of args or ... table
  306.     local self = setmetatable({}, T)
  307.     if useLog == nil then useLog = false end
  308.     self.x = 0
  309.     self.y = 0
  310.     self.z = 0
  311.     self.facing = 0
  312.     self.compass = ""
  313.     self.equippedLeft = ""
  314.     self.equippedRight = ""
  315.     self.placeSlot = 0
  316.     self.placeItem = ""
  317.     self.osVersion = os.version() -- eg CraftOS 1.8
  318.     self.userBlocks = {}
  319.     self.useLog = useLog
  320.     if self.useLog then
  321.         print("Logging enabled")
  322.         sleep(1.5)
  323.     end
  324.     self.logFileName = "log.txt"
  325.     self.logFileExists = false
  326.     return self
  327. end
  328.  
  329. -- helper function for iterating lists
  330. function T:values(t) -- general diy iterator
  331.     local i = 0
  332.     return function()
  333.         i = i + 1
  334.         return t[i]
  335.     end
  336. end
  337. --[[
  338.     logging methods set from calling program
  339.     T:setLogFileName("log.txt") (example)
  340.     T:deleteLog() -- if new log required every time program is run
  341.     T:setUseLog(true) -- allows logging statements to be left in this class code, will only operate if this flag is set#
  342.     extensive use for debugging has been left in the function T.checkInventoryForItem() as this was a nightmare to get working
  343. ]]
  344. function T:getUseLog()
  345.     return self.useLog
  346. end
  347. function T:setUseLog(use, filename, delete)
  348.     self.logFileName = filename or "log.txt"
  349.     if delete ~= nil then
  350.         if delete then
  351.             self:deleteLog()
  352.         end
  353.     end
  354.     self.useLog = use
  355.     return use
  356. end
  357. function T:getLogExists()
  358.     local exists = false
  359.     if fs.exists(self.logFileName) then
  360.         exists = true
  361.     end
  362.     return exists
  363. end
  364. function T:getLogFileName()
  365.     return self.logFileName
  366. end
  367. function T:setLogFileName(value)
  368.     self.logFileName = value
  369. end
  370. function T:getCurrentFileSize()    
  371.     if self.logFileExists then
  372.         return fs.getSize(self.logFileName)
  373.     else
  374.         return 0
  375.     end
  376. end
  377. function T:deleteLog()     
  378.     if fs.exists(self.logFileName) then
  379.         fs.delete(self.logFileName)
  380.     end
  381.     self.logFileExists = false
  382.    
  383.     return true
  384. end
  385. function T:appendLine(newText)
  386.     local handle = ""
  387.    
  388.     if fs.exists(self.logFileName) then --logFile already created
  389.         handle = fs.open(self.logFileName, "a")
  390.     else
  391.         handle = fs.open(self.logFileName, "w") --create file
  392.     end
  393.     self.logFileExists = true
  394.     handle.writeLine(newText)
  395.     handle.close()
  396. end
  397. function T:saveToLog(text, toScreen)
  398.     if toScreen == nil then
  399.         toScreen = true
  400.     end
  401.     if text ~= "" and text ~= nil then
  402.         if toScreen then
  403.             print(text)
  404.         end
  405.         if self.useLog then
  406.             self:appendLine(text)
  407.             return true
  408.         end
  409.     end
  410.     return false
  411. end
  412. -- getters and setters
  413. function T:getValue() return self.value end --debug test for object creation with args
  414. function T:getX() return self.x end
  415. function T:setX(newVal) self.x = newVal end
  416. function T:getY() return self.y end
  417. function T:setY(newVal) self.y = newVal end
  418. function T:getZ() return self.z end
  419. function T:setZ(newVal) self.z = newVal end
  420. function T:getFacing() return self.facing end
  421. function T:setFacing(newVal)
  422.     local direction = {"south", "west", "north", "east"}
  423.     self.facing = newVal
  424.     if self.facing < 0 then
  425.         self.facing = 3
  426.     elseif self.facing > 3 then
  427.         self.facing = 0
  428.     end
  429.     self.compass = direction[self.facing + 1] --+1 to allow for lua indexing at 1
  430. end
  431. function T:getCompass() return self.compass end
  432. function T:getPlaceItem() return self.placeItem end
  433. function T:setPlaceItem(item)
  434.     local success = false
  435.     local slot = self:getItemSlot(item)
  436.     if slot > 0 then
  437.         self.placeItem = item
  438.     end
  439. end
  440. function T:setEquipped(side, value)
  441.     if side == "left" then
  442.         self.equippedLeft = value
  443.     elseif side == "right" then
  444.         self.equippedRight = value
  445.     end
  446. end
  447. function T:getUserBlocks()
  448.     -- self.userBlocks[1] = {string<name>, string<item>, int<count>, bool<userChoice>}
  449.     return self.userBlocks
  450. end
  451. function T:getUserBlockType(name)
  452.     local retValue = {} --retValue["minecraft:stone_pressure_plate"] = 2
  453.     if next(self.userBlocks) ~= nil then
  454.         for i = 1, #self.userBlocks do
  455.             if self.userBlocks[i][1] == name then
  456.                 retValue[self.userBlocks[i][2]] = self.userBlocks[i][3]
  457.             end
  458.         end
  459.     end
  460.     return retValue --retValue["minecraft:stone_pressure_plate"] = 2, retValue["minecraft:oak_pressure_plate"] = 2
  461. end
  462. function T:addUserBlocks(blockTable)
  463.     table.insert(self.userBlocks, blockTable)
  464. end
  465. function T:getMASeeds()
  466.     -- return list of Mystical Agriculture seeds
  467.     return maSeeds
  468. end
  469. -- change direction and movement methods
  470. function T:attack(direction)   
  471.     direction = direction or "all"
  472.     turtle.select(1)
  473.     local success = false
  474.     local attackLimit = 30 -- won't get in infinite loop attacking a minecart chest
  475.     local Attack
  476.     local up, down, forward = true, true, true
  477.     if direction == "up" then
  478.         Attack = turtle.attackUp
  479.     elseif direction == "down" then
  480.         Attack = turtle.attackDown
  481.     elseif direction == "forward" then
  482.         Attack = turtle.attack
  483.     end
  484.     if direction == "all" then
  485.         while up or down or forward do
  486.             forward = turtle.attack()
  487.             up = turtle.attackUp()
  488.             down = turtle.attackDown()
  489.             if up or down or forward then -- attack in at least 1 direction succeeeded
  490.                 sleep(0.5)
  491.             end
  492.             attackLimit = attackLimit - 1
  493.             if attackLimit <= 0 then
  494.                 break
  495.             end
  496.         end
  497.     else
  498.         while Attack() do --in case mob around
  499.             sleep(0.5)
  500.             attackLimit = attackLimit - 1
  501.             if attackLimit <= 0 then
  502.                 break
  503.             end        
  504.         end
  505.     end
  506.  
  507.     if attackLimit > 0 then
  508.         success = true
  509.     end
  510.     return success
  511. end
  512.  
  513. function T:doMoves(numBlocksRequested, direction)
  514.     local errorMsg = nil
  515.     local numBlocksMoved = 0
  516.     local Move, Dig, Detect
  517.     local bypass = false
  518.     local bypassCount = 0
  519.     local blockType
  520.     numBlocksRequested = numBlocksRequested or 1
  521.     self:refuel(steps)
  522.     turtle.select(1)
  523.     -- re-assign turtle functions to new variables
  524.     if direction == "forward" or direction == "back" then
  525.         Move = turtle.forward
  526.         Dig = turtle.dig
  527.         Detect = turtle.detect
  528.     elseif direction == "up" then
  529.         Move = turtle.up
  530.         Dig = turtle.digUp
  531.         Detect = turtle.detectUp
  532.     else
  533.         Move = turtle.down
  534.         Dig = turtle.digDown
  535.         Detect = turtle.detectDown
  536.     end
  537.    
  538.     if direction == "back" then
  539.         self:turnRight(2)
  540.     end
  541.     if numBlocksRequested == 0 then
  542.         return 0
  543.     end
  544.     for i = 1, numBlocksRequested, 1 do
  545.         local digOK, digError
  546.         if bypass then
  547.             bypassCount = bypassCount + 1
  548.             if bypassCount >= 1 then
  549.                 bypass = false
  550.                 bypassCount = 0
  551.             end
  552.         else
  553.             if Detect() then -- block above/forward/below
  554.                 blockType, bypass = self:checkNoDigBlocks(direction)
  555.                 if bypass then --moved round bedrock, spawner, turtle or full chest
  556.                     digOK, digError = Dig()
  557.                     if digOK then
  558.                         sleep(g) -- allow sand / gravel to drop if digging forward / up
  559.                     else -- unable to dig, or nothing to dig
  560.                         if digError:lower():find("unbreakable") ~= nil then -- different messages between versions all contain 'unbreakable'
  561.                             errorMsg = digError
  562.                             print(digError)
  563.                             break
  564.                         else -- not bedrock, could be mob or minecart
  565.                             self:attack()
  566.                         end
  567.                     end
  568.                 end
  569.             end
  570.             if not bypass then
  571.                 local moveOK, moveError = Move() -- try to move forward/up/down
  572.                 if moveOK then
  573.                     numBlocksMoved = numBlocksMoved + 1
  574.                     self:changeCoords(direction)
  575.                 else
  576.                     while not moveOK do -- did not move if obstruction
  577.                         digOK, digError = Dig()
  578.                         if digOK then
  579.                             sleep(g) -- allow sand / gravel to drop if digging forward / up
  580.                         else -- unable to dig, or nothing to dig
  581.                             if digError:lower():find("unbreakable") ~= nil then -- different messages between versions all contain 'unbreakable'
  582.                                 errorMsg = digError
  583.                                 print(digError)
  584.                                 break
  585.                             else -- not bedrock, could be mob or minecart
  586.                                 self:attack()
  587.                             end
  588.                         end
  589.                         moveOK, moveError = Move() -- try to move forward/up/down again
  590.                         if moveOK then
  591.                             numBlocksMoved = numBlocksMoved + 1
  592.                             self:changeCoords(direction)
  593.                         end
  594.                     end
  595.                 end
  596.             end
  597.         end
  598.     end
  599.    
  600.     if direction == "back" then
  601.         self:turnRight(2)
  602.     end
  603.    
  604.     return numBlocksMoved, errorMsg
  605. end
  606.  
  607. function T:back(steps)
  608.     steps = steps or 1
  609.     local success = false
  610.     local blocksMoved, errorMsg = self:doMoves(steps, "back")
  611.     if blocksMoved == steps then
  612.         success = true
  613.     end
  614.     return success, blocksMoved, errorMsg
  615. end
  616.  
  617. function T:changeCoords(direction)
  618.     --  0 = go south (z increases)
  619.     --  1 = go west  (x decreases)
  620.     --  2 = go north (z decreases
  621.     --  3 = go east  (x increases)
  622.     if direction == "forward" then
  623.         if self.facing == 0 then
  624.             self.z = self.z + 1
  625.         elseif self.facing == 1 then
  626.             self.x = self.x - 1
  627.         elseif self.facing == 2 then
  628.             self.z = self.z - 1
  629.         else
  630.             self.x = self.x + 1
  631.         end
  632.     elseif direction == "back" then
  633.         if self.facing == 0 then
  634.             self.z = self.z - 1
  635.         elseif self.facing == 1 then
  636.             self.x = self.x + 1
  637.         elseif self.facing == 2 then
  638.             self.z = self.z + 1
  639.         else
  640.             self.x = self.x - 1
  641.         end
  642.     elseif direction == "up" then
  643.         self.y = self.y + 1
  644.     elseif direction == "down" then
  645.         self.y = self.y - 1
  646.     end
  647. end
  648.  
  649. function T:down(steps, getBlock)
  650.     steps = steps or 1
  651.     if steps == 0 then
  652.         return true, 0, "", ""
  653.     end
  654.     if getBlock == nil then getBlock = false end
  655.     local success = false
  656.     local blockType = nil
  657.     local blocksMoved, errorMsg = self:doMoves(steps, "down")
  658.     if blocksMoved == steps then
  659.         success = true
  660.     end
  661.     if getBlock then
  662.         blockType = self:getBlockType("down")
  663.     end
  664.     return success, blocksMoved, errorMsg, blockType
  665. end
  666.  
  667. function T:forward(steps)    
  668.     steps = steps or 1
  669.     local success = false
  670.     local blocksMoved, errorMsg = self:doMoves(steps, "forward")
  671.     if blocksMoved == steps then
  672.         success = true
  673.     end
  674.     return success, blocksMoved, errorMsg
  675. end
  676.  
  677. function T:turnLeft(steps)
  678.     steps = steps or 1
  679.     for i = 1, steps do
  680.         turtle.turnLeft()
  681.         self.facing = self.facing - 1
  682.         if self.facing < 0 then
  683.             self.facing = 3
  684.         end
  685.     end
  686. end
  687.  
  688. function T:turnRight(steps)
  689.     steps = steps or 1
  690.     for i = 1, steps do
  691.         turtle.turnRight()
  692.         self.facing = self.facing + 1
  693.         if self.facing > 3 then
  694.             self.facing = 0
  695.         end
  696.     end
  697. end
  698.  
  699. function T:up(steps)
  700.     steps = steps or 1
  701.     if steps == 0 then
  702.         return true, 0, ""
  703.     end
  704.     local success = false
  705.     local blocksMoved, errorMsg = self:doMoves(steps, "up")
  706.     if blocksMoved == steps then
  707.         success = true
  708.     end
  709.     return success, blocksMoved, errorMsg
  710. end
  711. -- other methods
  712. function T:checkInventoryForItem(items, quantities, required, message)
  713.     local lib = {}
  714.    
  715.     function lib.checkInventory(self, items, quantities, inventory)
  716.         --[[
  717.         Does the inventory have any or all of the items required?
  718.         Multiple choices are similar alternatives eg cobblestone, dirt. Most requests are single item
  719.         eg requested: {"stone", "dirt"}, quantities: { 256, 256 }
  720.         eg inventory: {"minecraft:cobblestone" = 128, "minecraft:dirt" = 100}  = 228: still need 28 cobble or dirt
  721.         could be empty!
  722.         ]]
  723.         -- on first entry currentItems = {}
  724.         local waiting = true            -- return value. set to false if not enough put in the inventory for this item
  725.         local quantitiesFound = {}      -- table of quantities found, based on no of items needed
  726.         for i = 1, #quantities do       -- Initialise table eg {0, 0, 0}   
  727.             table.insert(quantitiesFound, 0)
  728.         end
  729.         for i = 1, #items do                                -- check if item(s) already present
  730.             local findItem = items[i]
  731.             for k, v in pairs(inventory) do                 -- eg: {"minecraft:cobblestone" = 128, "minecraft:dirt" = 203}
  732.                 if findItem:find("\:") ~= nil then          -- specific item requested eg "minecraft:cobblestone"
  733.                     if findItem == k then                   -- exact match eg "minecraft:cobblestone" requested, "minecraft:cobblestone" found
  734.                         item = k                            -- item = "minecraft:cobblestone"
  735.                         quantitiesFound[i] = v
  736.                     end
  737.                 else                                        -- non specific eg "log", "pressure_plate", "stone"
  738.                     if findItem == "stone" then             -- check for allowed blocks in global table stone
  739.                         for _, stonetype in ipairs(stone) do
  740.                             if k == stonetype then          -- k = "minecraft:cobblestone" matches with stonetype
  741.                                 quantitiesFound[i] = quantitiesFound[i] + v
  742.                             end
  743.                         end
  744.                     elseif k:find(findItem) ~= nil then     -- similar item exists already eg "pressure_plate" is in "minecraft:stone_pressure_plate"
  745.                         quantitiesFound[i] = quantitiesFound[i] + v
  746.                     end
  747.                 end
  748.             end
  749.         end
  750.         local totalFound = 0
  751.         for i = 1, #quantities do
  752.             totalFound = totalFound + quantitiesFound[i]
  753.             if totalFound >= quantities[i] then
  754.                 waiting = false
  755.             end
  756.         end
  757.         if waiting then -- update quantities
  758.             for i = 1, #quantitiesFound do
  759.                 quantitiesFound[i] = totalFound -- when player asked to supply alternatives, this gives quantities
  760.             end
  761.         end
  762.         return waiting, quantitiesFound
  763.     end
  764.    
  765.     --[[
  766.     function lib.getInventory(self)
  767.         -- table eg: {"minecraft:cobblestone" = 256, "minecraft:cobbled_deepslate = 256"}
  768.         local inventory = {}
  769.         for slot = 1, 16 do
  770.             local slotContains, slotCount = self:getSlotContains(slot)
  771.             if slotContains ~= "" then -- eg "minecraft:cobblestone"
  772.                 if inventory[slotContains] ~= nil then --already exists in inventory
  773.                     inventory[slotContains] = inventory[slotContains] + slotCount --update quantities
  774.                 else
  775.                     inventory[slotContains] = slotCount
  776.                 end
  777.             end
  778.         end
  779.         return inventory
  780.     end]]
  781.        
  782.     if required == nil then required = true end
  783.     if message == nil then message = "" end     -- add further instructions
  784.     --local inventory = lib.getInventory(self)  -- create table of blocktypes and quantities
  785.     local inventory = self:getInventoryItems()  -- create table of blocktypes and quantities
  786.     local inventoryChanged = false
  787.     local enteringLoop = true
  788.     local waiting = true
  789.     local quantitiesFound = {}
  790.     while waiting do -- true on first entry, also enteringLoop = true check slots and inventory in waiting loop to see if new items are added
  791.         --T.clear(self)
  792.         if inventoryChanged or enteringLoop then
  793.             self:clear()
  794.             enteringLoop = false --flag reset loop has iterated at least once
  795.             inventoryChanged = false
  796.             waiting, quantitiesFound = lib.checkInventory(self, items, quantities, inventory) -- are we still waiting, table of found quantities
  797.             if waiting then --insufficient items so ask for more
  798.                 if message ~= "" then
  799.                     print(message)
  800.                 end
  801.                 if quantitiesFound[1] < quantities[1] then
  802.                     print("Add "..quantities[1] - quantitiesFound[1].." "..self:trimItemName(items[1]).." to any slot(s)")
  803.                     for i = 2, #items do
  804.                         print("Or add "..quantities[i] - quantitiesFound[i].." "..self:trimItemName(items[i]).." to any slot(s)")
  805.                     end
  806.                 end
  807.                 if not required then
  808.                     print("(Optional: 'Enter' if not required)")
  809.                 end
  810.             end
  811.         end
  812.         if waiting then -- not enough onBoard, so wait for user to interact with inventory or press Enter
  813.             local event, param1 = os.pullEvent()
  814.             if event == "turtle_inventory" then
  815.                 inventoryChanged = true
  816.                 -- user has added, moved or removed inventory items
  817.                 --inventory = lib.getInventory(self)
  818.                 inventory = self:getInventoryItems()
  819.             --elseif event == "key"  and not required then
  820.             elseif event == "key" then
  821.                 if param1 == keys.enter then
  822.                     self:clear()
  823.                     return nil -- stop waiting and exit
  824.                 end
  825.             end
  826.         end
  827.     end
  828.     if required then
  829.         return quantitiesFound
  830.     else
  831.         return nil --if item not required
  832.     end
  833. end
  834.  
  835. function T:checkNoDigBlocks(direction, moveRound)
  836.     if moveRound == nil then
  837.         moveRound = false
  838.     end
  839.     local bypass = false
  840.     local isSpawner = false
  841.     local blockType = self:getBlockType(direction)
  842.     -- if mob spawner or chest found, go round it. Do not break!
  843.     if blockType ~= "" then
  844.         if blockType:find("spawner") ~= nil then
  845.             self:writeCoords("SpawnerCoords.txt")
  846.             bypass = true
  847.             isSpawner = true
  848.             print("Spawner Found!")
  849.         elseif blockType:find("turtle") ~= nil then --do not break another turtle
  850.             bypass = true
  851.         elseif blockType:find("chest") ~= nil or blockType:find("minecart") ~= nil then
  852.             local success, msg
  853.             repeat
  854.                 success, msg = self:suck(direction)
  855.             until not success
  856.             if self:getFirstEmptySlot() == 0 then -- turtle is full
  857.                 bypass = true
  858.             else
  859.                 self:dig(direction)
  860.             end
  861.         end
  862.     end
  863.     if bypass and (moveRound or isSpawner)  then
  864.         if direction == "up" then
  865.             self:go("F1U2R2F1R2", false, 0, false, false)
  866.         elseif direction == "forward" then
  867.             self:go("U1F2D1", false, 0, false, false)
  868.         elseif direction == "down" then
  869.             self:go("F1D2R2F1R2", false, 0, false, false)
  870.         end
  871.     end
  872.     return blockType, bypass -- bypass true should be used to reduce steps in calling routine
  873. end
  874.  
  875. function T:clear()
  876.     term.clear()
  877.     term.setCursorPos(1,1)
  878. end
  879.  
  880. function T:craft(item, quantity)   
  881.     --[[
  882.         eg stairs, 40
  883.         setup for crafting chest, planks, stairs
  884.     ]]
  885.     local craftOK = false
  886.     local message = ""
  887.     local chestSlot = 0
  888.     local holeSlot = 0
  889.     local sourceSlot = 0
  890.     local turns = 0
  891.    
  892.     chestSlot = self:getItemSlot("chest") --get the slot number containing a chest
  893.     if chestSlot == 0 then -- chest not found
  894.         if item == "planks" then -- could be new turtle and tree felling
  895.             sourceSlot = self:getItemSlot("log") --get the slot number containing log(s)   
  896.             if sourceSlot > 0 then
  897.                 if turtle.craft() then
  898.                     return true, ""
  899.                 else
  900.                     return false, "Unable to craft planks"
  901.                 end
  902.             end
  903.         elseif item == "chest" then -- assume chest needed and logs are onboard
  904.             sourceSlot = self:getItemSlot("log") --get the slot number containing log(s)   
  905.             if sourceSlot > 0 then
  906.                 if turtle.craft() then-- craft logs into planks
  907.                     sourceSlot = self:getItemSlot("planks") --get the slot number containing planks
  908.                     if sourceSlot > 0 then             
  909.                         turtle.select(sourceSlot)
  910.                         turtle.transferTo(16) --move crafting item to 16
  911.                         turtle.select(16)
  912.                         turtle.transferTo(1, 1)
  913.                         turtle.transferTo(2, 1)
  914.                         turtle.transferTo(3, 1)
  915.                         turtle.transferTo(5, 1)
  916.                         turtle.transferTo(7, 1)
  917.                         turtle.transferTo(9, 1)
  918.                         turtle.transferTo(10, 1)
  919.                         turtle.transferTo(11, 1)
  920.                         if turtle.craft() then
  921.                             return true, ""
  922.                         else
  923.                             return false, "Unable to craft chest"
  924.                         end
  925.                     end
  926.                 else
  927.                     return false, "Unable to craft planks"
  928.                 end
  929.             else
  930.                 return false, "No logs available"
  931.             end
  932.         elseif item == "slab" then -- assume 3 stone -> slab for mob spawner
  933.             sourceSlot = self:getItemSlot("stone") --get the slot number containing stone
  934.             if sourceSlot > 0 then
  935.                 turtle.select(sourceSlot)
  936.                 turtle.transferTo(16) --move crafting item to 16
  937.                 turtle.select(16)
  938.                 turtle.transferTo(1, 1)
  939.                 turtle.transferTo(2, 1)
  940.                 turtle.transferTo(3, 1)
  941.                 for i = 4, 16 do
  942.                     if turtle.getItemCount(i) > 0 then
  943.                         turtle.select(i)
  944.                         turtle.dropUp()
  945.                     end
  946.                 end
  947.                 if turtle.craft(1) then
  948.                     return true, ""
  949.                 else
  950.                     return false, "Unable to craft slab"
  951.                 end
  952.             end
  953.         else
  954.             return false, "No chest for crafting"
  955.         end
  956.     end
  957.     local stock = {}
  958.     --[[
  959.         rt.total = total
  960.         rt.mostSlot = slotData.mostSlot
  961.         rt.leastSlot = slotData.leastSlot
  962.         rt.mostCount = slotData.mostCount
  963.         rt.leastCount = slotData.leastCount
  964.         ]]
  965.     if item:find("stairs") ~= nil then -- craft stairs, check stone, planks
  966.         stock = self:getStock("stone", -1) --get the slot number containing stone
  967.         sourceSlot = stock.mostSlot
  968.         message = "No stone for crafting stairs"
  969.     elseif item:find("chest") ~= nil then -- craft chest
  970.         stock = self:getStock("planks", -1) --get the slot number containing planks
  971.         sourceSlot = stock.mostSlot
  972.         message = "No planks for crafting chest"
  973.     elseif item:find("planks") ~= nil then -- craft planks
  974.         stock = self:getStock("log", -1) --get the slot number containing logs
  975.         sourceSlot = stock.mostSlot
  976.         message = "No logs for crafting planks"
  977.     end
  978.     if sourceSlot == 0 then
  979.         return false, message
  980.     end
  981.     while turtle.detect() do --check for clear space to place chest
  982.         self:turnRight(1)
  983.         turns = turns + 1
  984.         if turns == 4 then
  985.             turns = 0
  986.             break
  987.         end
  988.     end
  989.     turtle.select(1)
  990.     while turtle.detect() do --clear space in front. Use loop in case of sand/gravel falling
  991.         turtle.dig()
  992.         sleep(g)
  993.     end
  994.     turtle.select(chestSlot) --should be slot with chest
  995.     while not turtle.place() do
  996.         self:attack()
  997.     end
  998.     -- fill chest with everything except required items
  999.     for i = 1, 16 do
  1000.         if i ~= sourceSlot then
  1001.             turtle.select(i)
  1002.             turtle.drop()
  1003.         end
  1004.     end
  1005.     -- error here if turtle empty
  1006.     turtle.select(sourceSlot)
  1007.     turtle.transferTo(16) --move crafting item to 16
  1008.     --ready to craft
  1009.     turtle.select(16)
  1010.     if item:find("planks") ~= nil then -- crafting planks
  1011.         turtle.transferTo(1, quantity / 4)
  1012.     elseif item:find("chest") ~= nil then  --craft("chest", 1)
  1013.         --8 planks = 1 chest
  1014.         turtle.transferTo(1, 1)
  1015.         turtle.transferTo(2, 1)
  1016.         turtle.transferTo(3, 1)
  1017.         turtle.transferTo(5, 1)
  1018.         turtle.transferTo(7, 1)
  1019.         turtle.transferTo(9, 1)
  1020.         turtle.transferTo(10, 1)
  1021.         turtle.transferTo(11, 1)
  1022.     elseif item:find("stairs") ~= nil then  --craft("stairs", 40)
  1023.         --6 cobblestone = 4 stairs
  1024.         turtle.transferTo(1, math.ceil(quantity / 4))
  1025.         turtle.transferTo(5, math.ceil(quantity / 4))
  1026.         turtle.transferTo(6, math.ceil(quantity / 4))
  1027.         turtle.transferTo(9, math.ceil(quantity / 4))
  1028.         turtle.transferTo(10, math.ceil(quantity / 4))
  1029.         turtle.transferTo(11, math.ceil(quantity / 4))
  1030.     end
  1031.     turtle.select(16)
  1032.     turtle.drop() --drop remaining resources before crafting
  1033.     -- Attempt to craft item into slot 16
  1034.     if turtle.craft() then
  1035.         craftOK = true
  1036.         --now put crafted item in chest first, so will mix with any existing similar items
  1037.         turtle.drop()
  1038.     else --crafting not successful, so empty out all items into chest
  1039.         for i = 1, 16 do
  1040.             if turtle.getItemCount(i) > 0 then
  1041.                 turtle.select(i)
  1042.                 turtle.drop()
  1043.             end
  1044.         end
  1045.     end
  1046.     turtle.select(1) --empty chest into slot 1 onwards
  1047.     while turtle.suck() do end
  1048.     turtle.dig() -- collect chest
  1049.     if turns > 0 then --return to original position
  1050.         self:turnLeft(turns)
  1051.     end
  1052.    
  1053.     return craftOK, message
  1054. end
  1055.  
  1056. function T:createTunnel(length, allowAbandon)
  1057.     -- T.go(self, path, useTorch, torchInterval, leaveExisting)
  1058.     -- assume at floor level at start
  1059.     local leaveExisting = true
  1060.     local useTorch = false
  1061.     local distance  = 1
  1062.     local blockAbove = ""
  1063.     local blockBelow = ""
  1064.     local waterCountAbove = 0
  1065.     local waterCountBelow = 0
  1066.     local onGround = true
  1067.     for i = 1, length do
  1068.         if onGround then -- 1, 3, 5, 7 etc
  1069.             blockBelow = self:getBlockType("down")
  1070.             if blockBelow ~= "" then
  1071.                 if blockBelow:find("lava") ~= nil then
  1072.                     self:go("C2L1C1R2C1L1", useTorch, 0, leaveExisting)
  1073.                 elseif blockBelow:find("water") ~= nil then
  1074.                     self:go("C2", useTorch, 0, leaveExisting)
  1075.                     waterCountBelow = waterCountBelow + 1
  1076.                 else
  1077.                     self:go("C2", useTorch, 0, leaveExisting)
  1078.                 end
  1079.             else
  1080.                 self:go("C2", useTorch, 0, leaveExisting)
  1081.             end
  1082.             self:up(1)
  1083.             onGround = false
  1084.             blockAbove = self:getBlockType("up")
  1085.             if blockAbove ~= "" then
  1086.                 if blockAbove:find("lava") ~= nil then
  1087.                     self:go("C0L1C1R2C1L1", useTorch, 0, leaveExisting)
  1088.                 elseif blockAbove:find("water") ~= nil then
  1089.                     self:go("C0L1C1R2C1L1", useTorch, 0, leaveExisting)
  1090.                     waterCountAbove = waterCountAbove + 1
  1091.                 else
  1092.                     self:go("C0", useTorch, 0, leaveExisting)
  1093.                 end
  1094.             else
  1095.                 self:go("C0", useTorch, 0, leaveExisting)
  1096.             end
  1097.             -- if on first block check behind
  1098.             if i == 1 then
  1099.                 self:go("R2C1R2", useTorch, 0, leaveExisting)
  1100.             end
  1101.             if distance >= 8 then
  1102.                 if distance % 8 == 0 then -- 8th or other position
  1103.                     self:go("t5", useTorch, 0, false)
  1104.                 end
  1105.             end
  1106.         else -- at ceiling 2, 4, 6, 8 etc
  1107.             blockAbove = self:getBlockType("up")
  1108.             if blockAbove ~= "" then
  1109.                 if blockAbove:find("lava") ~= nil then
  1110.                     self:go("C0L1C1R2C1L1", useTorch, 0, leaveExisting)
  1111.                 elseif blockAbove:find("water") ~= nil then
  1112.                     self:go("C0L1C1R2C1L1", useTorch, 0, leaveExisting)
  1113.                     waterCountAbove = waterCountAbove + 1
  1114.                 else
  1115.                     self:go("C0", useTorch, 0, leaveExisting)
  1116.                 end
  1117.             else
  1118.                 self:go("C0", useTorch, 0, leaveExisting)
  1119.             end
  1120.             if distance == 2 then
  1121.                 self:go("t1", useTorch, 0, false)
  1122.             end
  1123.             self:down(1)
  1124.             onGround = true
  1125.             blockBelow = self:getBlockType("down")
  1126.             if blockBelow ~= "" then
  1127.                 if blockBelow:find("lava") ~= nil then
  1128.                     self:go("C2L1C1R2C1L1", useTorch, 0, leaveExisting)
  1129.                 elseif blockBelow:find("water") ~= nil then
  1130.                     self:go("C2", useTorch, 0, leaveExisting)
  1131.                     waterCountBelow = waterCountBelow + 1
  1132.                 else
  1133.                     self:go("C2", useTorch, 0, leaveExisting)
  1134.                 end
  1135.             else
  1136.                 self:go("C2", useTorch, 0, leaveExisting)
  1137.             end
  1138.         end
  1139.         -- now move forward if length > 1
  1140.         if length > 1 then
  1141.             if i < length then -- not on last iteration
  1142.                 self:forward(1)
  1143.                 distance = distance + 1
  1144.             else -- on last iteration
  1145.                 if not onGround then
  1146.                     self:go("C1", useTorch, 0, leaveExisting)
  1147.                     self:down(1)
  1148.                     onGround = true
  1149.                 end
  1150.             end
  1151.         else -- one unit only so move down
  1152.             self:down(1)
  1153.             onGround = true
  1154.         end
  1155.        
  1156.         if allowAbandon then
  1157.             if waterCountAbove + waterCountBelow >= 6 then
  1158.                 if not onGround then
  1159.                     self:down(1)
  1160.                     onGround = true
  1161.                 end
  1162.                 break
  1163.             end
  1164.         end
  1165.     end
  1166.    
  1167.     return distance -- 1 to length. cut short if 3 or more water
  1168. end
  1169.  
  1170. function T:detect(direction)
  1171.     direction = direction or "forward"
  1172.    
  1173.     local Detect = turtle.detect
  1174.     if direction == "up" then
  1175.         Detect = turtle.detectUp
  1176.     elseif direction == "down" then
  1177.         Detect = turtle.detectDown
  1178.     end
  1179.    
  1180.     return Detect()
  1181. end
  1182.  
  1183. function T:dig(direction, bypass, slot)
  1184.     --[[ To dig a chest use T:dig(direction, false)  ]]
  1185.     direction = direction or "forward"
  1186.     bypass = bypass or true -- allows digging any block including chests and spawners
  1187.     slot = slot or 1
  1188.     local success = false
  1189.     local blockType = ""
  1190.     local Dig = turtle.dig
  1191.     if direction == "up" then
  1192.         Dig = turtle.digUp
  1193.     elseif direction == "down" then
  1194.         Dig = turtle.digDown
  1195.     end
  1196.    
  1197.     turtle.select(slot)
  1198.     if bypass then
  1199.         blockType, bypass = self:checkNoDigBlocks(direction, false)
  1200.     end
  1201.     if not bypass then --bypass true if chest, turtle or minecart
  1202.         while Dig() do
  1203.             sleep(g)
  1204.             success = true
  1205.         end
  1206.     end
  1207.     turtle.select(1)
  1208.     return success
  1209. end
  1210.  
  1211. function T:digGravityBlock(direction)
  1212.     local Dig = turtle.dig
  1213.     if direction == "up" then
  1214.         Dig = turtle.digUp
  1215.     elseif direction == "down" then
  1216.         Dig = turtle.digDown
  1217.     end
  1218.     local blockType = self:getBlockType(direction)
  1219.     turtle.select(1)
  1220.     if blockType:find("sand") ~= nil or blockType:find("gravel") ~= nil then
  1221.         Dig()
  1222.         return true
  1223.     else
  1224.         return false
  1225.     end
  1226.  
  1227. end
  1228.  
  1229. function T:digValuable(direction)
  1230.     local Dig = turtle.dig
  1231.     if direction == "up" then
  1232.         Dig = turtle.digUp
  1233.     elseif direction == "down" then
  1234.         Dig = turtle.digDown
  1235.     end
  1236.     local isValuable, blockType = self:isValuable(direction)
  1237.     if isValuable then
  1238.         Dig()
  1239.         return true
  1240.     else --check for lava
  1241.         if blockType:find("lava") ~= nil then
  1242.             self:place("minecraft:bucket", direction)  -- will automatically find empty bucket and refuel
  1243.             return true
  1244.         else
  1245.             return false
  1246.         end
  1247.     end
  1248.     turtle.select(1)
  1249. end
  1250.  
  1251. function T:drop(direction, slot, amount)
  1252.     direction = direction or "forward"
  1253.     if type(slot) == "number" then
  1254.         slot = slot or 1
  1255.     elseif type(slot) == "string" then
  1256.         slot = self:getItemSlot(slot)
  1257.     end
  1258.     local success = false
  1259.     local drop = turtle.drop
  1260.     if direction == "up" then
  1261.         drop = turtle.dropUp
  1262.     elseif direction == "down" then
  1263.         drop = turtle.dropDown
  1264.     end
  1265.     if slot > 0 then
  1266.         turtle.select(slot)
  1267.  
  1268.         if amount == nil then
  1269.             success = drop()
  1270.         else
  1271.             success = drop(amount)
  1272.         end
  1273.     end
  1274.     turtle.select(1)
  1275.     return success
  1276. end
  1277.  
  1278. function T:dropAll(direction)
  1279.     direction = direction or "forward"
  1280.     local Drop = turtle.drop
  1281.     local success = true
  1282.     if direction == "up"  then
  1283.         Drop = turtle.dropUp
  1284.     elseif direction == "down" then
  1285.         Drop = turtle.dropDown
  1286.     end
  1287.     for i = 1, 16 do
  1288.         if turtle.getItemCount(i) > 0 then
  1289.             turtle.select(i)
  1290.             if not Drop() then
  1291.                 success = false
  1292.             end
  1293.         end
  1294.     end
  1295.     turtle.select(1)
  1296.     return success
  1297. end
  1298.  
  1299. function T:dropItem(item, direction, keepAmount, except)
  1300.     --[[ eg T:dropItem("essence", direction, 0, {"inferium", "prudentium", "tertium","imperium", "supremium"})
  1301.         drop all MysticalAgriculture essences except those listed
  1302.     ]]
  1303.     direction = direction or "forward"
  1304.     except = except or {}
  1305.     local itemSlot = 0
  1306.     local stockData = {}
  1307.     local success = false
  1308.    
  1309.     if #except > 0 then -- drop everything except specified
  1310.         for slot = 1, 16 do
  1311.             local itemInSlot = self:getSlotContains(slot)   -- eg mysticalagriculture:inferium_essence
  1312.             if itemInSlot ~= "" then
  1313.                 local keep = false
  1314.                 if itemInSlot:find(item) ~= nil then        -- slot contains partial match
  1315.                     for _, v in ipairs(except) do           -- {"inferium", "prudentium", "tertium","imperium", "supremium"}
  1316.                         if itemInSlot:find(v) ~= nil then   -- eg inferium_essence
  1317.                             keep = true
  1318.                             break
  1319.                         end
  1320.                     end
  1321.                     if not keep then                -- not an excluded item: drop it
  1322.                         success = self:drop(direction, slot)
  1323.                     end
  1324.                 end
  1325.             end
  1326.         end
  1327.     elseif keepAmount == nil or keepAmount <= 0 then -- drop everything
  1328.         itemSlot = self:getItemSlot(item)
  1329.         while itemSlot > 0 do
  1330.             if not self:drop(direction, itemSlot) then
  1331.                 return false
  1332.             end
  1333.             itemSlot = self:getItemSlot(item)
  1334.         end
  1335.     else -- keep a specific amount
  1336.         -- {rt.total, rt.mostSlot, rt.leastSlot, rt.mostCount, rt.leastCount}
  1337.         stockData = self:getStock(item)
  1338.         while stockData.total > keepAmount do
  1339.             if stockData.total - stockData.leastCount > keepAmount then
  1340.                 success = self:drop(direction, stockData.leastSlot)
  1341.             else
  1342.                 success = self:drop(direction, stockData.leastSlot, stockData.total - keepAmount)
  1343.             end
  1344.             if not success then
  1345.                 break
  1346.             end
  1347.             stockData = self:getStock(item)
  1348.         end
  1349.     end
  1350.     turtle.select(1)
  1351.     return success
  1352. end
  1353.  
  1354. function T:dumpRefuse(direction, keepCobbleStacks)
  1355.     --dump dirt, cobble, sand, gravel
  1356.     local Drop = turtle.drop
  1357.     if direction == "up" then
  1358.         Drop = turtle.dropUp
  1359.     else
  1360.         Drop = turtle.dropDown
  1361.     end
  1362.     keepCobbleStacks = keepCobbleStacks or 0
  1363.     local itemlist = {"gravel", "stone", "sand", "flint"}
  1364.     local cobbleCount = 0
  1365.  
  1366.     for i = 1, 16 do
  1367.         local blockType, slotCount,  blockModifier = self:getSlotContains(i)
  1368.        
  1369.         if blockType:find("cobble") ~= nil or blockType:find("netherrack") then
  1370.             if cobbleCount > keepCobbleStacks then
  1371.                 turtle.select(i)
  1372.                 Drop()
  1373.             else
  1374.                 cobbleCount = cobbleCount + 1
  1375.             end
  1376.         end
  1377.         for j = 1, #itemlist do
  1378.             if blockType:find(itemlist[j]) ~= nil then
  1379.                 turtle.select(i)
  1380.                 Drop()
  1381.                 break
  1382.             end
  1383.         end
  1384.     end
  1385.     turtle.select(1)
  1386. end
  1387.  
  1388. function T:emptyInventory(direction)
  1389.     --[[ Dump everything!]]
  1390.     direction = direction or "down"
  1391.     local Drop = turtle.dropDown
  1392.     if direction == "up" then
  1393.         Drop = turtle.dropUp
  1394.     elseif direction == "forward" then
  1395.         Drop = turtle.drop
  1396.     end
  1397.     for i = 1, 16 do
  1398.         turtle.select(i)
  1399.         Drop()
  1400.     end
  1401.     turtle.select(1)
  1402. end
  1403.  
  1404. function T:emptyInventorySelection(direction, exceptions, quantities)
  1405.     --[[ Dump everything except items in exceptions eg {"oak_sapling"}, {64} ]]
  1406.     self:sortInventory()
  1407.     direction = direction or "down"
  1408.     local Drop = turtle.dropDown
  1409.     if direction == "up" then
  1410.         Drop = turtle.dropUp
  1411.     elseif direction == "forward" then
  1412.         Drop = turtle.drop
  1413.     end
  1414.     for i = 1, 16 do
  1415.         turtle.select(i)
  1416.         if turtle.getItemCount(i) > 0 then
  1417.             local item = turtle.getItemDetail(i)
  1418.             local name = item.name
  1419.             local inKeepItems = false
  1420.             local index = 0
  1421.             for j = 1, #exceptions do
  1422.                 --print("Checking "..i.." "..name.." with "..exceptions[j])
  1423.                 if name:find(exceptions[j]) ~= nil then
  1424.                     inKeepItems = true
  1425.                     index = j
  1426.                 end
  1427.             end
  1428.             if inKeepItems then
  1429.                 --print(name.." found "..quantities[index].." needed")
  1430.                 if quantities[index] > 0 then
  1431.                     local dropAmount = item.count - quantities[index]
  1432.                     quantities[index] = quantities[index] - item.count
  1433.                     if quantities[index] < 0 then
  1434.                         quantities[index] = 0
  1435.                     end
  1436.                     if dropAmount > 0 then
  1437.                         Drop(dropAmount)
  1438.                     end
  1439.                     -- else if 0 do not drop
  1440.                 end
  1441.             else
  1442.                 Drop()
  1443.             end
  1444.         end
  1445.     end
  1446.     turtle.select(1)
  1447. end
  1448.  
  1449. function T:emptyTrash(direction)
  1450.     direction = direction or "down"
  1451.     local Drop = turtle.dropDown
  1452.     if direction == "up" then
  1453.         Drop = turtle.dropUp
  1454.     elseif direction == "forward" then
  1455.         Drop = turtle.drop
  1456.     end
  1457.     local slotData = {}
  1458.     local itemName = ""
  1459.     local move = false
  1460.     -- store these items permanently inside turtle
  1461.     local keepItems =   {"netherrack", "cobble", "chest", "torch", "ore", "bucket", "coal", "diamond", "debris", "deepslate","iron","gold","copper"}           
  1462.     local keepit = false                   
  1463.     -- empty excess cobble, dirt, all gravel, unknown minerals
  1464.     -- keep max of 1 stack
  1465.     self:sortInventory(false) -- do not use chest for sorting as may leave items behind
  1466.     for i = 1, 16 do
  1467.         keepit = false
  1468.         if turtle.getItemCount(i) > 0 then
  1469.             itemName = self:getItemName(i) -- eg 'minecraft:andesite'
  1470.             for _,v in pairs(keepItems) do
  1471.                 --if v == item then
  1472.                 if itemName:find(v) ~= nil then
  1473.                     keepit = true
  1474.                     break
  1475.                 end
  1476.             end
  1477.             if not keepit then
  1478.                 turtle.select(i)
  1479.                 Drop()
  1480.                 sleep(g)
  1481.             end
  1482.         end
  1483.     end
  1484.     self:sortInventory(false)
  1485.     self:emptyTrashItem(direction, "minecraft:cobblestone", 192)
  1486.     self:emptyTrashItem(direction, "minecraft:netherrack", 192)
  1487.     self:emptyTrashItem(direction, "minecraft:cobbled_deepslate", 192)
  1488.     slotData = self:getStock("minecraft:coal", 0)
  1489.     if slotData.total > 64 then
  1490.         if slotData.mostSlot ~= slotData.leastSlot and slotData.leastSlot ~= 0 then
  1491.             turtle.select(slotData.leastSlot)
  1492.             turtle.refuel()
  1493.         end
  1494.     end
  1495.     turtle.select(1)
  1496. end
  1497.  
  1498. function T:emptyTrashItem(direction, item, keepAmount)
  1499.     --[[ deprecated. left for legacy applications ]]
  1500.     self:dropItem(item, direction, keepAmount)
  1501. end
  1502.  
  1503. function T:equip(side, useItem)
  1504.     --slotData.leastSlot, slotData.leastModifier, total, slotData
  1505.     local slot = self:getItemSlot(useItem)
  1506.     local currentSlot = turtle.getSelectedSlot()
  1507.     local success = false
  1508.     --[[
  1509.     minecraft:crafting_table
  1510.     minecraft:diamond_pickaxe
  1511.     minecraft:diamond_sword
  1512.     minecraft:diamond_shovel
  1513.     minecraft:diamond_hoe
  1514.     minecraft:diamond_axe
  1515.     wireless modem = ComputerCraft:CC-Peripheral
  1516.     ]]
  1517.     if slot > 0 then
  1518.         turtle.select(slot)
  1519.         if side == "right" then
  1520.             if turtle.equipRight() then
  1521.                 success = true
  1522.                 self.equippedRight = useItem
  1523.             end
  1524.         elseif side == "left" then
  1525.             if turtle.equipLeft() then
  1526.                 success = true
  1527.                 self.equippedLeft = useItem
  1528.             end
  1529.         end
  1530.     end
  1531.     turtle.select(currentSlot)
  1532.    
  1533.     return success
  1534. end
  1535.  
  1536. function T:unequip(side)
  1537.     local success = false
  1538.     local slot = self:getFirstEmptySlot()
  1539.     turtle.select(slot)
  1540.     if side == "right" then
  1541.         if turtle.equipRight() then
  1542.             success = true
  1543.             self.equippedRight = ""
  1544.         end
  1545.     else
  1546.         if turtle.equipLeft() then
  1547.             success = true
  1548.             self.equippedLeft = ""
  1549.         end
  1550.     end
  1551.    
  1552.     return success, slot
  1553. end
  1554.  
  1555. function T:fillVoid(direction, tblPreferredBlock, leaveExisting)
  1556.     assert(type(direction) == "string", "direction is not a string: "..tostring(direction))
  1557.     assert( tblPreferredBlock == nil or
  1558.             type(tblPreferredBlock) == "string" or
  1559.             type(tblPreferredBlock) == "table", "tblPreferredBlock is not nil, string or table: "..tostring(tblPreferredBlock))
  1560.     assert( leaveExisting == nil or type(leaveExisting) == "boolean", "leaveExisting is not boolean: "..tostring(leaveExisting))
  1561.    
  1562.     if tblPreferredBlock == nil or tblPreferredBlock == "" then tblPreferredBlock = {} end
  1563.     if type(tblPreferredBlock) ~= "table" then tblPreferredBlock = {tblPreferredBlock} end -- always use a table
  1564.     if leaveExisting == nil then leaveExisting = true end
  1565.    
  1566.    
  1567.    
  1568.     local Detect = turtle.detect
  1569.     local Place = turtle.place
  1570.     local Dig = turtle.dig
  1571.     if direction == "up" then
  1572.         Detect = turtle.detectUp
  1573.         Place = turtle.placeUp
  1574.         Dig = turtle.digUp
  1575.     elseif direction == "down" then
  1576.         Detect = turtle.detectDown
  1577.         Place = turtle.placeDown
  1578.         Dig = turtle.digDown
  1579.     end
  1580.    
  1581.    
  1582.    
  1583.     local lib = {}
  1584.    
  1585.     function lib.place(direction, placeBlock, currentBlock, slot)
  1586.         if placeBlock ~= currentBlock then -- current block does not match type to be used as filler
  1587.             turtle.select(slot)
  1588.             Dig()
  1589.             local attempts = 0
  1590.             while not Place() do
  1591.                 Dig()   -- added here in case of falling gravel etc.
  1592.                 attempts = attempts + 1
  1593.                 self:attack()
  1594.                 print("Attacking: "..attempts.." / 5")
  1595.                 sleep(g)
  1596.                 if attempts == 5 then
  1597.                     break
  1598.                 end
  1599.             end
  1600.         end
  1601.         turtle.select(1)
  1602.         return true
  1603.     end
  1604.    
  1605.     local placed = false
  1606.     local noBlocks = false
  1607.     local slot = 0
  1608.     local placeBlock = ""
  1609.     --check if vegetation and remove
  1610.     if self:isSeaweed(direction) then
  1611.         Dig()
  1612.     end
  1613.     if self:isGravityBlock(direction) then
  1614.         Dig()
  1615.     end
  1616.    
  1617.     local continue = false
  1618.     local currentBlock = self:getBlockType(direction)
  1619.     if currentBlock ~= "" then      -- solid block already present
  1620.         if  currentBlock:find("gravel") == nil and
  1621.             currentBlock:find("sand") == nil  and
  1622.             currentBlock:find("water") == nil and
  1623.             currentBlock:find("lava") == nil then -- not water, lava, sand or gravel
  1624.             if leaveExisting then
  1625.                 turtle.select(1)
  1626.                 return true, false
  1627.             end
  1628.         end
  1629.     end
  1630.    
  1631.     -- make a table of all existing blocks
  1632.     local stock = self:getCurrentInventory() -- eg stock[1] = minecraft:dirt, stock[2] = "", stock[3] = "minecraft:cobblestone"
  1633.    
  1634.     --[[ debugging
  1635.     for k,v in pairs(tblPreferredBlock) do
  1636.         print("k: "..k.." v: "..tostring(v))
  1637.     end
  1638.     print("Enter to continue")
  1639.     read()]]
  1640.    
  1641.     if next(tblPreferredBlock) ~= nil then -- check for preferredBlock
  1642.         local found = false
  1643.         for i = 1, 16 do
  1644.             for k,v in pairs(tblPreferredBlock) do
  1645.                 if stock[i]:find(v) ~= nil then -- eg stock[3] = "minecraft:cobblestone"
  1646.                     slot = i
  1647.                     placeBlock = stock[i]
  1648.                     found = true
  1649.                     break
  1650.                 end
  1651.             end
  1652.             if found then -- block found
  1653.                 break
  1654.             end
  1655.         end
  1656.     end
  1657.     -- print("using slot no "..slot) read()
  1658.  
  1659.  
  1660.     if slot == 0 then -- no preferred block or not found
  1661.         -- check for any stock of stone in order
  1662.         local found = false
  1663.         for i = 1, #stone do -- using 'stone' table (class variable)
  1664.             for j = 1, 16 do
  1665.                 if stock[j] == stone[i] then
  1666.                     slot = j --slot no
  1667.                     placeBlock = stock[j]
  1668.                     found = true
  1669.                     break
  1670.                 end
  1671.             end
  1672.             if found then
  1673.                 break
  1674.             end
  1675.         end
  1676.     end
  1677.    
  1678.     if slot == 0 then -- no suitable block found
  1679.     -- print("No blocks found") read()
  1680.         noBlocks = true
  1681.     else
  1682.     -- print("Placing: "..placeBlock) read()
  1683.         placed = lib.place(direction, placeBlock, currentBlock, slot)
  1684.     end
  1685.    
  1686.     turtle.select(1)
  1687.     return placed, noBlocks
  1688. end
  1689.  
  1690. function T:findBedrockTop(height)
  1691.     -- T.place(self, blockType, direction, leaveExisting
  1692.     -- T.go(self, path, useTorch, torchInterval, leaveExisting)
  1693.     local bedrockFound = false
  1694.     repeat
  1695.         bedrockFound = false
  1696.         self:clear()
  1697.         print("Checking surrounding  blocks...")
  1698.         for i = 1, 4 do
  1699.             self:turnLeft(1)
  1700.             local block = self:getBlockType("forward")
  1701.             if block:find("bedrock") then
  1702.                 bedrockFound = true
  1703.                 print("Bedrock found...")
  1704.             else
  1705.                 print("Found: "..block .. " in front")
  1706.             end
  1707.         end
  1708.         if bedrockFound then
  1709.             self:up(1)
  1710.             height = height -1
  1711.             self:place("stone", "down", true)
  1712.             print("Moving up...")
  1713.         end
  1714.     until not bedrockFound
  1715.     repeat
  1716.         bedrockFound = false
  1717.         local moved = 0
  1718.         for i = 1, 5 do
  1719.             if self:forward(1) then
  1720.                 moved = moved + 1
  1721.                 for i = 1, 4 do
  1722.                     self:turnLeft(1)
  1723.                     if self:getBlockType("forward"):find("bedrock") then
  1724.                         bedrockFound = true
  1725.                         print("Bedrock found: not continuing this row")
  1726.                     end
  1727.                 end
  1728.                 if bedrockFound then
  1729.                     break
  1730.                 end
  1731.             else -- hit bedrock
  1732.                 print("Hit bedrock in front")
  1733.                 bedrockFound = true
  1734.                 break
  1735.             end
  1736.         end
  1737.         self:turnLeft(2)
  1738.         for i = 1, moved do
  1739.             self:go("C2F1", false, 0, true)
  1740.         end
  1741.         self:go("L2C1", false, 0, true)
  1742.         if bedrockFound then
  1743.             print("Moving up...")
  1744.             self:go("U1C2", false, 0, true)
  1745.             height = height -1
  1746.         else
  1747.             print("Area clear of bedrock...")
  1748.         end
  1749.     until bedrockFound == false
  1750.     return height
  1751. end
  1752.  
  1753. function T:getBlockType(direction)
  1754.     --[[ turtle.inspect() returns two values
  1755.         1) boolean (true/false) success
  1756.         2) table with two or more values:
  1757.         .name (string) e.g. "minecraft:log"
  1758.         .metadata (integer) e.g. 0
  1759.         oak has metadata of 0, spruce 1, birch 2 etc
  1760.         CC Tweaked / later MC versions:
  1761.         No .metadata
  1762.         .state {axis = "y"}
  1763.         .tags (["minecraft:logs"] = true, ["minecraft:logs_that_burn"] = true, ["minecraft:oak_logs"] = true}
  1764.     ]]
  1765.     local blockType = ""
  1766.     local success = false
  1767.     local data = {} --initialise empty table variable
  1768.  
  1769.     if direction == "up" then
  1770.         success, data = turtle.inspectUp() -- store information about the block above in a table   
  1771.     elseif direction == "down" then
  1772.         success, data = turtle.inspectDown() -- store information about the block below in a table
  1773.     else
  1774.         success, data = turtle.inspect() -- store information about the block ahead in a table
  1775.     end
  1776.     if success then -- block found
  1777.         blockType = data.name -- eg "minecraft:log"
  1778.     end
  1779.    
  1780.     return blockType, data -- eg "minecraft:oak_log" , {table}
  1781. end
  1782.  
  1783. function T:getCoords(fromFile)
  1784.     fromFile = fromFile or false
  1785.     --get world coordinates from player
  1786.     local coord = 0
  1787.     local response = ""
  1788.     local continue = true
  1789.     local event = ""
  1790.     local param1 = ""
  1791.     local getInput = true
  1792.    
  1793.     self:clear()
  1794.     -- use built-in filesystem fs
  1795.     if fs.exists("homeCoords.txt") then --ask user if current coords are correct
  1796.         local fileHandle = fs.open("homeCoords.txt", "r")
  1797.         strText = fileHandle.readLine()
  1798.         self.x = tonumber(string.sub(strText, 3))
  1799.         strText = fileHandle.readLine()
  1800.         self.y = tonumber(string.sub(strText, 3))
  1801.         strText = fileHandle.readLine()
  1802.         self.z = tonumber(string.sub(strText, 3))
  1803.         strText = fileHandle.readLine()
  1804.         self:setFacing(tonumber(string.sub(strText, 3)))
  1805.         fileHandle.close()
  1806.         self:saveToLog("Coordinates loaded from file:", false)
  1807.         self:saveToLog("x = "..self.x..", y = "..self.y..", z = "..self.z..", f = "..self.facing, false)
  1808.         print("Coordinates loaded from file:\n")
  1809.         print("XYZ: - "..self.x.." / "..self.y.." / "..self.z.."\n")
  1810.         print("Facing: "..self.compass)
  1811.         print("\nUse F3 to check these coordinates")
  1812.         write("\nAre they correct (y/n + Enter)?")
  1813.         response = read()
  1814.         if response == "y" or response == "" then
  1815.             getInput = false
  1816.         else
  1817.             self:clear()
  1818.         end
  1819.     end
  1820.     if getInput then
  1821.         print("IMPORTANT! Stand directly behind turtle")
  1822.         print("Press F3 to read coordinates")
  1823.         print()
  1824.         continue = true
  1825.         while continue do
  1826.             print("Please enter your X coordinate")
  1827.             write("  x = ")
  1828.             coord = nil
  1829.             while coord == nil do
  1830.                 coord = tonumber(read())
  1831.                 if coord == nil then
  1832.                     self:clear()
  1833.                     print("Incorrect input. Use numbers only!")
  1834.                     print()
  1835.                     print("Please enter your X coordinate")
  1836.                     write("  x = ")
  1837.                 end
  1838.             end
  1839.             self.x = coord
  1840.             self:clear()
  1841.             print("Please enter your Y coordinate")
  1842.             write("  y = ")
  1843.             coord = nil
  1844.             while coord == nil do
  1845.                 coord = tonumber(read())
  1846.                 if coord == nil then
  1847.                     self:clear()
  1848.                     print("Incorrect input. Use numbers only")
  1849.                     print()
  1850.                     print("Please enter your y coordinate")
  1851.                     write("  y = ")
  1852.                 end
  1853.             end
  1854.             self.y = coord
  1855.             self:clear()
  1856.             print("Please enter your Z coordinate")
  1857.             write("  z = ")
  1858.             coord = nil
  1859.             while coord == nil do
  1860.                 coord = tonumber(read())
  1861.                 if coord == nil then
  1862.                     self:clear()
  1863.                     print("Incorrect input. Use numbers only")
  1864.                     print()
  1865.                     print("Please enter your z coordinate")
  1866.                     write("  z = ")
  1867.                 end
  1868.             end
  1869.             self.z = coord
  1870.             response = true
  1871.             while response do
  1872.                 response = false
  1873.                 self:clear()
  1874.                 print("Enter Direction you are facing:")
  1875.                 print("  0,1,2,3 (s,w,n,e)")
  1876.                 print()
  1877.                 print(  "  Direction = ")
  1878.                 event, param1 = os.pullEvent ("char")
  1879.                 if param1 == "s" or param1 == "S" then
  1880.                     coord = 0
  1881.                 elseif param1 == "w" or param1 == "W" then
  1882.                     coord = 1
  1883.                 elseif param1 == "n" or param1 == "N" then
  1884.                     coord = 2
  1885.                 elseif param1 == "e" or param1 == "E" then
  1886.                     coord = 3
  1887.                 elseif param1 == "0" or param1 == "1" or param1 == "2" or param1 == "3" then
  1888.                     coord = tonumber(param1)
  1889.                 else
  1890.                     print()
  1891.                     print("Incorrect input: "..param1)
  1892.                     print()
  1893.                     print("Use 0,1,2,3,n,s,w,e")
  1894.                     sleep(2)
  1895.                     response = true
  1896.                 end
  1897.             end
  1898.             self:setFacing(coord)
  1899.             self:clear()
  1900.             print("Your current location is:")
  1901.             print()
  1902.             print("  x = "..self.x)
  1903.             print("  y = "..self.y)
  1904.             print("  z = "..self.z)
  1905.             print("  facing "..self.compass.." ("..self.facing..")")
  1906.             print()
  1907.             write("Is this correct? (y/n)")
  1908.             event, param1 = os.pullEvent ("char")
  1909.             if param1 == "y" or param1 == "Y" then
  1910.                 continue = false
  1911.             end
  1912.         end
  1913.         -- correct coords to compensate for player standing position
  1914.         -- First tree is considered as point zero, on startup, turtle is in front of this tree
  1915.         -- Player is behind turtle, use 2 blocks to compensate
  1916.         -- facing:      Change:
  1917.         -- 0 (S)        z+1
  1918.         -- 1 (W)        x-1
  1919.         -- 2 (N)        z-1
  1920.         -- 3 (E)        x+1
  1921.         if self.facing == 0 then
  1922.             self.z = self.z + 2
  1923.         elseif self.facing == 1 then
  1924.             self.x = self.x - 2
  1925.         elseif self.facing == 2 then
  1926.             self.z = self.z - 2
  1927.         elseif self.facing == 3 then
  1928.             self.x = self.x + 2
  1929.         end
  1930.        
  1931.         -- create/overwrite 'homeCoords.txt'
  1932.         local fileHandle = fs.open("homeCoords.txt", "w")
  1933.         fileHandle.writeLine("x="..self.x)
  1934.         fileHandle.writeLine("y="..self.y)
  1935.         fileHandle.writeLine("z="..self.z)
  1936.         fileHandle.writeLine("f="..self.facing)
  1937.         fileHandle.close()
  1938.         self:saveToLog("homeCoords.txt file created", true)
  1939.         self:saveToLog("x = "..self:getX()..", y = "..self:getY()..", z = "..self:getZ()..", f = "..self:getFacing(), false)
  1940.     end
  1941. end
  1942.  
  1943. function T:getCurrentInventory()
  1944.     -- make a table of all existing blocks
  1945.     local stock = {"","","","","","","","","","","","","","","",""}
  1946.     for i = 1, 16 do
  1947.         if turtle.getItemCount(i) > 0 then
  1948.             data = turtle.getItemDetail(i) -- returns {count = x, name = 'minecraft:item" in 1.16.2)
  1949.             stock[i] = data.name
  1950.         end
  1951.     end
  1952.     return stock -- eg stock[1] = minecraft:dirt, stock[2] = "", stock[3] = "minecraft:cobblestone"
  1953. end
  1954.  
  1955. function T:getEquipped(side, keepInInventory)
  1956.     -- side = "right", "left", "minecraft:diamond_pickaxe"
  1957.     keepInInventory = keepInInventory or true
  1958.     local item = ""
  1959.     local slot = self:getFirstEmptySlot()
  1960.     turtle.select(slot)
  1961.     if side == "right" then
  1962.         if turtle.equipRight() then             -- take out whatever is on right side
  1963.             item = self:getSlotContains(slot)
  1964.             if item == "" then
  1965.                 slot = 0
  1966.             else
  1967.                 if not keepInInventory then
  1968.                     turtle.equipRight()         -- put it back
  1969.                 end
  1970.             end
  1971.         end
  1972.     elseif side == "left" then
  1973.         if turtle.equipLeft() then
  1974.             item = self:getSlotContains(slot)
  1975.             if item == "" then
  1976.                 slot = 0
  1977.             else
  1978.                 if not keepInInventory then
  1979.                     turtle.equipLeft()          -- put it back
  1980.                 end
  1981.             end
  1982.         end
  1983.     else    -- side will be item required eg axe
  1984.         if turtle.equipRight() then             -- take out whatever is on right side
  1985.             item = self:getSlotContains(slot)
  1986.             if item ~= side then                -- not the required item
  1987.                 turtle.equipRight()             -- put it back
  1988.                 if turtle.equipLeft() then
  1989.                     item = self:getSlotContains(slot)
  1990.                     if item ~= side then                -- not the required item
  1991.                         turtle.equipLeft()
  1992.                         item = ""
  1993.                         slot = 0
  1994.                     end
  1995.                 end
  1996.             end
  1997.         end
  1998.     end
  1999.    
  2000.     return item, slot   -- 0, "" if nothing equipped; 1, "minecraft:crafting_table"
  2001. end
  2002.  
  2003. function T:getFirstEmptySlot()
  2004.     local slot = 0
  2005.     local emptySlots = 0
  2006.     for i = 1,16 do
  2007.         if turtle.getItemCount(i) == 0 then
  2008.             if slot == 0 then
  2009.                 slot = i
  2010.             end
  2011.             emptySlots = emptySlots  + 1
  2012.         end
  2013.     end
  2014.     return slot, emptySlots
  2015. end
  2016.  
  2017. function T:getFirstItem(ofType)
  2018.     -- eg ofType = "seeds"
  2019.     ofType = ofType or ""
  2020.     local slot = 0
  2021.     local item, itemName, shortName = "", "", ""
  2022.     for i = 1, 16 do
  2023.         item = self:getItemName(i)      -- get full name of item
  2024.         if ofType == "" then            -- find any item
  2025.             itemName, shortName = self:getName(item)
  2026.             return item, itemName, shortName, i
  2027.         else
  2028.             if item:find(ofType) ~= nil then
  2029.                 itemName, shortName = self:getName(item)
  2030.                 return item, itemName, shortName, i
  2031.             end
  2032.         end
  2033.     end
  2034.     return item, itemName, shortName, slot
  2035. end
  2036.  
  2037. function T:getInput()
  2038.     --[[
  2039.         returns correctly interpreted value without errors caused by user
  2040.         usage: in main program - local input = T:getInput()
  2041.         local choice = input.getBoolean("Do you like Lua?")
  2042.         if choice then
  2043.             print("Yay!")
  2044.         end
  2045.     ]]
  2046.    
  2047.     local function tchelper(first, rest)
  2048.         return first:upper()..rest:lower()
  2049.     end
  2050.    
  2051.     local input = {}
  2052.    
  2053.     function input.getBoolean(prompt) -- assumes yes/no type entries from user
  2054.         while true do
  2055.             write(prompt.."_")
  2056.             userInput = read()
  2057.             --if string.len(userInput) == 0 then
  2058.             if userInput:len() == 0 then
  2059.                 print("\nJust pressing the Enter key doesn't work...")
  2060.             else       
  2061.                 --if string.sub(userInput, 1, 1):lower() == "y" then
  2062.                 if userInput:sub(1, 1):lower() == "y" then
  2063.                     userInput = true
  2064.                     break
  2065.                 --elseif string.sub(userInput, 1, 1):lower() == "n" then
  2066.                 elseif userInput:sub(1, 1):lower() == "n" then
  2067.                     userInput = false
  2068.                     break
  2069.                 else
  2070.                     print("\nOnly anything starting with y or n is accepted...")
  2071.                 end
  2072.             end
  2073.         end
  2074.         return userInput
  2075.     end
  2076.    
  2077.     function input.getString(prompt, withTitle, minInt, maxInt) -- withTitle, minInt and maxInt are given defaults if not passed
  2078.         withTitle = withTitle or false
  2079.         minInt = minInt or 1
  2080.         maxInt = maxInt or 20
  2081.         while true do
  2082.             write(prompt.."_")
  2083.             userInput = read()
  2084.             if userInput == "" then
  2085.                 print("\nJust pressing Enter doesn't work...")
  2086.             else       
  2087.                 --if string.len(userInput) >= minInt and string.len(userInput) <= maxInt then
  2088.                 if userInput:len() >= minInt and userInput:len() <= maxInt then
  2089.                     if withTitle then
  2090.                         userInput = input.toTitle(userInput)
  2091.                     end
  2092.                     break
  2093.                 else
  2094.                     print("\nTry entering text between "..minInt.." and "..maxInt.." characters...")
  2095.                 end
  2096.             end
  2097.         end
  2098.    
  2099.         return userInput
  2100.     end
  2101.    
  2102.     function input.getInteger(prompt, minInt, maxInt) -- minInt and maxInt are given defaults if not passed
  2103.         minInt = minInt or 0
  2104.         maxInt = maxInt or 65536
  2105.         while true do
  2106.             write(prompt.."_")
  2107.             userInput = read()
  2108.             if userInput == "" then
  2109.                 print("\nJust pressing the Enter key doesn't work...")
  2110.             else
  2111.                 if tonumber(userInput) ~= nil then
  2112.                     userInput = tonumber(userInput)
  2113.                     if userInput >= minInt and userInput <= maxInt then
  2114.                         break
  2115.                     else
  2116.                         print("\nTry a number from "..minInt.." to "..maxInt.."...")
  2117.                     end
  2118.                 else
  2119.                     print("\nTry entering a number - "..userInput.." does not cut it...")
  2120.                 end
  2121.             end
  2122.         end
  2123.         return userInput
  2124.     end
  2125.    
  2126.     function input.toTitle(inputText) --converts any string to Title Case
  2127.         return inputText:gsub("(%a)([%w_']*)", tchelper)
  2128.     end
  2129.    
  2130.     return input
  2131. end
  2132.  
  2133. function T:getInventory()
  2134.     local temp = {}
  2135.     local blockCount = 0
  2136.     local blockTypeCount = 0
  2137.     local mostItem = ""
  2138.    
  2139.     for slot = 1, 16 do
  2140.         local slotContains, slotCount = T:getSlotContains(slot)
  2141.         if slotContains ~= "" then
  2142.             if temp[slotContains] == nil then
  2143.                 temp[slotContains] = slotCount
  2144.                 blockTypeCount = blockTypeCount + 1
  2145.             else
  2146.                 temp[slotContains] = temp[slotContains] + slotCount
  2147.             end
  2148.             blockCount = blockCount + slotCount
  2149.         end
  2150.     end
  2151.     local names = {}
  2152.     local blockCounts = {}
  2153.     local mostCount = 0
  2154.     local mostItem = ""
  2155.     for block, count in pairs (temp) do
  2156.         table.insert(names, block)
  2157.         table.insert(blockCounts, count)
  2158.         if count > mostCount then
  2159.             mostCount = count
  2160.             mostItem = block
  2161.         end
  2162.     end
  2163.     local inventory = temp
  2164.     inventory.names = names
  2165.     inventory.quantities = blockCounts
  2166.     inventory.blockTypeCount = blockTypeCount
  2167.     inventory.blockCount = blockCount
  2168.     inventory.useBlock = mostItem
  2169.     inventory.mostBlock = mostItem
  2170.     inventory.mostCount = mostCount
  2171.  
  2172.     return inventory
  2173.     --[[
  2174.         {
  2175.         inventory["minecraft:cobblestone"] = 128
  2176.         inventory["minecraft:stone"] = 64
  2177.         inventory.names = {minecraft:cobblestone, minecraft:stone}
  2178.         inventory.quantities = {128, 64}
  2179.         inventory.blockTypeCount = 2,
  2180.         inventory.blockCount = 196,
  2181.         inventory.useBlock = "minecraft:cobblestone"
  2182.         inventory.mostBlock = "minecraft:cobblestone"
  2183.         inventory.mostCount = 128
  2184.         }
  2185.     ]]
  2186. end
  2187.  
  2188. function T:updateInventory()
  2189.     local quit = false
  2190.     repeat
  2191.         local event, param1 = os.pullEvent()
  2192.         if event == "turtle_inventory" then -- user has added, moved or removed inventory items
  2193.             return self:getInventory()
  2194.         elseif event == "key" then
  2195.             if param1 == keys.enter then
  2196.                 quit = true
  2197.             end
  2198.         end
  2199.     until quit
  2200.    
  2201.     return nil
  2202. end
  2203.  
  2204. function T:getInventoryItems()
  2205.     --[[ table eg: {"minecraft:cobblestone" = 256, "minecraft:cobbled_deepslate = 256"}  ]]
  2206.     local inventory = {}
  2207.     for slot = 1, 16 do
  2208.         local slotContains, slotCount = self:getSlotContains(slot)
  2209.         if slotContains ~= "" then -- eg "minecraft:cobblestone"
  2210.             if inventory[slotContains] ~= nil then --already exists in inventory
  2211.                 inventory[slotContains] = inventory[slotContains] + slotCount --update quantities
  2212.             else
  2213.                 inventory[slotContains] = slotCount
  2214.             end
  2215.         end
  2216.     end
  2217.     return inventory
  2218. end
  2219.  
  2220. function T:getItemCount(item)
  2221.     local slot, total, slotData = self:getItemSlot(item) --return .leastSlot, .leastModifier, total, slotData
  2222.     return total
  2223. end
  2224.  
  2225. function T:getItemName(slot)
  2226.     local data = {} --initialise empty table variable
  2227.     data.name = ""
  2228.     if slot > 0 and slot < 17 then
  2229.         if turtle.getItemCount(slot) > 0 then
  2230.             data = turtle.getItemDetail(slot)
  2231.         end
  2232.     end
  2233.    
  2234.     return data.name
  2235. end
  2236.  
  2237. function T:getItemSlot(item)
  2238.     -- return slot no with least count, and total count available
  2239.     -- along with a table of mostSlot, mostCount, leastSlot, leastCount
  2240.     -- if minecraft:log or log2, names for 1.16.2 changed to 'minecraft:oak_log' etc so use wildcards 'log'
  2241.     item = item or "stone"
  2242.    
  2243.     local slotData = {} -- setup return table
  2244.     slotData.firstSlot = 0
  2245.     slotData.lastSlot = 0
  2246.     slotData.mostSlot = 0
  2247.     slotData.mostName = ""
  2248.     slotData.mostCount = 0
  2249.     slotData.leastSlot = 0
  2250.     slotData.leastName = ""
  2251.     slotData.leastCount = 0
  2252.    
  2253.     local lib = {}
  2254.    
  2255.     function lib.update(i, v)
  2256.         local count = turtle.getItemCount(i)
  2257.         if slotData.firstSlot == 0 then     -- setup .firstSlot
  2258.             slotData.firstSlot = i
  2259.         end
  2260.         if count > slotData.mostCount then  -- setup .mostCount
  2261.             slotData.mostSlot = i
  2262.             slotData.mostName = v
  2263.             slotData.mostCount = count
  2264.         end
  2265.         if count < slotData.leastCount then
  2266.             slotData.leastSlot = i
  2267.             slotData.leastName = v
  2268.             slotData.leastCount = count
  2269.         end
  2270.         slotData.lastSlot = i               -- setup / edit .lastSlot
  2271.        
  2272.         return count
  2273.     end
  2274.    
  2275.     local total = 0
  2276.     -- make a table of all existing blocks
  2277.     local stock = self:getCurrentInventory()        -- returns full names of all inventory items in a list 1-16
  2278.     --self:saveToLog("currentInventory = "..textutils.serialise(stock))
  2279.     if item:find("\:") ~= nil then                  -- find exact match only
  2280.         for i,v in ipairs(stock) do                 -- iterate current inventory
  2281.             if v == item then                       -- item found
  2282.                 total = total + lib.update(i, v)
  2283.             end
  2284.         end
  2285.     elseif item:find("common") ~= nil or item == "any" or item == "stone" then -- find any stone, in prefence order from stone table
  2286.         local stoneFound = false
  2287.         for j = 1, #stone do -- using 'stone' table (class variable)
  2288.             for i = 1, 16 do
  2289.                 if stock[i] == stone[j] then
  2290.                     stoneFound = true -- found match in list priority
  2291.                     item = stone[j]
  2292.                     break
  2293.                 end
  2294.             end
  2295.             if stoneFound then
  2296.                 break
  2297.             end
  2298.         end
  2299.         if stoneFound then
  2300.             for i,v in ipairs(stock) do             -- iterate current inventory
  2301.                 if v:find(item) ~= nil then         -- item found
  2302.                     total = total + lib.update(i, v)
  2303.                 end
  2304.             end
  2305.         end
  2306.     else -- find matching name
  2307.         for i,v in ipairs(stock) do             -- iterate current inventory
  2308.             if v:find(item) ~= nil then         -- item found
  2309.                 total = total + lib.update(i, v)
  2310.             end
  2311.         end
  2312.     end
  2313.  
  2314.     if slotData.mostSlot > 0 then
  2315.         if slotData.leastSlot == 0 then
  2316.             slotData.leastSlot = slotData.mostSlot
  2317.             slotData.leastName = slotData.mostName
  2318.             slotData.leastCount = slotData.mostCount
  2319.         end
  2320.     end
  2321.     return slotData.lastSlot, total, slotData -- integer, integer, table
  2322. end
  2323.  
  2324. function T:getMostItem(excludeItem, stoneOnly)
  2325.     --[[ Used to get user choice of stone based on quantity ]]
  2326.     local lib = {}
  2327.    
  2328.     function lib.checkStone(item)
  2329.         for k,v in ipairs (stone) do
  2330.             if item == v then
  2331.                 return true
  2332.             end
  2333.         end
  2334.         return false
  2335.     end
  2336.    
  2337.     excludeItem = excludeItem or ""
  2338.     stoneOnly = stoneOnly or false
  2339.     local data = {}
  2340.     local inventory = {}
  2341.     local mostItem = ""
  2342.     local leastItem = ""
  2343.     local mostCount = 0
  2344.     local leastCount = 0
  2345.     local itemTypeCount = 0
  2346.     local itemCount = 0
  2347.     for i = 1, 16 do
  2348.         if turtle.getItemCount(i) > 0 then
  2349.             data = turtle.getItemDetail(i)
  2350.             if inventory[data.name] == nil then
  2351.                 inventory[data.name] = data.count
  2352.                 itemTypeCount = itemTypeCount + 1
  2353.             else
  2354.                 inventory[data.name] = inventory[data.name] + data.count
  2355.             end
  2356.             itemCount = itemCount + data.count
  2357.         end
  2358.     end
  2359.     for k,v in pairs(inventory) do
  2360.         if mostItem == "" then                  -- not yet found mostItem
  2361.             if stoneOnly then                   -- only check for stone blocks
  2362.                 if lib.checkStone(k) then       -- stone found
  2363.                     if excludeItem == "" then   -- no excluded items
  2364.                         mostItem = k
  2365.                         mostCount = v
  2366.                         leastItem = k
  2367.                         leastCount = v
  2368.                     else
  2369.                         if k:find(excludeItem) == nil then -- not found
  2370.                             mostItem = k
  2371.                             mostCount = v
  2372.                             leastItem = k
  2373.                             leastCount = v
  2374.                         end
  2375.                     end
  2376.                 end
  2377.             else                                -- not just stone in count
  2378.                 if excludeItem == "" then       -- no excluded items
  2379.                     mostItem = k
  2380.                     mostCount = v
  2381.                     leastItem = k
  2382.                     leastCount = v
  2383.                 else
  2384.                     if k:find(excludeItem) == nil then -- not found
  2385.                         mostItem = k
  2386.                         mostCount = v
  2387.                         leastItem = k
  2388.                         leastCount = v
  2389.                     end
  2390.                 end
  2391.             end
  2392.         else                                    -- mostItem found
  2393.             if stoneOnly then                   -- only check for stone blocks
  2394.                 if lib.checkStone(k) then
  2395.                     if excludeItem == "" then   -- no excluded items
  2396.                         if inventory[k] > mostCount then
  2397.                             mostItem = k
  2398.                             mostCount = v
  2399.                         end
  2400.                         if inventory[k] < leastCount then
  2401.                             leastItem = k
  2402.                             leastCount = v
  2403.                         end
  2404.                     else
  2405.                         if k:find(excludeItem) == nil then -- not found
  2406.                             if inventory[k] > mostCount then
  2407.                                 mostItem = k
  2408.                                 mostCount = v
  2409.                             end
  2410.                             if inventory[k] < leastCount then
  2411.                                 leastItem = k
  2412.                                 leastCount = v
  2413.                             end
  2414.                         end
  2415.                     end
  2416.                 end
  2417.             else
  2418.                 if excludeItem == "" then           -- no excluded items
  2419.                     if inventory[k] > mostCount then
  2420.                         mostItem = k
  2421.                         mostCount = v
  2422.                     end
  2423.                     if inventory[k] < leastCount then
  2424.                         leastItem = k
  2425.                         leastCount = v
  2426.                     end
  2427.                 else
  2428.                     if k:find(excludeItem) == nil then -- not found
  2429.                         if inventory[k] > mostCount then
  2430.                             mostItem = k
  2431.                             mostCount = v
  2432.                         end
  2433.                         if inventory[k] < leastCount then
  2434.                             leastItem = k
  2435.                             leastCount = v
  2436.                         end
  2437.                     end
  2438.                 end
  2439.             end
  2440.         end
  2441.     end
  2442.    
  2443.     --print("mostItem: "..mostItem.." mostCount: "..mostCount) read()
  2444.     return mostItem, mostCount, leastItem, leastCount, inventory
  2445. end
  2446.  
  2447. function T:getName(item)
  2448.     -- eg minecraft:beetroot_seeds -> beetroot_seeds or beetroot
  2449.     local marker, name, shortName = 0, "", ""
  2450.     marker = item:find("\:")            -- eg minecraft:beetroot_seeds -> 10
  2451.     if marker ~= nil then
  2452.         name = item:sub(marker + 1)     -- eg minecraft:beetroot_seeds -> beetroot_seeds
  2453.     else
  2454.         name = item
  2455.     end
  2456.     marker = name:find("_")             -- eg beetroot_seeds -> 9
  2457.     if marker ~= nil then
  2458.         shortName = name:sub(1, marker) -- eg beetroot_seeds -> beetroot
  2459.     else
  2460.         shortName = name
  2461.     end
  2462.     return name, shortName
  2463. end
  2464.  
  2465. function T:getNames(item)
  2466.     -- item MUST be full name
  2467.     -- eg minecraft:beetroot_seeds
  2468.     local parts, tblParts, mod, rest
  2469.     if item:find("\:") == nil then
  2470.         if item:find("_") == nil then
  2471.             return {item}   -- eg "seeds"
  2472.         else
  2473.             mod = ""
  2474.             rest = item
  2475.         end
  2476.     else
  2477.         parts = item:split(":")
  2478.         mod = parts[1]
  2479.         rest = parts[2]
  2480.     end
  2481.     tblParts = {}
  2482.     table.insert(tblParts, mod)
  2483.     parts = rest:split("_")
  2484.     for _, v in ipairs(parts) do
  2485.         table.insert(tblParts, v)
  2486.     end
  2487.  
  2488.     return tblParts -- eg {"minecraft","crafting", "table"}
  2489. end
  2490.  
  2491. function T:getPolishedItem(blockType)
  2492.     --[[
  2493.         local blockType, count = T:getPolishedItem("")
  2494.         local blockType, count = T:getPolishedItem("slab")
  2495.         local blockType, count = T:getPolishedItem("stairs")
  2496.         local blockType, count = T:getPolishedItem("wall")
  2497.     ]]
  2498.     local data = {}
  2499.     local inventory = {}
  2500.    
  2501.     for i = 1, 16 do
  2502.         if turtle.getItemCount(i) > 0 then
  2503.             data = turtle.getItemDetail(i)
  2504.             if inventory[data.name] == nil then
  2505.                 inventory[data.name] = data.count
  2506.             else
  2507.                 inventory[data.name] = inventory[data.name] + data.count
  2508.             end
  2509.         end
  2510.     end
  2511.     for k,v in pairs(inventory) do
  2512.         if k:find("polished") ~= nil then -- contains polished stairs, wall, slab,
  2513.             if blockType == "slab" or blockType == "stairs" or blockType == "wall" then
  2514.                 if K:find(blockType) ~= nil then
  2515.                     return k, v
  2516.                 end
  2517.             else -- looking for polished_andesite, granite etc block
  2518.                 if k:find("slab") == nil and k:find("stairs") == nil and k:find("wall") == nil then
  2519.                     return k, v
  2520.                 end
  2521.             end
  2522.         end
  2523.     end
  2524.    
  2525.     return "", 0
  2526. end
  2527.  
  2528. function T:getSaplingSlot(name)
  2529.     local saplingSlot = 0
  2530.     local count = 0
  2531.     local data = {name = ""} -- in case no saplings found
  2532.     if name == nil then
  2533.         name = "sapling"
  2534.     end
  2535.     for i = 1, 16 do
  2536.         count = turtle.getItemCount(i)
  2537.         if count > 0 then
  2538.             data = turtle.getItemDetail(i) -- returns {count = x, name = 'minecraft:item" in 1.16.2)
  2539.             if data.name:find(name) ~= nil then
  2540.                 saplingSlot = i
  2541.                 break
  2542.             end
  2543.         end
  2544.     end
  2545.     return saplingSlot, data.name, count -- 0, "" / "minecraft:oak_sapling", 6
  2546. end
  2547.  
  2548. function T:getPlaceChestDirection()
  2549.     local facing = self.facing
  2550.     local chestDirection = "forward"
  2551.     local turns = 0
  2552.    
  2553.     for i = 1, 4 do
  2554.         if turtle.detect() then
  2555.             self:turnRight(1)
  2556.             turns = turns + 1
  2557.         else
  2558.             break
  2559.         end
  2560.     end
  2561.     if turns == 4 then -- full circle
  2562.         turns = 0
  2563.         if turtle.detectDown() then -- no space below
  2564.             if turtle.detectUp() then
  2565.                 if self:dig("up") then
  2566.                     chestDirection = "up"
  2567.                 end
  2568.             else
  2569.                 chestDirection = "up"
  2570.             end
  2571.         else
  2572.             chestDirection = "down"
  2573.         end
  2574.     end
  2575.     return chestDirection, turns -- will be "forward" or "up", 0 - 3
  2576. end
  2577.  
  2578. function T:getSlotContains(slotNo)
  2579.     local data = {} --initialise empty table variable
  2580.    
  2581.     local slotCount = 0
  2582.     local slotContains = ""
  2583.     if turtle.getItemCount(slotNo) > 0 then
  2584.         data = turtle.getItemDetail(slotNo)
  2585.         slotCount = data.count
  2586.         slotContains = data.name
  2587.     end
  2588.    
  2589.     return slotContains, slotCount
  2590. end
  2591.  
  2592. function T:getStock(item)
  2593.     -- return total units and slot numbers of max and min amounts
  2594.     local slot, total, slotData = self:getItemSlot(item) --return .leastSlot, .leastModifier, total, slotData
  2595.     local rt = {}
  2596.     rt.total = total
  2597.     rt.mostSlot = slotData.mostSlot
  2598.     rt.leastSlot = slotData.leastSlot
  2599.     rt.mostCount = slotData.mostCount
  2600.     rt.leastCount = slotData.leastCount
  2601.     --[[
  2602.     if slot == 0 then
  2603.         if modifier == nil then
  2604.             T.saveToLog(self, "getStock()"..tostring(item).."= not found", true)
  2605.         else
  2606.             T.saveToLog(self, "getStock()"..tostring(item).."("..tostring(modifier)..")= not found")
  2607.         end
  2608.     end
  2609.     ]]
  2610.    
  2611.     return rt --{rt.total, rt.mostSlot, rt.leastSlot, rt.mostCount, rt.leastCount}
  2612. end
  2613.  
  2614. function T:getSolidBlockCount()
  2615.     local retValue = 0
  2616.     --local solids = {'cobble', 'stone', 'dirt', 'granite', 'andesite', 'diorite', 'deepslate', 'glass', 'tuff'}
  2617.     local slotCount, slotContains
  2618.     for i = 1, 16 do
  2619.         slotContains, slotCount = self:getSlotContains(i)
  2620.         for _, v in ipairs(stone) do
  2621.             if slotContains:find(v) ~= nil then
  2622.                 retValue = retValue + slotCount
  2623.                 break
  2624.             end
  2625.         end
  2626.     end
  2627.     return retValue
  2628. end
  2629.  
  2630. function T:getTotalItemCount()
  2631.     local count = 0
  2632.    
  2633.     for i = 1, 16 do
  2634.         count = count + turtle.getItemCount(i)
  2635.     end
  2636.     return count
  2637. end
  2638.  
  2639. function T:getWater(direction)
  2640.     direction = direction or "forward"
  2641.     -- assign place methods according to direction
  2642.     local Place = turtle.place
  2643.     if direction == "up" then
  2644.         Place = turtle.placeUp
  2645.     elseif direction == "down" then
  2646.         Place = turtle.placeDown
  2647.     end
  2648.     local slot = self:getItemSlot("minecraft:bucket")
  2649.     if slot > 0 then
  2650.         turtle.select(slot)
  2651.         if Place() then
  2652.             turtle.select(1)
  2653.             return true
  2654.         end
  2655.     end
  2656.     turtle.select(1)
  2657.     return false
  2658. end
  2659.  
  2660. function T:go(path, useTorch, torchInterval, leaveExisting, preferredBlock)
  2661.     useTorch = useTorch or false -- used in m and M to place torches in mines
  2662.     if leaveExisting == nil then
  2663.         leaveExisting = false
  2664.     end
  2665.     torchInterval = torchInterval or 8
  2666.     if preferredBlock == nil or preferredBlock == "" then
  2667.         preferredBlock = {} -- used for C command to allow faster placing of specific block
  2668.     else
  2669.         if type(preferredBlock) ~= "table" then
  2670.             preferredBlock = {preferredBlock}
  2671.         end
  2672.     end
  2673.    
  2674.     local intervalList =
  2675.     {
  2676.         1,
  2677.         torchInterval * 1 + 1,
  2678.         torchInterval * 2 + 1,
  2679.         torchInterval * 3 + 1,
  2680.         torchInterval * 4 + 1,
  2681.         torchInterval * 5 + 1,
  2682.         torchInterval * 6 + 1
  2683.     }
  2684.     local slot = turtle.getSelectedSlot()
  2685.     turtle.select(1)       
  2686.     local commandList = {}
  2687.     local command = ""
  2688.     local direction = {"up", "forward", "down"}
  2689.     -- remove spaces from path
  2690.     path = string.gsub(path, " ", "")
  2691.     -- make a list of commands from path string eg "x0F12U1" = x0, F12, U1
  2692.     for i = 1, string.len(path) do
  2693.         local character = string.sub(path, i, i) -- examine each character in the string
  2694.         if tonumber(character) == nil then -- char is NOT a number
  2695.             if command ~= "" then -- command is NOT empty eg "x0"
  2696.                 table.insert(commandList, command) -- add to table eg "x0"
  2697.             end
  2698.             command = character -- replace command with new character eg "F"
  2699.         else -- char IS a number
  2700.             command = command..character -- add eg 1 to F = F1, 2 to F1 = F12
  2701.             if i == string.len(path) then -- last character in the string
  2702.                 table.insert(commandList, command)
  2703.             end
  2704.         end
  2705.     end
  2706.     -- R# L# F# B# U# D# +0 -0 d0 = Right, Left, Forward, Back, Up, Down, up while detect and return, down while not detect, down and place while not detect
  2707.     -- dig:           x0,x1,x2 (up/fwd/down)
  2708.     -- suck:          s0,s1,s2
  2709.     -- place chest:   H0,H1,H2
  2710.     -- place sapling: S0,S1,S2
  2711.     -- place Torch:   T0,T1,T2
  2712.     -- place Hopper:  P0,P1,P2
  2713.     -- mine floor:    m# = mine # blocks above and below, checking for valuable items below, and filling space with cobble or dirt
  2714.     -- mine ceiling:  M# = mine # blocks, checking for valuable items above, and filling space with cobble or dirt
  2715.     -- mine ceiling:  N# same as M but not mining block below unless valuable
  2716.     -- mine floor:    n# mine block below and/or fill void + check left side
  2717.     -- mine ceiling:  Q# same as M + mine block below if valuable + left side
  2718.     -- mine wall:     q# mine # blocks forward, check left side and patch
  2719.     -- tunnel top:    A# fill voids above and both sides
  2720.     -- tunnel bottom: E# fill voids both sides, remove floor
  2721.     -- tunnel bottom: X# fill voids both sides and floor
  2722.     -- QuickMine      V# take block above if valuable, fill void above, fill void below
  2723.     -- QuickCoridoor  W# take block below, take block above if valuable, fill void above
  2724.     -- mine - bedrock Z#
  2725.     -- 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
  2726.     -- place:         t = 0/1 = place behind, 2 = cobble first, up torch, forward down
  2727.     -- place:         e = ladder, direction
  2728.     -- place:         @ = any block in inventory, direction
  2729.     -- X              mine block below and/or fill void, then check sides (X = trench)
  2730.     -- +              while detectUp digUp and move up. Return to original position after
  2731.     -- ^              place stairs
  2732.     self:refuel(15)
  2733.     turtle.select(1)
  2734.     for cmd in self:values(commandList) do -- eg F12 or x1
  2735.         local move = string.sub(cmd, 1, 1)
  2736.         local modifier = tonumber(string.sub(cmd, 2))
  2737.         if move == "A" then --mine block above and/or fill void + fill both sides
  2738.             turtle.select(1)
  2739.             --[[check block behind
  2740.             T.turnRight(self, 2)
  2741.             T.digValuable(self, "forward")
  2742.             T.fillVoid(self, "forward")
  2743.             T.turnRight(self, 2)]]
  2744.             for i = 1, modifier + 1 do --eg A16 run loop 17 x
  2745.                 self:digValuable("up")
  2746.                 self:fillVoid("up")
  2747.                 self:turnLeft(1)
  2748.                 self:digValuable("forward")
  2749.                 self:fillVoid("forward")
  2750.                 self:turnRight(2)
  2751.                 self:digValuable("forward")
  2752.                 self:fillVoid("forward")
  2753.                 self:turnLeft(1)
  2754.                 --T.dig(self, "down") -- create player coridoor
  2755.                 if i <= modifier then -- A16 = move forward 16x
  2756.                     self:forward(1)
  2757.                 end
  2758.             end
  2759.             --check block at end
  2760.             self:digValuable("forward")
  2761.             self:fillVoid("forward")
  2762.         elseif move == "B" then
  2763.             self:back(modifier)
  2764.         elseif move == "c" then
  2765.             if turtle.detectDown() then
  2766.                 --check if vegetation and remove
  2767.                 if self:isSeaweed("down") then
  2768.                     turtle.digDown()
  2769.                 end
  2770.             end
  2771.             if not turtle.detectDown() then
  2772.                 if not self:place("minecraft:cobblestone", "down") then
  2773.                     self:place("minecraft:dirt", "down")
  2774.                 end
  2775.             end
  2776.         elseif move == "C" then
  2777.             -- fillVoid(self, direction, tblPreferredBlock, leaveExisting)
  2778.             local fill = false
  2779.             if leaveExisting then -- leave alone if non-gravity
  2780.                 if self:detect(direction[modifier + 1]) then -- solid block ahead, not air, water or lava
  2781.                     if self:digValuable(direction[modifier + 1]) then
  2782.                         fill = true
  2783.                     elseif self:digGravityBlock(direction[modifier + 1]) then -- sand or gravel
  2784.                         fill = true
  2785.                     end
  2786.                 else    -- air, water or lava ahead
  2787.                     fill = true
  2788.                 end
  2789.             else
  2790.                 fill = true
  2791.             end
  2792.             if fill then
  2793.                 self:fillVoid(direction[modifier + 1], preferredBlock, false)
  2794.             end
  2795.         elseif move == "d" then -- down and place while not detect
  2796.             if modifier == 1 then
  2797.                 self:fillVoid("forward", preferredBlock)
  2798.             end
  2799.             while not turtle.detectDown() do
  2800.                 self:down(1)
  2801.                 if modifier == 1 then
  2802.                     self:fillVoid("forward", preferredBlock)
  2803.                 end
  2804.             end
  2805.             if modifier == 1 then
  2806.                 self:fillVoid("forward", preferredBlock)
  2807.             end
  2808.         elseif move == "D" then
  2809.             self:down(modifier)
  2810.         elseif move == "e" then -- ladder above / in front / below
  2811.             self:place("minecraft:ladder", direction[modifier + 1], false)
  2812.         elseif move == "E" then --mine block below + fill both sides
  2813.             turtle.select(1)
  2814.             --check block behind
  2815.             self:turnRight(2)
  2816.             self:digValuable("forward")
  2817.             self:fillVoid("forward")
  2818.             self:turnRight(2)
  2819.             for i = 1, modifier + 1 do --eg A16 run loop 17 x
  2820.                 self:turnLeft(1)
  2821.                 self:digValuable("forward")
  2822.                 self:fillVoid("forward")
  2823.                 self:turnRight(2)
  2824.                 self:digValuable("forward")
  2825.                 self:fillVoid("forward")
  2826.                 self:turnLeft(1)
  2827.                 self:dig("down") -- create player coridoor
  2828.                 if i <= modifier then -- A16 = move forward 16x
  2829.                     self:forward(1)
  2830.                 end
  2831.             end
  2832.             --check block at end
  2833.             self:digValuable("forward")
  2834.             self:fillVoid("forward")
  2835.         elseif move == "F" then
  2836.             self:forward(modifier)
  2837.         elseif move == "H" then
  2838.             self:place("minecraft:chest", direction[modifier + 1], leaveExisting)
  2839.         elseif move == "L" then
  2840.             self:turnLeft(modifier)
  2841.         elseif move == "m" then --mine block below and/or fill void
  2842.             --T:go("m8", false, 0, false, brick)
  2843.             for i = 1, modifier + 1 do --eg m8 run loop 9 x
  2844.                 turtle.select(1)
  2845.                 local isValuable, blockType = self:isValuable("down")
  2846.                 if isValuable or blockType == "minecraft:gravel" or not leaveExisting then
  2847.                     turtle.digDown() -- dig if gravel
  2848.                 elseif blockType == "minecraft:lava" then
  2849.                     self:place("minecraft:bucket", "down")
  2850.                 end
  2851.                 self:dig("up") -- create player coridoor
  2852.                 if not turtle.detectDown() then
  2853.                     self:fillVoid("down", preferredBlock, leaveExisting)
  2854.                 end
  2855.                 if i <= modifier then -- n8 = move forward 8x. check for initial use in mine to avoid torch in wrong place
  2856.                     if useTorch then
  2857.                         if  (i == intervalList[1] and modifier >= 16) or
  2858.                             i == intervalList[2] or
  2859.                             i == intervalList[3] or
  2860.                             i == intervalList[4] or
  2861.                             i == intervalList[5] then
  2862.                             self:up(1)
  2863.                             self:place("minecraft:torch", "down", false)
  2864.                             self:forward(1)
  2865.                             self:down(1)
  2866.                         else
  2867.                             self:forward(1)
  2868.                         end
  2869.                     else
  2870.                         self:forward(1)
  2871.                     end
  2872.                 end
  2873.             end
  2874.         elseif move == "M" then --mine block above and/or fill void
  2875.             for i = 1, modifier + 1 do
  2876.                 turtle.select(1)
  2877.                 local isValuable, blockType = self:isValuable("up")
  2878.                 if isValuable then
  2879.                     self:dig("up")
  2880.                 elseif blockType == "minecraft:lava" then
  2881.                     self:place("minecraft:bucket", "up")
  2882.                 end
  2883.                 if not turtle.detectUp()then
  2884.                     self:fillVoid("up", preferredBlock)
  2885.                 end
  2886.                 if i <= modifier then -- will not move forward if modifier = 0
  2887.                     if useTorch then
  2888.                         if  i == intervalList[1] or
  2889.                             i == intervalList[2] or
  2890.                             i == intervalList[3] or
  2891.                             i == intervalList[4] or
  2892.                             i == intervalList[5] then
  2893.                             self:place("minecraft:torch", "down", false)
  2894.                         end
  2895.                     end
  2896.                     self:forward(1)
  2897.                 end
  2898.             end
  2899.         elseif move == "n" then --mine block below and/or fill void + check left side
  2900.             for i = 1, modifier + 1 do --eg m8 run loop 9 x
  2901.                 turtle.select(1)
  2902.                 local isValuable, blockType = self:isValuable("down")
  2903.                 if isValuable or blockType == "minecraft:gravel" or not leaveExisting then
  2904.                     turtle.digDown() -- dig if valuable or gravel
  2905.                 else --check for lava
  2906.                     if blockType == "minecraft:lava" then
  2907.                         self:place("minecraft:bucket", "down")
  2908.                     end
  2909.                 end
  2910.                 self:dig("up") -- create player coridoor
  2911.                 if not turtle.detectDown() then
  2912.                     self:fillVoid("down", preferredBlock)
  2913.                 end
  2914.                 self:turnLeft(1)
  2915.                 local isValuable, blockType = self:isValuable("forward")
  2916.                 if isValuable or not leaveExisting then
  2917.                     turtle.dig() -- dig if valuable
  2918.                 end
  2919.                 if not turtle.detect() then
  2920.                     self:fillVoid("forward", preferredBlock, leaveExisting)
  2921.                 end
  2922.                 self:turnRight(1)  
  2923.                 if i <= modifier then -- m8 = move forward 8x
  2924.                     if useTorch then
  2925.                         if  i == intervalList[1] or
  2926.                             i == intervalList[2] or
  2927.                             i == intervalList[3] or
  2928.                             i == intervalList[4] or
  2929.                             i == intervalList[5] then
  2930.                             self:up(1)
  2931.                             self:place("minecraft:torch", "down", false)
  2932.                             self:forward(1)
  2933.                             self:down(1)
  2934.                         else
  2935.                             self:forward(1)
  2936.                         end
  2937.                     else
  2938.                         self:forward(1)
  2939.                     end
  2940.                 end
  2941.             end
  2942.         elseif move == "N" then --mine block above and/or fill void + mine block below if valuable
  2943.             for i = 1, modifier + 1 do
  2944.                 turtle.select(1)
  2945.                 local isValuable, blockType = self:isValuable("up")
  2946.                 if isValuable then
  2947.                     self:dig("up")
  2948.                 elseif blockType == "minecraft:lava" then
  2949.                     self:place("minecraft:bucket", "up")
  2950.                 end
  2951.                 if not turtle.detectUp() then
  2952.                     self:fillVoid("up", preferredBlock)
  2953.                 end
  2954.                 turtle.select(1)
  2955.                 isValuable, blockType = self:isValuable("down")
  2956.                 if isValuable then
  2957.                     self:dig("down")
  2958.                     self:fillVoid("down", preferredBlock)
  2959.                 else
  2960.                     if self:getBlockType("down") == "minecraft:lava" then
  2961.                         self:place("minecraft:bucket", "down")
  2962.                     end
  2963.                 end
  2964.                 if i <= modifier then
  2965.                     self:forward(1)
  2966.                 end
  2967.             end
  2968.         elseif move == "P" then
  2969.             self:place("minecraft:hopper", direction[modifier + 1], leaveExisting)
  2970.         elseif move == "q" then --mine block and fill voids on left side left side
  2971.             for i = 1, modifier + 1 do
  2972.                 turtle.select(1)
  2973.                 self:turnLeft(1)
  2974.                 if self:isValuable("forward") then
  2975.                     turtle.dig() -- dig if valuable
  2976.                 end
  2977.                 if not turtle.detect() then
  2978.                     self:fillVoid("forward", preferredBlock)
  2979.                 end
  2980.                 self:turnRight(1)  
  2981.                 if i <= modifier then
  2982.                     self:forward(1)
  2983.                 end
  2984.             end
  2985.         elseif move == "Q" then --mine block above and/or fill void + mine block below if valuable + left side
  2986.             for i = 1, modifier + 1 do
  2987.                 turtle.select(1)
  2988.                 if self:isValuable("up") then
  2989.                     self:dig("up")
  2990.                 else --check for lava
  2991.                     if self:getBlockType("up") == "minecraft:lava" then
  2992.                         self:place("minecraft:bucket", "up")
  2993.                     end
  2994.                 end
  2995.                 if not turtle.detectUp() then
  2996.                     self:fillVoid("up", preferredBlock)
  2997.                 end
  2998.                 self:turnLeft(1)
  2999.                 if self:isValuable("forward") then
  3000.                     turtle.dig() -- dig if valuable
  3001.                 end
  3002.                 if not turtle.detect() then
  3003.                     self:fillVoid("forward", preferredBlock)
  3004.                 end
  3005.                 self:turnRight(1)  
  3006.                 if self:isValuable("down") then
  3007.                     turtle.digDown()
  3008.                 end
  3009.                 if i <= modifier then
  3010.                     self:forward(1)
  3011.                 end
  3012.             end
  3013.         elseif move == "r" then
  3014.             self:place("minecraft:dirt", direction[modifier + 1], leaveExisting)
  3015.         elseif move == "R" then
  3016.             self:turnRight(modifier)
  3017.         elseif move == "s" then
  3018.             if modifier == 0 then
  3019.                 while turtle.suckUp() do end
  3020.             elseif modifier == 1 then
  3021.                 while turtle.suck() do end
  3022.             elseif modifier == 2 then
  3023.                 while turtle.suckDown() do end
  3024.             end
  3025.         elseif move == "S" then
  3026.             self:place("sapling", direction[modifier + 1], leaveExisting)
  3027.         elseif move == "t" then
  3028.             -- 0 = placeUp does not work with os 1.8
  3029.             -- 1 = turn round, placeForward
  3030.             -- 2 = placeDown
  3031.             -- 3 = turnLeft, placeUp
  3032.             -- 4 = turnround, placeUp
  3033.             -- 5 = place down without block
  3034.             if modifier == 0 then -- os < 1.8
  3035.                 self:place("minecraft:torch", "up", false)
  3036.             elseif modifier == 1 then --place behind
  3037.                 self:turnLeft(2)
  3038.                 self:place("minecraft:torch", "forward", false)
  3039.                 self:turnLeft(2)
  3040.             elseif modifier == 2 then -- place below for 2
  3041.                 if not self:place("minecraft:cobblestone", "down") then
  3042.                     self:place("minecraft:dirt", "down")
  3043.                 end
  3044.                 self:up(1)
  3045.                 self:place("minecraft:torch", "down", false)
  3046.                 self:forward(1)
  3047.                 self:down(1)
  3048.             elseif modifier == 3 then --turnLeft, placeUp (on ground to wall)
  3049.                 self:turnLeft(1)
  3050.                 self:place("minecraft:torch", "up", false)
  3051.                 self:turnRight(1)
  3052.             elseif modifier == 4 then --turnLeft, placeUp (on ground to wall)
  3053.                 self:turnLeft(2)
  3054.                 self:place("minecraft:torch", "up", false)
  3055.                 self:turnLeft(2)
  3056.             elseif modifier == 5 then --cobble first, then torch
  3057.                 self:place("minecraft:torch", "down", false)
  3058.             end
  3059.         elseif move == "T" then
  3060.             self:place("minecraft:torch", direction[modifier + 1], leaveExisting)
  3061.         elseif move == "u" then -- move up and place forward/down
  3062.             repeat
  3063.                 if modifier == 1 then
  3064.                     self:fillVoid("forward", preferredBlock, leaveExisting)
  3065.                 end
  3066.                 self:up(1)
  3067.                 if modifier == 1 then
  3068.                     self:fillVoid("forward", preferredBlock, leaveExisting)
  3069.                 end
  3070.                 self:fillVoid("down", preferredBlock, leaveExisting)
  3071.             until not turtle.inspectUp()
  3072.             if modifier == 1 then
  3073.                 self:fillVoid("forward", preferredBlock, leaveExisting)
  3074.             end
  3075.         elseif move == "U" then
  3076.             self:up(modifier)
  3077.         elseif move == "x" then
  3078.             if modifier == 0 then
  3079.                 self:dig("up")
  3080.             elseif modifier == 1 then
  3081.                 self:dig("forward")
  3082.             elseif modifier == 2 then
  3083.                 while turtle.detectDown() do
  3084.                     turtle.digDown()
  3085.                 end
  3086.             end
  3087.         elseif move == "V" then --QuickMine
  3088.             for i = 1, modifier + 1 do
  3089.                 turtle.select(1)
  3090.                 local isValuable, blockType = self:isValuable("down")
  3091.                 if isValuable then
  3092.                     self:dig("down")
  3093.                 elseif blockType == "minecraft:lava" then
  3094.                     self:place("minecraft:bucket", "down")
  3095.                 end
  3096.                 isValuable, blockType = self:isValuable("up")
  3097.                 if isValuable then
  3098.                     self:dig("up")
  3099.                 elseif blockType == "minecraft:lava" then
  3100.                     self:place("minecraft:bucket", "up")
  3101.                 end
  3102.                 if not turtle.detectUp()then
  3103.                     self:fillVoid("up", preferredBlock)
  3104.                 end
  3105.                 if not turtle.detectDown()then
  3106.                     self:fillVoid("down", preferredBlock)
  3107.                 end
  3108.                 if i <= modifier then -- will not move forward if modifier = 0
  3109.                     self:forward(1)
  3110.                 end
  3111.             end
  3112.         elseif move == "W" then --QuickCoridoor
  3113.             for i = 1, modifier + 1 do
  3114.                 turtle.select(1)
  3115.                 local isValuable, blockType = self:isValuable("up")
  3116.                 if isValuable then
  3117.                     self:dig("up")
  3118.                 elseif blockType == "minecraft:lava" then
  3119.                     self:place("minecraft:bucket", "up")
  3120.                 end
  3121.                 if not turtle.detectUp()then
  3122.                     self:fillVoid("up", preferredBlock)
  3123.                 end
  3124.                 self:isWaterOrLava("down") -- automatically removes lava
  3125.                 self:dig("down")
  3126.                 if i <= modifier then -- will not move forward if modifier = 0
  3127.                     self:forward(1)
  3128.                 end
  3129.             end
  3130.         elseif move == "X" then --mine block below and/or fill void, then check sides (X = trench)
  3131.             for i = 1, modifier + 1 do --eg X8 run loop 9 x
  3132.                 turtle.select(1)
  3133.                 local isValuable, blockType = self:isValuable("down")
  3134.                 if isValuable or blockType == "minecraft:gravel" or blockType == "minecraft:sand" or blockType:find("grass") ~= nil then
  3135.                     turtle.digDown() -- dig if gravel
  3136.                 else --check for lava
  3137.                     if blockType == "minecraft:lava" then
  3138.                         self:place("minecraft:bucket", "down")
  3139.                     end
  3140.                 end
  3141.                 self:dig("up") -- create player coridoor
  3142.                 if not turtle.detectDown() then
  3143.                     self:fillVoid("down")
  3144.                 end
  3145.                 self:turnRight(1)
  3146.                 if not turtle.detect() then
  3147.                     self:fillVoid("forward")
  3148.                 end
  3149.                 self:turnLeft(2)
  3150.                 if not turtle.detect() then
  3151.                     self:fillVoid("forward")
  3152.                 end
  3153.                 self:turnRight(1)
  3154.                 if i <= modifier then -- X8 = move forward 8x
  3155.                     self:forward(1)
  3156.                 end
  3157.             end
  3158.         elseif move == "Z" then -- mine to bedrock
  3159.             for i = 1, modifier + 1 do 
  3160.                 turtle.select(1)
  3161.                 local goUp = 0
  3162.                 while self:down(1) do
  3163.                     goUp = goUp + 1
  3164.                 end
  3165.                 for j = goUp, 1, -1 do
  3166.                     for k = 1, 4 do
  3167.                         self:turnRight(1)
  3168.                         if self:isValuable("forward") then
  3169.                             self:place("minecraft:cobblestone", "forward")
  3170.                         end
  3171.                     end
  3172.                     self:up(1)
  3173.                     self:place("minecraft:cobblestone", "down")
  3174.                     turtle.select(1)
  3175.                 end
  3176.                 if i <= modifier then
  3177.                     self:forward(2)
  3178.                 end
  3179.             end
  3180.         elseif move == "+" then
  3181.             local height = 0
  3182.             while turtle.detectUp() do
  3183.                 self:up(1)
  3184.                 height = height + 1
  3185.             end
  3186.             self:down(height)
  3187.         elseif move == "-" then
  3188.             while not turtle.inspectDown() do
  3189.                 self:down(1)
  3190.             end
  3191.         elseif move == "*" then
  3192.             local goUp = 0
  3193.             while not turtle.inspectDown() do
  3194.                 self:down(1)
  3195.                 goUp = goUp + 1
  3196.             end
  3197.             if goUp > 0 then
  3198.                 for i = 1, goUp do
  3199.                     self:up(1)
  3200.                     if not self:place("minecraft:cobblestone", "down") then
  3201.                         self:place("minecraft:dirt", "down")
  3202.                     end
  3203.                 end
  3204.                 goUp = 0
  3205.             else
  3206.                 turtle.digDown()
  3207.                 if not self:place("minecraft:cobblestone", "down") then
  3208.                     self:place("minecraft:dirt", "down")
  3209.                 end
  3210.             end
  3211.         elseif move == "^" then --place stair
  3212.             if not self:place("stairs", direction[modifier + 1], false) then -- ending false forces block replacement
  3213.                 print("could not place stairs "..direction[modifier + 1])
  3214.                 self:place("minecraft:cobblestone", direction[modifier + 1], false)
  3215.             end
  3216.         elseif move == "@" then -- any item in inventory
  3217.             self:place("", direction[modifier + 1], leaveExisting)
  3218.         end
  3219.     end
  3220.     turtle.select(slot)
  3221. end
  3222.  
  3223. function T:harvestTree(extend, craftChest, direction)
  3224.     extend = extend or false
  3225.     craftChest = craftChest or false
  3226.     direction = direction or "forward"
  3227.     local goHeight = 0
  3228.     local onLeft = true     -- default position in double tree
  3229.     if direction == "forward" then
  3230.         turtle.dig()        -- dig base of tree
  3231.         self:forward(1) -- go under tree with 1 log. Will refuel if needed
  3232.     end
  3233.     -- check if on r or l of double width tree
  3234.     self:turnLeft(1)
  3235.     local blockType = self:getBlockType("forward")
  3236.     if blockType:find("log") ~= nil then
  3237.         extend = true
  3238.         onLeft = false              -- placed on right side of 2 block tree
  3239.     end
  3240.     self:turnRight(2)   -- check if double tree
  3241.     blockType = self:getBlockType("forward")
  3242.     if blockType:find("log") ~= nil then
  3243.         extend = true
  3244.         onLeft = true               -- placed on left side of 2 block tree
  3245.     end
  3246.     self:turnLeft(1)    -- return to correct position
  3247.     if craftChest then
  3248.         self:dig("up")
  3249.         self:up(1)
  3250.         self:dig("up")
  3251.         while not self:detect("down") do
  3252.             self:down(1)
  3253.         end
  3254.         self:craft("planks", 8)
  3255.         self:craft("chest", 1)
  3256.         while self:detect("up") do
  3257.             self:up(1)
  3258.             goHeight = goHeight + 1
  3259.         end
  3260.     end
  3261.     -- Loop to climb up tree and harvest trunk and surrounding leaves
  3262.     while self:dig("up") do -- continue loop while block detected above
  3263.         self:up(1)  -- Move up
  3264.         goHeight = goHeight + 1
  3265.         -- Inner loop to check for leaves/ break double tree logs
  3266.         for i = 1, 4 do
  3267.             blockType = self:getBlockType("forward")
  3268.             if blockType:find("log") ~= nil or blockType:find("leaves") ~= nil then
  3269.                 self:dig("forward") --Dig leaves / logs in double tree. Leave bee nests
  3270.             end
  3271.             self:turnRight(1)
  3272.         end
  3273.     end
  3274.     -- At top of the tree. New loop to return to ground
  3275.     if extend then
  3276.         if onLeft then
  3277.             self:go("F1R1F1R2")
  3278.         else
  3279.             self:go("F1L1F1R2")
  3280.         end
  3281.         while turtle.detectUp() do
  3282.             self:up(1)
  3283.             goHeight = goHeight + 1
  3284.         end
  3285.     end
  3286.     for i = 1, goHeight do
  3287.         if extend then
  3288.             for j = 1, 4 do
  3289.                 self:dig("forward")
  3290.                 self:turnRight(1)
  3291.             end
  3292.         end
  3293.         self:down(1)
  3294.     end
  3295.     -- check for logs below in case felling started above ground
  3296.     while self:getBlockType("down"):find("log") ~= nil do
  3297.         self:down(1)
  3298.     end
  3299.     if extend then
  3300.         if onLeft then
  3301.             self:go("F1L1F1R2")
  3302.         else
  3303.             self:go("F1R1F1R2")
  3304.         end
  3305.     end
  3306.     return extend   -- true if double tree
  3307. end
  3308.  
  3309. function T:harvestWholeTree(direction) 
  3310.     --RECURSIVE FUNCTION - BEWARE!
  3311.     local blockType, height
  3312.  
  3313.     if direction == "up" then
  3314.         self:refuel(15)
  3315.         if self:isLog("up") then
  3316.             self:up(1)
  3317.             if self:isLog("up") then
  3318.                 self:harvestWholeTree("up")
  3319.             end
  3320.         end
  3321.         self:down(1)
  3322.         for i = 1, 4 do
  3323.             -- check all round
  3324.             if self:isLog("forward") then
  3325.                 self:harvestWholeTree("forward")
  3326.             else
  3327.                 blockType = self:getBlockType("forward")
  3328.                 if blockType ~= "" then
  3329.                     if blockType:find("leaves") ~= nil then
  3330.                         self:forward(1)
  3331.                         self:harvestWholeTree("forward")
  3332.                         self:back(1)
  3333.                     end
  3334.                 end
  3335.             end
  3336.             self:turnRight(1)
  3337.         end
  3338.     elseif direction == "forward" then
  3339.         if self:isLog("forward") then
  3340.             self:refuel(15)
  3341.            
  3342.             self:forward(1)
  3343.             if turtle.detectUp() then
  3344.                 turtle.digUp()
  3345.             end
  3346.             if self:isLog("forward") then
  3347.                 self:harvestWholeTree("forward")
  3348.             end
  3349.             --check left side
  3350.             self:turnLeft(1)
  3351.             if self:isLog("forward") then
  3352.                 self:harvestWholeTree("forward")
  3353.             end
  3354.             -- check right side
  3355.             self:turnRight(2)
  3356.             if self:isLog("forward") then
  3357.                 self:harvestWholeTree("forward")
  3358.             end
  3359.             self:turnLeft(1)
  3360.             self:back(1)
  3361.         end
  3362.     end
  3363. end
  3364.  
  3365. function T:isEmpty()
  3366.     local isEmpty = true
  3367.     for i = 1, 16 do
  3368.         if turtle.getItemCount(i) > 0 then
  3369.             isEmpty = false
  3370.             break
  3371.         end
  3372.     end
  3373.     return isEmpty
  3374. end
  3375.  
  3376. function T:isLog(direction)
  3377.     local success = false
  3378.     local blockType, modifier
  3379.    
  3380.     local Detect = turtle.detect
  3381.     if direction == 'up' then
  3382.         Detect = turtle.detectUp
  3383.     elseif direction == 'down' then
  3384.         Detect = turtle.detectDown
  3385.     end
  3386.    
  3387.     if Detect() then
  3388.         blockType = self:getBlockType(direction)
  3389.         if blockType:find('log') ~= nil then
  3390.             success = true
  3391.         end
  3392.     end
  3393.    
  3394.     return success
  3395. end
  3396.  
  3397. function T:isValuable(direction)
  3398.     local success = false
  3399.     local ignoreList = "minecraft:dirt,minecraft:grass,minecraft:stone,minecraft:gravel,minecraft:chest,"..
  3400.                      "minecraft:cobblestone,minecraft:sand,minecraft:torch,minecraft:bedrock,minecraft:ladder"..
  3401.                      "minecraft:netherrack,minecraft:blackstone,minecraft:basalt"..
  3402.                      "minecraft:granite,minecraft:diorite,minecraft:andesite"..
  3403.                      "minecraft:deepslate,minecraft:cobbled_deepslate,minecraft:tuff,minecraft:obsidian,minecraft:end_stone"
  3404.  
  3405.                      
  3406.     local Detect = turtle.detect
  3407.    
  3408.     if direction == "up" then
  3409.         Detect = turtle.detectUp
  3410.     elseif direction == "down" then
  3411.         Detect = turtle.detectDown
  3412.     end
  3413.    
  3414.     local blockType = self:getBlockType(direction)
  3415.    
  3416.     if blockType ~= "" then --block found
  3417.         success = true
  3418.         if ignoreList:find(blockType) ~= nil then
  3419.             success = false
  3420.         end
  3421.         if blockType:find("lava") ~= nil or blockType:find("water") ~= nil then
  3422.             success = false
  3423.         end
  3424.     end
  3425.     if success then
  3426.         -- check if diamond. or netherite if so ensure space in inventory
  3427.         if blockType:find("diamond") ~= nil or blockType:find("debris") ~= nil then
  3428.             self:dumpRefuse(direction)
  3429.         end
  3430.     end
  3431.     return success, blockType
  3432. end
  3433.  
  3434. function T:isSeaweed(direction)
  3435.     --[[ look for seaweed in specified direction ]]
  3436.     local Detect = turtle.detect
  3437.     local blockName
  3438.     if direction == "up" then
  3439.         Detect = turtle.detectUp
  3440.     elseif direction == "down" then
  3441.         Detect = turtle.detectDown
  3442.     end
  3443.     if Detect() then
  3444.         blockName = self:getBlockType(direction)
  3445.     end
  3446.     if self:isVegetation(blockName) then
  3447.         return true
  3448.     end
  3449.     return false
  3450. end
  3451.  
  3452. function T:isGravityBlock(direction)
  3453.     --[[ look for sand, gravel, concrete powder ]]
  3454.     local blockName = self:getBlockType(direction)
  3455.     if blockName:find("sand") ~= nil or blockName:find("gravel") ~= nil then
  3456.         return true
  3457.     end
  3458.     return false
  3459. end
  3460.  
  3461. function T:isEmpty()
  3462.     for i = 1, 16 do
  3463.         if turtle.getItemCount(i) > 0 then
  3464.             return false
  3465.         end
  3466.     end
  3467.     return true
  3468. end
  3469.  
  3470. function T:isVegetation(blockName)
  3471.     blockName = blockName or ""
  3472.     for _, v in pairs(flowers) do
  3473.         if blockName == v then
  3474.             return true
  3475.         end
  3476.     end
  3477.    
  3478.     return false
  3479. end
  3480.  
  3481. function T:isWater(direction)
  3482.     direction = direction or "forward"
  3483.     local isWater = false
  3484.     local isSource = false
  3485.     local isIce = false
  3486.     local level = nil
  3487.     local isAir = false
  3488.    
  3489.     local blockType, data = self:getBlockType(direction)       
  3490.     if blockType == "" then
  3491.         return false, false, false, nil, true
  3492.     end
  3493.     if blockType:find("water") ~= nil or blockType == "minecraft:bubble_column" or blockType:find("ice") ~= nil then -- allows for bubble_column
  3494.         isWater = true
  3495.         if blockType == "minecraft:bubble_column" then
  3496.             isSource = true
  3497.         end
  3498.         if blockType:find("ice") ~= nil then
  3499.             isSource = true
  3500.             isIce = true
  3501.         end
  3502.         level = data.state.level
  3503.         if level ~= nil then
  3504.             if level == 0 then
  3505.                 isSource = true
  3506.             end
  3507.         end
  3508.     end
  3509.    
  3510.     -- isWater  = source, ice, flowing water or bubble column
  3511.     -- isSource = source or ice
  3512.     -- isIce    = ice
  3513.     -- level    = nil or flowing water value
  3514.     -- air      = NOT water, source, ice
  3515.     return isWater, isSource, isIce, level, isAir --true, true, false, 0, false = source true, false, false, 1 if next to source
  3516. end
  3517.  
  3518. function T:isWaterOrLava(direction)
  3519.     direction = direction or "forward"
  3520.     local blockType = ""
  3521.     if not self:detect(direction) then --air, water or lava
  3522.         blockType = self:getBlockType(direction)
  3523.         if blockType:find("lava") ~= nil then
  3524.             self:place("minecraft:bucket", direction, false) -- auto refuel
  3525.         end
  3526.     end
  3527.     return blockType  -- "" or 'minecraft:xxxx"
  3528. end
  3529.  
  3530. function T:menu(title, list)
  3531.     local retValue = 0
  3532.     response = true
  3533.     while response do
  3534.         response = false
  3535.         self:clear()
  3536.         print(title.."\n")
  3537.         for i = 1, #list, 1 do
  3538.             print("\t"..i..".  "..list[i])
  3539.         end
  3540.         print("Type number of your choice (1 to "..#list..")_")
  3541.         event, param1 = os.pullEvent ("char")
  3542.         local choice = tonumber(param1)
  3543.         if choice ~= nil then -- number typed
  3544.             if choice >= 1 or choice <= #list then
  3545.                 retValue = choice
  3546.             end
  3547.         else
  3548.             print()
  3549.             print("Incorrect input: "..param1)
  3550.             print()
  3551.             print("Type numbers only, from 1 to "..#list)
  3552.             sleep(2)
  3553.             response = true
  3554.         end
  3555.     end
  3556.     return retValue -- 1 to no of items in the list
  3557. end
  3558.  
  3559. function T:place(blockType, direction, leaveExisting, signText, doNotAttack)
  3560.     -- originally place(blockType, damage, direction, leaveExisting, signText): "log", -1, "forward", true, "Hello"
  3561.     if blockType == "" then --use any
  3562.         blockType = "stone"
  3563.     end
  3564.     if tonumber(direction) ~= nil then
  3565.         -- legacy call -> shift parameters "log", -1(direction), "forward"(leaveExisting), true(signText), "Hello" -> "log", "forward", true, "Hello"
  3566.         leaveExisting = signText
  3567.         direction = leaveExisting
  3568.     else
  3569.         -- correct params
  3570.         direction = direction or "forward"
  3571.     end
  3572.    
  3573.     if leaveExisting == nil then
  3574.         leaveExisting = true
  3575.     end
  3576.     signText = signText or ""
  3577.     doNotAttack = doNotAttack or false
  3578.    
  3579.     local success = false
  3580.     local doContinue = true
  3581.     local dig = true
  3582.     -- assign place methods according to direction
  3583.     local Place = turtle.place
  3584.     if direction == "up" then
  3585.         Place = turtle.placeUp
  3586.     elseif direction == "down" then
  3587.         Place = turtle.placeDown
  3588.     end
  3589.     local slot = self:getItemSlot(blockType)
  3590.     if blockType == "minecraft:bucket" then -- empty bucket for lava or water
  3591.         if slot > 0 then
  3592.             turtle.select(slot)
  3593.             if Place() then -- lava or water collected
  3594.                 if self:getSlotContains(slot) == "minecraft:lava_bucket" then
  3595.                     self:refuel(0)
  3596.                 end
  3597.                 turtle.select(1)
  3598.                 return true, slot
  3599.             end
  3600.         end
  3601.     else
  3602.         local existingBlock = self:getBlockType(direction)
  3603.         if leaveExisting then -- do not remove existing block unless sand gravel water or lava
  3604.             if self:detect(direction) then -- not water or lava
  3605.                 if existingBlock:find("sand") ~= nil or existingBlock:find("gravel") ~= nil then --leave anything except sand/gravel   
  3606.                     doContinue = true
  3607.                 end
  3608.             else
  3609.                 doContinue = true
  3610.             end
  3611.         end
  3612.         if existingBlock:find("cobble") ~= nil and blockType:find("cobble") ~= nil then -- already cobble or cobbled deepslate
  3613.             doContinue = false-- no need to replace 1 cobble with another
  3614.         end
  3615.         if doContinue then -- air / water or lava in next block or leaveExisting = false
  3616.             self:dig(direction)
  3617.             if slot > 0 then -- item to place found
  3618.                 turtle.select(slot)
  3619.                 if doNotAttack then
  3620.                      Place(signText)
  3621.                 else
  3622.                     local attempts = 0
  3623.                     while not Place(signText) do
  3624.                         if self:attack() then
  3625.                             print("Attacking "..blockType.." ? chest or minecart below")
  3626.                             sleep(1)
  3627.                             --T.saveToLog("Error placing "..blockType.." ? chest or minecart below")
  3628.                         end
  3629.                         attempts = attempts + 1
  3630.                         if attempts > 1 then
  3631.                             turtle.select(1)
  3632.                             return false, slot
  3633.                         end
  3634.                     end
  3635.                 end
  3636.                 success = true
  3637.             end
  3638.         end
  3639.     end
  3640.     turtle.select(1)
  3641.     return success, slot
  3642. end
  3643.  
  3644. function T:placeWater(direction)
  3645.     direction = direction or "forward"
  3646.     -- assign place methods according to direction
  3647.     local Place = turtle.place
  3648.     if direction == "up" then
  3649.         Place = turtle.placeUp
  3650.     elseif direction == "down" then
  3651.         Place = turtle.placeDown
  3652.     end
  3653.     local slot = self:getItemSlot("minecraft:water_bucket")
  3654.     if slot > 0 then
  3655.         turtle.select(slot)
  3656.         if Place() then
  3657.             turtle.select(1)
  3658.             return true
  3659.         end
  3660.     end
  3661.    
  3662.     turtle.select(1)
  3663.     return false
  3664. end
  3665.  
  3666. function T:refuel(minLevel, toLimitOnly)   
  3667.     minLevel = minLevel or 15
  3668.     if toLimitOnly == nil then
  3669.         toLimitOnly = false
  3670.     end
  3671.     if minLevel == 0 then
  3672.         toLimitOnly = true
  3673.     end
  3674.     local itemSlot = 0
  3675.     local slot = turtle.getSelectedSlot()
  3676.     local count = 0
  3677.     local item = ""
  3678.     local refuelOK = false
  3679.    
  3680.     if turtle.getFuelLevel() >= minLevel and minLevel > 0 then -- prevent auto refuel eg fell tree
  3681.         return false
  3682.     end
  3683.  
  3684.     if turtle.getFuelLevel() >= turtle.getFuelLimit() and toLimitOnly then
  3685.         return false
  3686.     else
  3687.         if turtle.getFuelLevel() < minLevel or minLevel == 0 then
  3688.             -- check each slot for fuel item
  3689.             for i = 1, 16 do
  3690.                 item, count = self:getSlotContains(i)
  3691.                 if item == "minecraft:lava_bucket" then
  3692.                     turtle.select(i)
  3693.                     if turtle.refuel() then
  3694.                         print("refuelled with lava: "..turtle.getFuelLevel().." / "..turtle.getFuelLimit())
  3695.                         refuelOK = true
  3696.                         break
  3697.                     else
  3698.                         print("lava block deleted: "..turtle.getFuelLevel().." / "..turtle.getFuelLimit())
  3699.                         refuelOK = true
  3700.                         break
  3701.                     end
  3702.                 end
  3703.                 if item == "minecraft:coal" then
  3704.                     turtle.select(i)
  3705.                     if turtle.refuel(1) then
  3706.                         while turtle.getFuelLevel() < minLevel and turtle.getItemCount(i) > 0 do
  3707.                             turtle.refuel(1)
  3708.                         end
  3709.                         print("refuelled with coal: "..turtle.getFuelLevel())
  3710.                         refuelOK = true
  3711.                     end
  3712.                 end
  3713.             end
  3714.             if not refuelOK then
  3715.                 for i = 1, 16 do
  3716.                     item, count = self:getSlotContains(i)
  3717.                     if item:find("planks") ~= nil then
  3718.                         turtle.select(i)
  3719.                         if turtle.refuel() then
  3720.                             print("refuelled with planks: "..turtle.getFuelLevel())
  3721.                             refuelOK = true
  3722.                         end
  3723.                     end
  3724.                 end
  3725.             end
  3726.             if not refuelOK then
  3727.                 local success = false
  3728.                 for i = 1, 16 do
  3729.                     item, count = self:getSlotContains(i)
  3730.                     if string.find(item, "log") ~= nil then --logs onboard
  3731.                         print("Refuelling with log slot "..tostring(i)..", crafting planks")
  3732.                         if self:craft("planks", 4) then
  3733.                             success = true
  3734.                         else
  3735.                             print("refuel() error crafting planks")
  3736.                         end
  3737.                         if success then
  3738.                             local planksSlot, count = self:getItemSlot("planks")
  3739.                             turtle.select(planksSlot)
  3740.                             if turtle.refuel() then
  3741.                                 refuelOK = true
  3742.                             end
  3743.                         end
  3744.                     end
  3745.                 end
  3746.             end
  3747.             if not refuelOK and turtle.getFuelLevel() == 0 then
  3748.                 term.clear()
  3749.                 term.setCursorPos(1,1)
  3750.                 print("Unable to refuel: "..turtle.getFuelLevel().." fuel remaining")
  3751.                 --checkInventoryForItem(self, items, quantities, required, message)
  3752.                 local result = self:checkInventoryForItem({"minecraft:lava_bucket", "coal", "planks"}, {1, 10, 32}, false) -- false allows user to press enter
  3753.                 if result == nil then
  3754.                     return false
  3755.                 end
  3756.             end
  3757.         end
  3758.         turtle.select(slot)
  3759.        
  3760.         return refuelOK
  3761.     end
  3762. end
  3763.  
  3764. function T:selectPlaceItem(item)
  3765.     self.placeSlot = self:getItemSlot(item)
  3766.     if self.placeSlot > 0 then
  3767.         self.placeItem = item
  3768.         return true
  3769.     end
  3770.     return false
  3771. end
  3772.  
  3773. function T:setEquipment()
  3774.     -- if contains a crafting table, puts it on the right. Any other tool on the left
  3775.     self:clear()
  3776.     print("Setting up equipment...")
  3777.     local emptySlotR = self:getFirstEmptySlot() -- first empty slot
  3778.     if emptySlotR == 0 then -- all slots full
  3779.         turtle.select(16)
  3780.         turtle.drop()
  3781.         emptySlotR = 16
  3782.     end
  3783.     local emptySlotL = 0 -- used later
  3784.     local eqRight = ""
  3785.     local eqLeft = ""
  3786.     local equippedRight = ""
  3787.     local equippedLeft = ""
  3788.     local count = 0
  3789.     local pickaxeSlot, total = self:getItemSlot("minecraft:diamond_pickaxe")
  3790.     local swordSlot, total = self:getItemSlot("minecraft:diamond_sword")
  3791.     local hoeSlot, total = self:getItemSlot("minecraft:diamond_hoe")
  3792.     local craftTableSlot, total = self:getItemSlot("minecraft:crafting_table")
  3793.     if emptySlotR > 0 then -- empty slot found
  3794.         turtle.select(emptySlotR)
  3795.         if turtle.equipRight() then -- remove tool on the right
  3796.             eqRight, count = self:getSlotContains(emptySlotR) -- eqRight contains name of tool from right side
  3797.             if eqRight == "minecraft:crafting_table" then
  3798.                 craftTableSlot = emptySlotR
  3799.                 eqRight = ""
  3800.             elseif eqRight == "minecraft:diamond_pickaxe" then
  3801.                 pickaxeSlot = emptySlotR
  3802.                 eqRight = ""
  3803.             elseif eqRight == "minecraft:diamond_sword" then
  3804.                 swordSlot = emptySlotR
  3805.                 eqRight = ""
  3806.             elseif eqRight == "minecraft:diamond_hoe" then
  3807.                 hoeSlot = emptySlotR
  3808.                 eqRight = ""
  3809.             end -- eqRight
  3810.             emptySlotL = self:getFirstEmptySlot() -- get next empty slot
  3811.             if emptySlotL == 0 then -- all slots full
  3812.                 if emptySlotR ~= 15 then
  3813.                     turtle.select(15)
  3814.                     turtle.drop()
  3815.                     emptySlotL = 15
  3816.                 else
  3817.                     turtle.select(16)
  3818.                     turtle.drop()
  3819.                     emptySlotL = 16
  3820.                 end
  3821.             end
  3822.         else -- nothing equipped on right side
  3823.             emptySlotL = emptySlotR
  3824.         end
  3825.         if emptySlotL > 0 then -- empty slot found
  3826.             turtle.select(emptySlotL)
  3827.             if turtle.equipLeft() then -- remove tool on the left
  3828.                 eqLeft, count = self:getSlotContains(emptySlotL) -- eqLeft contains name of tool from left side
  3829.                 if eqLeft == "minecraft:diamond_pickaxe" then
  3830.                     pickaxeSlot = emptySlotL
  3831.                     eqLeft = ""
  3832.                 elseif eqLeft == "minecraft:diamond_sword" then
  3833.                     swordSlot = emptySlotL
  3834.                     eqLeft = ""
  3835.                 elseif eqLeft == "minecraft:diamond_hoe" then
  3836.                     hoeSlot = emptySlotL
  3837.                     eqLeft = ""
  3838.                 elseif eqLeft == "minecraft:crafting_table" then
  3839.                     craftTableSlot = emptySlotL
  3840.                     eqLeft = ""
  3841.                 end
  3842.             end
  3843.         end
  3844.         if pickaxeSlot > 0 then
  3845.             turtle.select(pickaxeSlot)
  3846.             turtle.equipLeft()
  3847.             equippedLeft = "minecraft:diamond_pickaxe" 
  3848.             self.equippedLeft = "minecraft:diamond_pickaxe"
  3849.         end
  3850.         if craftTableSlot > 0 then
  3851.             turtle.select(craftTableSlot)
  3852.             turtle.equipRight()
  3853.             equippedRight = "minecraft:crafting_table"
  3854.             self.equippedRight = "minecraft:crafting_table"
  3855.         end
  3856.         inInventory = ""
  3857.         if swordSlot > 0 then
  3858.             inInventory = "minecraft:diamond_sword"
  3859.         elseif hoeSlot > 0 then
  3860.             inInventory = "minecraft:diamond_hoe"
  3861.         end
  3862.     end
  3863.     -- any tools equipped except diamond_pickaxe and crafting_table have been removed to inventory
  3864.     return equippedRight, equippedLeft, inInventory
  3865. end
  3866.    
  3867. function T:sortInventory(useChest)
  3868.     useChest = useChest or true
  3869.     local lib = {}
  3870.    
  3871.     function lib.checkForStorage(self)
  3872.         local blockType = self:getBlockType("forward")
  3873.         if blockType:find("barrel") ~= nil or blockType:find("chest") ~= nil then
  3874.             return "forward"
  3875.         end
  3876.         blockType = self:getBlockType("up")
  3877.         if blockType:find("barrel") ~= nil or blockType:find("chest") ~= nil then
  3878.             return "up"
  3879.         end
  3880.         blockType = self:getBlockType("down")
  3881.         if blockType:find("barrel") ~= nil or blockType:find("chest") ~= nil then
  3882.             return "down"
  3883.         end
  3884.         return ""
  3885.     end
  3886.    
  3887.     function lib.chestSort(self, chestDirection)
  3888.         for i = 1, 16 do    -- fill chest with everything
  3889.             self:drop(chestDirection, i)
  3890.         end
  3891.         local success, msg
  3892.         repeat
  3893.             success, msg = self:suck(chestDirection)
  3894.         until not success or message ~= nil
  3895.         --while self:suck(chestDirection) do end -- remove everything
  3896.     end
  3897.    
  3898.     local chestSlot = 0
  3899.     if useChest then
  3900.         chestSlot = self:getItemSlot("minecraft:chest") --get the slot number containing a chest
  3901.         if chestSlot == 0 then
  3902.             chestSlot = self:getItemSlot("minecraft:barrel") --get the slot number containing a barrel
  3903.         end
  3904.     end
  3905.     local blockType
  3906.     local facing = self.facing
  3907.     local chestPlaced = false
  3908.     local chestDirection, turns = "", 0
  3909.     if chestSlot > 0 then -- chest found
  3910.         local chestDirection, turns = self:getPlaceChestDirection() -- find empty block to place it.
  3911.         for i = 1, 10 do    -- attempt to place chest 10x
  3912.             if self:place("minecraft:chest", chestDirection) then
  3913.                 chestPlaced = true
  3914.                 break
  3915.             elseif self:place("minecraft:barrel", chestDirection) then
  3916.                 chestPlaced = true
  3917.                 break
  3918.             else
  3919.                 self:attack() -- will force wait for mob
  3920.             end
  3921.         end
  3922.         if chestPlaced then
  3923.             lib.chestSort(self, chestDirection)
  3924.             if chestDirection == "forward" then
  3925.                 turtle.dig()
  3926.             elseif chestDirection == "up" then
  3927.                 turtle.digUp()
  3928.             else
  3929.                 turtle.digDown()
  3930.             end
  3931.         end
  3932.         if turns > 0 then   --return to original position
  3933.             self:turnLeft(turns)
  3934.         end
  3935.     else
  3936.         if useChest then -- false eg in mining, where storage chest may be full, and essential items not retrieved
  3937.             chestDirection = lib.checkForStorage(self)
  3938.             if chestDirection ~= "" then
  3939.                 chestPlaced = true
  3940.                 lib.chestSort(self, chestDirection)
  3941.                 self:dig(chestDirection, false) -- collect chest (false = do not bypass chest)
  3942.             end
  3943.         end
  3944.     end
  3945.     if not chestPlaced then -- no chest or unable to place it
  3946.         for x = 1, 15 do -- sort inventory
  3947.             for i = x + 1 , 16 do
  3948.                 if turtle.getItemCount(i) > 0 then
  3949.                     turtle.select(i)
  3950.                     if turtle.compareTo(x) then
  3951.                         turtle.transferTo(x)
  3952.                     end
  3953.                 end
  3954.             end
  3955.         end
  3956.         for  x = 1, 15 do
  3957.             if turtle.getItemCount(x) == 0 then -- slot empty, move from next available filled slot
  3958.                 for i = x + 1, 16 do
  3959.                     if turtle.getItemCount(i) > 0 then
  3960.                         turtle.select(i)
  3961.                         turtle.transferTo(x)
  3962.                         break
  3963.                     end
  3964.                 end
  3965.             end
  3966.         end
  3967.     end
  3968.     turtle.select(1)
  3969. end
  3970.  
  3971. function T:split(textToSplit, sSeparator)
  3972.     return textToSplit:split(sSeparator)
  3973. end
  3974.  
  3975. function T:suck(direction, slot, quantity)
  3976.     direction = direction or "forward"
  3977.     slot = slot or 1
  3978.     --[[success, msg = turtle.suck()
  3979.         success = true / false
  3980.         msg = "No space for items" / nil]]
  3981.    
  3982.     turtle.select(slot)
  3983.     local Suck = turtle.suck
  3984.     if direction == "up" then
  3985.         Suck = turtle.suckUp
  3986.     elseif direction == "down" then
  3987.         Suck = turtle.suckDown
  3988.     end
  3989.     local success, msg
  3990.     if quantity == nil then
  3991.         success, msg = Suck()
  3992.     else
  3993.         success, msg = Suck(quantity)
  3994.     end
  3995.     turtle.select(1)
  3996.     return success, msg
  3997. end
  3998.  
  3999. function T:trimItemName(item)
  4000.     itemName = item
  4001.     colonPos = item:find(":")
  4002.     if colonPos ~= nil then
  4003.         itemName = item:sub(colonPos + 1)
  4004.     end
  4005.    
  4006.     return itemName
  4007. end
  4008.  
  4009. function T:writeCoords(filename)
  4010.     -- create/append e.g 'SpawnerCoords.txt'
  4011.     local fileHandle = fs.open(filename, "a")
  4012.     fileHandle.writeLine("x="..self.x)
  4013.     fileHandle.writeLine("y="..self.y)
  4014.     fileHandle.writeLine("z="..self.z)
  4015.     fileHandle.writeLine("f="..self.facing)
  4016.     fileHandle.close()
  4017.     self:saveToLog(filename.." file created", true)
  4018.     self:saveToLog("x = "..self:getX()..", y = "..self:getY()..", z = "..self:getZ()..", f = "..self:getFacing(), false)
  4019. end
  4020.  
  4021. return T
Add Comment
Please, Sign In to add comment