Advertisement
Lion4ever

Turtle Simulation on Computer

May 31st, 2015
741
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
Lua 12.43 KB | None | 0 0
  1. --block={"item",count,1/chance}
  2. local drops = {stone={"cobblestone"},coal_ore={"coal"},redstone_ore={"redstone",4,2,"redstone",5},grass={"dirt"},diamond_ore={"diamond"}}
  3. --{"block",minDepth,1/chance}
  4. local mapGen = {{"iron_ore",4,60},{"coal_ore",5,40},{"gold_ore",25,90},{"redstone_ore",40,70},{"diamond_ore",45,200}}
  5. --{"result",{"item",firstslot,secondslot},{"nextitem",slot}}
  6. local craftings = {{"chest",{"planks",1,2,3,5,7,9,10,11}},{"furnace",{"cobblestone",1,2,3,5,7,9,10,11}},{"planks",{"log",1}}}
  7. local colorMap = {stone=colors.gray,dirt=colors.brown,[false]=colors.black,grass=colors.green,coal_ore=colors.brown,bedrock=colors.lime,cobblestone=colors.gray,iron_ore=colors.orange,gold_ore=colors.yellow,diamond_ore=colors.lightBlue,redstone_ore=colors.red}
  8. local containers = {chest=27,dispenser=9,dropper=9,hopper=5}
  9. local fuels = {coal=80,planks=15,log=15,sticks=5}
  10. local env = {shell=shell,multishell=multishell}
  11.  
  12. local args = {...}
  13. env.turtle = {}
  14. local t = env.turtle
  15. local mu
  16. if multishell then
  17. mu = {launch=function(pEnv, ...) return multishell.launch(pEnv,"rom/programs/shell",...) end}
  18. setmetatable(mu,{__index=multishell})
  19. end
  20. local isUsedBy = 0
  21. local s
  22. s = {run=function(wEnv, ...)
  23. if not wEnv.turtle then
  24. wEnv.turtle = t
  25. wEnv.os = s
  26. wEnv.multishell = mu
  27. end
  28. isUsedBy = isUsedBy + 1
  29. _G.os.run(wEnv, ...)
  30. isUsedBy = isUsedBy - 1
  31. if isUsedBy == 0 then
  32. os.queueEvent("simulator_exit")
  33. end
  34. end}
  35. setmetatable(s,{__index=os})
  36. env.os = s
  37. env.multishell=mu
  38. local ti = {fx=1,fz=0,x=0,y=0,z=0,fuel=1000,fuelMax=20000,inv={},slot=1,eL="minecraft:diamond_pickaxe",eR="minecraft:crafting_table",delay=0.4}
  39.  
  40. local map = {}
  41. local function setM(v,x,y,z)
  42.  if not map[x] then
  43.    map[x]={}
  44.  end
  45.  if not map[x][y] then
  46.    map[x][y]={}
  47.  end
  48.  map[x][y][z]=v
  49. end
  50. local function getM(x,y,z,gen)
  51.   if not map[x] or not map[x][y] or map[x][y][z] == nil then
  52.     if gen ~= false then
  53.       local b = "stone"
  54.       if y >= 0  or y < -60 then
  55.         b = false
  56.       elseif y == -1 then
  57.         b = "grass"
  58.       elseif y > -5 then
  59.         b = "dirt"
  60.       elseif y == -60 then
  61.         b = "bedrock"
  62.       else
  63.         for i,j in ipairs(mapGen) do
  64.           if y < -j[2] then
  65.             if math.random(j[3]) == 1 then
  66.               b = j[1]
  67.               break
  68.             end
  69.           else
  70.             break
  71.           end
  72.         end
  73.       end
  74.       setM(b,x,y,z)
  75.       return b
  76.     end
  77.     return nil
  78.   else
  79.     return map[x][y][z]
  80.   end
  81. end  
  82. local x,y= term.getSize()
  83. local mX,mY = 5,5
  84. local oT = term.current()
  85. local mW = window.create(oT,1,1,x-mX-8,y)
  86. term.redirect(mW)
  87.  
  88. local tOrient = {">","^","B",[-3]="F",[-2]="V",[-1]="<"}
  89.  
  90. local function drawMap(px,py,sx,sy,dim,off,static)
  91.   local lx1,lx2,ly,lz,ix,iy
  92.   if static then
  93.     lx1,lx2,ly,lz = ti.x-(ti.x-sy)%sy+sy,ti.x-ti.x%sx,ti.y-(ti.y-sy)%sy+sy,ti.z-ti.z%sx
  94.     if dim == 1 then
  95.       ix,iy = ti.z-lz,(ly-ti.y)%sy
  96.     elseif dim == 2 then
  97.       ix,iy = ti.z-lz,(lx1-ti.x)%sy
  98.     else
  99.       ix,iy = ti.x-lx2,(ly-ti.y)%sy
  100.     end
  101.   else
  102.     lx1,lx2,ly,lz = ti.x+sy/2-0.5,ti.x-sx/2+0.5,ti.y+sy/2-0.5,ti.z-sx/2+0.5
  103.     ix,iy = sx/2-0.5,sy/2-0.5
  104.   end
  105.   local o=off or 0
  106.   for i=0,sy-1 do
  107.     oT.setCursorPos(px,py+i)
  108.     for j=0,sx-1 do
  109.       if (i==iy) and (j==ix) then
  110.         oT.setBackgroundColor(colors.lightGray)
  111.         oT.write(tOrient[dim==1 and ti.fx*3+ti.fz or dim==2 and ti.fx*2+ti.fz or ti.fx-3*ti.fz])
  112.       else
  113.         if dim == 1 then
  114.           oT.setBackgroundColor(colorMap[getM(ti.x+o,ly-i,lz+j)] or colors.magenta)
  115.         elseif dim == 2 then
  116.           oT.setBackgroundColor(colorMap[getM(lx1-i,ti.y+o,lz+j)] or colors.magenta)
  117.         else
  118.           oT.setBackgroundColor(colorMap[getM(lx2+j,ly-i,ti.z+o)] or colors.magenta)
  119.         end
  120.         oT.write(" ")
  121.       end
  122.     end
  123.   end
  124.   oT.setBackgroundColor(colors.black)
  125. end
  126.  
  127. local function drawInv(px,py,maxw,start)
  128.   for i=start or 1,math.min(16,y+(start or 1)-1) do
  129.     oT.setCursorPos(px,py+i-1)
  130.     if ti.inv[i] then
  131.       oT.write(string.format("%2d %s           ",ti.inv[i].count,ti.inv[i].name:match(":(.*)")):sub(1,maxw))
  132.     else
  133.       oT.write("   Empty")
  134.     end
  135.   end
  136. end
  137. local drawCounter = 0
  138. local function updateMap()
  139.   if drawCounter == 0 then
  140.   drawMap(x-mX+1,1,mX,mY,2,0)
  141.   drawMap(x-mX+1,7,mX,mY,1,0)
  142.   drawMap(x-mX+1,13,mX,mY,3,0)
  143.   drawInv(x-mX-7,1,8)
  144.   oT.setCursorPos(x-mX+1,y-1)
  145.   oT.write(table.concat({ti.x,ti.y}," ").."  ")
  146.   oT.setCursorPos(x-mX+1,y)
  147.   oT.write(tostring(ti.z).." ")
  148.   mW.restoreCursor()
  149.   end
  150. end
  151.  
  152. local mutex = false
  153. local isWaiting = false
  154. local function wait()
  155.   if ti.delay >= 0 then
  156.     while mutex do
  157.       isWaiting = true
  158.       os.pullEvent("turtle")
  159.     end
  160.     mutex = true
  161.     local te,err = pcall(sleep,ti.delay) --because of terminate
  162.     mutex = false
  163.     if isWaiting then
  164.       isWaiting = false
  165.       os.queueEvent("turtle")
  166.     end
  167.     if not te then
  168.       error(err,0)
  169.     end
  170.   else
  171.     drawCounter = drawCounter - 1
  172.     if drawCounter <= ti.delay then
  173.       os.queueEvent("turtle")
  174.       os.pullEvent("turtle")
  175.       drawCounter = 0
  176.     end
  177.   end
  178. end
  179.  
  180. t.getSelectedSlot = function() return ti.slot end
  181. t.select = function(s) if type(s) == "number" and s>=1 and s<=16 then ti.slot = math.floor(s) end end
  182. t.getFuelLimit = function() return ti.fuelMax end
  183. t.getFuelLevel = function() return ti.fuel end
  184. t.getItemCount = function(s) return ti.inv[s or ti.slot] and ti.inv[s or ti.slot].count or 0 end
  185. t.getItemSpace = function(s) return ti.inv[s or ti.slot] and 64-ti.inv[s or ti.slot].count or 64 end
  186. t.getItemDetail = function(s) return ti.inv[s or ti.slot] end
  187.  
  188. t.refuel = function(n)
  189. if not ti.inv[ti.slot] then
  190. return false,"No Items to refuel"
  191. end
  192. if not fuels[ti.inv[ti.slot].name:match(":(.*)")] then
  193. return false,"Item isn't burnable"
  194. end
  195. if ti.fuel == "unlimited" then
  196. return false,"Unnesseary"
  197. end
  198. n=n or 64
  199. if type(n) ~= "number" or math.floor(n) ~= n or n < 0 then
  200. return false,"Invalid Parameter"
  201. end
  202. local c = math.min(n,ti.inv[ti.slot].count)
  203. ti.fuel=ti.fuel + fuels[ti.inv[ti.slot].name:match(":(.*)")] * c
  204. ti.inv[ti.slot].count = ti.inv[ti.slot].count - c
  205. if ti.inv[ti.slot].count == 0 then
  206. ti.inv[ti.slot] = nil
  207. end
  208. return true
  209. end
  210.  
  211. local function p(f,u)
  212.   return ti.x+ti.fx*f,ti.y+u,ti.z+ti.fz*f
  213. end
  214.  
  215. t.turnLeft = function()
  216.   wait()
  217.   ti.fx,ti.fz=ti.fz,-ti.fx
  218.   updateMap()
  219.   return true
  220. end
  221.  
  222. t.turnRight = function()
  223.   wait()
  224.   ti.fx,ti.fz=-ti.fz,ti.fx
  225.   updateMap()
  226.   return true
  227. end
  228.  
  229. local function move(f,u)
  230.   return function()
  231.     wait()
  232.     if ti.fuel == 0 then
  233.       return false,"No fuel"
  234.     end
  235.     if getM(p(f,u)) then
  236.       return false,"Path obstruced"
  237.     end
  238.     ti.x,ti.y,ti.z = p(f,u)
  239.     if ti.fuel ~= "unlimited" then
  240.       ti.fuel=ti.fuel-1
  241.     end
  242.     updateMap()
  243.     return true
  244.   end
  245. end
  246. t.back = move(-1,0)
  247. t.forward = move(1,0)
  248. t.up = move(0,1)
  249. t.down = move(0,-1)
  250.  
  251. local function det(f,u)
  252.   return function()
  253.     --wait()
  254.     return getM(p(f,u)) ~= false
  255.   end
  256. end
  257.  
  258. local function addToInv(item,startWithSlot,slotCount)
  259.   local s= startWithSlot or ti.slot
  260.   for i=1,slotCount or 16 do
  261.     if not ti.inv[s] then
  262.       ti.inv[s] = item
  263.       return true
  264.     end
  265.     if ti.inv[s].damage == item.damage and ti.inv[s].name == item.name then
  266.       if t.getItemSpace(s) - item.count < 0 then
  267.         item.count = item.count - t.getItemSpace(s)
  268.         ti.inv[s].count = ti.inv[s].count + t.getItemSpace(s)
  269.       else
  270.         ti.inv[s].count = ti.inv[s].count + item.count
  271.         return true
  272.       end
  273.     end
  274.     s=s%16+1
  275.   end
  276.   return false
  277. end
  278.  
  279. t.transferTo = function(s,n)
  280.   if s == ti.slot then
  281.     return false,"Same Slot"
  282.   end
  283.   if not ti.inv[ti.slot] then
  284.     return false,"No items to transfer"
  285.   end
  286.   if t.getItemSpace(s) == 0 then
  287.     return false,"Target Slot is full"
  288.   end
  289.   n = n or 64
  290.   s = s or -1 --Invalid
  291.   if s < 1 or s > 16 or n < 1 or math.floor(s) ~= s or math.floor(n) ~= n then
  292.     return false,"Invalid Parameter"
  293.   end
  294.   if not ti.inv[s] then
  295.     ti.inv[s] = {name=ti.inv[ti.slot].name,damage=ti.inv[ti.slot].damage,count=0}
  296.   end
  297.   if ti.inv[ti.slot].name ~= ti.inv[s].name or ti.inv[ti.slot].damage ~= ti.inv[s].damage then
  298.     return false,"Not same items"
  299.   end
  300.   local tcount = math.min(n,ti.inv[ti.slot].count,t.getItemSpace(s))
  301.   ti.inv[ti.slot].count = ti.inv[ti.slot].count - tcount
  302.   ti.inv[s].count = ti.inv[s].count + tcount
  303.   if ti.inv[ti.slot].count == 0 then
  304.     ti.inv[ti.slot] = nil
  305.   end
  306.   updateMap()
  307.   return true
  308. end
  309.  
  310. local function dg(f,u)
  311.   return function()
  312.     wait()
  313.     local b = getM(p(f,u))
  314.     if b == false then
  315.       return false,"No block"
  316.     end
  317.     if b == "bedrock" then
  318.       return false,"Cant break bedrock"
  319.     end
  320.     local i = {count=1,damage=0,name="minecraft:"..b}
  321.     if drops[b] then
  322.       local alts = 1
  323.       repeat
  324.         i.name="minecraft:"..drops[b][alts]
  325.         i.count = drops[b][alts+1] or i.count
  326.         alts = alts + 3
  327.       until not drops[b][alts-1] or math.random(drops[b][alts-1]) == 1
  328.     end
  329.     addToInv(i)
  330.     setM(false,p(f,u))
  331.     updateMap()
  332.     return true
  333.   end
  334. end
  335.  
  336. local function ins(f,u)
  337.   return function()
  338.     wait()
  339.     local b=getM(p(f,u))
  340.     if b~=false then
  341.       return true,"minecraft:"..b
  342.     end
  343.     return false
  344.   end
  345. end
  346.  
  347. local function plc(f,u)
  348.   return function()
  349.     wait()
  350.     if getM(p(f,u)) then
  351.       return false,"There is allready a block"
  352.     end
  353.     if not ti.inv[ti.slot] then
  354.       return false,"No block selected"
  355.     end
  356.     ti.inv[ti.slot].count = ti.inv[ti.slot].count-1
  357.     local n=ti.inv[ti.slot].name
  358.     if ti.inv[ti.slot].count == 0 then
  359.       ti.inv[ti.slot] = nil
  360.     end
  361.     setM(n:match("minecraft:(.*)"),p(f,u))
  362.     updateMap()
  363.     return true
  364.   end
  365. end
  366.  
  367. local function drp(s)
  368.   if not ti.inv[ti.slot] then
  369.     return false,"Slot empty"
  370.   end
  371.   s=tonumber(s)
  372.   if s and ti.inv[ti.slot].count>s then
  373.     ti.inv[ti.slot].count = ti.inv[ti.slot].count - s
  374.     return true
  375.   end
  376.   ti.inv[ti.slot] = nil
  377.   updateMap()
  378.   return true
  379. end
  380. local function drpa()
  381.  return drp
  382. end
  383.  
  384. local allDirs = {dig=dg,inspect=ins,detect=det,place=plc,drop=drpa}
  385. for i,j in pairs(allDirs) do
  386.   t[i] = j(1,0)
  387.   t[i.."Up"] = j(0,1)
  388.   t[i.."Down"] = j(0,-1)
  389. end
  390.  
  391. local randB = function()
  392. wait()
  393. return math.random(2) == 1
  394. end
  395. setmetatable(t,{__index=function() return randB end})
  396.  
  397. --local args = {...}
  398. local resultTurtle
  399. local resultMap
  400. local result
  401. local options = {
  402. delay=function(s) ti.delay = tonumber(s) end,
  403. nodelay=function() ti.delay = -1 return true end,
  404. lconf=function(s) ti = dofile(s) or ti end,
  405. sconf=function(s) resultTurtle=s end,
  406. map=function(s) map = dofile(s) or map end,
  407. smap=function(s) resultMap=s end,
  408. load=function(s) local a,b = dofile(s) ti=a or ti map=b or map end,
  409. saveas=function(s) result = s end,
  410. fuel=function(s) ti.fuel = tonumber(s) or ti.fuel end,
  411. nofuel=function() ti.fuel = "unlimited" return true end,
  412. bigmap=function() ti.useMulti = true return true end}
  413. options.save=options.saveas
  414. options.newtab=options.bigmap
  415. options.use = function(s) if fs.exists(s) then options.load(s) end options.save(s) end
  416. options.conf = function(s) if fs.exists(s) then options.lconf(s) end options.sconf(s) end
  417.  
  418. local argsP = 1
  419. while options[string.lower(args[argsP] or "")] do
  420. argsP = argsP + (options[args[argsP]](args[argsP+1]) and 1 or 2)
  421. end
  422.  
  423. local oPath = shell.path()
  424. if not string.match(oPath,"rom/programs/turtle") then
  425.   shell.setPath(oPath..":rom/programs/turtle")
  426. else
  427.   oPath = nil
  428. end
  429. if ti.useMulti and multishell then
  430.   updateMap = function() drawMap(1,1,y,y,1) end
  431.   os.run(env,"rom/programs/shell","fg",unpack(args,argsP))
  432.   mW.setVisible(false)
  433.   multishell.getTitle(multishell.getCurrent(),"Map")
  434. else
  435.   updateMap()
  436.   os.run(env,"rom/programs/shell",unpack(args,argsP))
  437.   if isUsedBy > 0 then
  438.     print("Waiting for remaining programs to exit")
  439.   end
  440. end
  441. while isUsedBy > 0 do
  442.   os.pullEvent("simulator_exit")
  443. end
  444. if oPath then
  445.   shell.setPath(oPath)
  446. end
  447. if result then
  448. local f = fs.open(result,"w")
  449. f.write("return ")
  450. f.write(textutils.serialize(ti))
  451. f.write(",")
  452. f.write(textutils.serialize(map))
  453. f.close()
  454. end
  455.  
  456. if resultTurtle then
  457. local f = fs.open(resultTurtle,"w")
  458. f.write("return ")
  459. f.write(textutils.serialize(ti))
  460. f.close()
  461. end
  462.  
  463. if resultMap then
  464. local f = fs.open(resultMap,"w")
  465. f.write("return ")
  466. f.write(textutils.serialize(map))
  467. f.close()
  468. end
  469.  
  470. oT.clear()
  471. mW.redraw()
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement